#ifndef PROPERTIES_H
#define PROPERTIES_H

#include "z3++.h"

class SMVerifier;

using PropertyFn = std::function<z3::expr(z3::expr_vector startStates, z3::expr_vector endStates,
                                          std::vector<z3::expr_vector> allDecisions,
                                          std::vector<std::vector<z3::expr_vector>> allMessages, SMVerifier& ver)>;

using ProtocolConstraintFn = std::function<z3::expr(
    std::vector<z3::expr_vector> allDecisions, std::vector<std::vector<z3::expr_vector>> allMessages, SMVerifier& ver)>;

z3::expr distributed_locking(z3::expr_vector start_states, z3::expr_vector end_states,
                             std::vector<z3::expr_vector> all_decisions,
                             std::vector<std::vector<z3::expr_vector>> all_messages, SMVerifier& ver);

z3::expr atomic_commit(z3::expr_vector start_states, z3::expr_vector end_states,
                       std::vector<z3::expr_vector> all_decisions,
                       std::vector<std::vector<z3::expr_vector>> all_messages, SMVerifier& ver);

z3::expr atomic_commit_protocol_constraints(std::vector<z3::expr_vector> all_decisions,
                                            std::vector<std::vector<z3::expr_vector>> all_messages, SMVerifier& ver);

z3::expr primary_backup(z3::expr_vector start_states, z3::expr_vector end_states,
                        std::vector<z3::expr_vector> all_decisions,
                        std::vector<std::vector<z3::expr_vector>> all_messages, SMVerifier& ver);

z3::expr primary_backup_protocol_constraints(std::vector<z3::expr_vector> all_decisions,
                                             std::vector<std::vector<z3::expr_vector>> all_messages, SMVerifier& ver);

std::vector<z3::expr_vector> node_crash(std::vector<std::vector<z3::expr_vector>> all_messages, int num_rounds,
                                        int num_nodes, z3::context& context, SMVerifier& ver);

z3::expr_vector reach_state_with_failure(std::vector<z3::expr_vector> all_decisions,
                                         std::vector<z3::expr_vector> crash_info, z3::expr target, int num_nodes,
                                         z3::context& context);

// Return an expression that represents any TARGET state exists in the expr_vector
z3::expr any_state(z3::expr_vector states, z3::expr target, z3::context& context);

// Return an expression that represents all states in the expr_vector are TARGET state
z3::expr all_state(z3::expr_vector states, z3::expr target, z3::context& context);

// Return an expr_vector that contains whether each node reaches the TARGET state(action)
z3::expr_vector reach_state(std::vector<z3::expr_vector> all_decisions, z3::expr target, int num_nodes,
                            z3::context& context);

#endif