#include <cmath>
#include <map>
#include <list>
#include <bitset>
#include <queue>
#include <iostream>
#include <string.h>
#include "DetectAmbiguity.h"

namespace solverbin{
  DetectABTNFA::DetectABTNFA(REnodeClass r){
    e1 = r;
    F1 = RegExpSymbolic::PositionAutomaton(e1);
    SSBegin = new TernarySimulationState(Begin, F1.NState, F1.NState, F1.NState);
    e1.ComputeAlphabet(e1.ByteMap, Alphabet);
    e1.BuildBytemapToString(e1.ByteMap);
    RegExpSymbolic::DumpAlphabet(Alphabet);
  }

  void DetectABTNFA::DumpTernarySimulationState(TernarySimulationState* TSS){
    std::cout << "SimulationState: " << TSS << " IFlag: " << TSS->IFlag << " IsIntersect: " << TSS->IsSat << 
    " IsDone" << TSS->IsDone << std::endl;
    std::cout << TSS->NS1->Node2Continuation.first << ": continuation" << e1.REnodeToString(TSS->NS1->Node2Continuation.second) << std::endl;
    std::cout << TSS->NS2->NS1->Node2Continuation.first << ": continuation" << e1.REnodeToString(TSS->NS2->NS1->Node2Continuation.second) << std::endl;
    std::cout << TSS->NS2->NS2->Node2Continuation.first << ": continuation" << e1.REnodeToString(TSS->NS2->NS2->Node2Continuation.second) << std::endl;
    F1.DumpState(TSS->NS1);
    F1.DumpState(TSS->NS2->NS1);
    F1.DumpState(TSS->NS2->NS2);
  }

  std::set<DetectABTNFA::TernarySimulationState> DetectABTNFA::DTSimulationState(TernarySimulationState* TS){
    std::set<DetectABTNFA::TernarySimulationState> TSSSet;
    if (TS->NS1->Node2Continuation.first == TS->NS2->NS1->Node2Continuation.first && TS->NS1->Node2Continuation.first != TS->NS2->NS2->Node2Continuation.first){
      TSSSet.insert(TernarySimulationState(Normal, TS->NS1, TS->NS2->NS2, TS->NS2->NS2));
      TSSSet.insert(TernarySimulationState(Normal, TS->NS2->NS2, TS->NS1, TS->NS2->NS2));
    }
    else if (TS->NS1->Node2Continuation.first == TS->NS2->NS2->Node2Continuation.first && TS->NS1->Node2Continuation.first != TS->NS2->NS1->Node2Continuation.first){
      TSSSet.insert(TernarySimulationState(Normal, TS->NS1, TS->NS2->NS1, TS->NS2->NS1));
      TSSSet.insert(TernarySimulationState(Normal, TS->NS2->NS1, TS->NS2->NS1, TS->NS1));
    }
    else if (TS->NS2->NS1->Node2Continuation.first == TS->NS2->NS2->Node2Continuation.first && TS->NS1->Node2Continuation.first != TS->NS2->NS1->Node2Continuation.first){
      TSSSet.insert(TernarySimulationState(Normal, TS->NS1, TS->NS1, TS->NS2->NS1));
      TSSSet.insert(TernarySimulationState(Normal, TS->NS1, TS->NS2->NS1, TS->NS1));
    }
    return TSSSet;
  }

  bool DetectABTNFA::DetectABTOFS(TernarySimulationState* TSS, std::set<TernarySimulationState> TSSET){
    std::cout << "witness str: " << WitnessStr << std::endl;
    // DumpTernarySimulationState(TSS);
    for (auto c : Alphabet){
      std::cout << "matching: " << int(c) << " " << std::endl;
      // if (TSS->byte2state.find(ByteMap[c]) == TSS->byte2state.end()){
        std::set<TernarySimulationState*> TernarySimulationSet;
        auto nextns1 = F1.StepOneByte(TSS->NS1, c);
        auto nextns2 = F1.StepOneByte(TSS->NS2->NS1, c);
        auto nextns3 = F1.StepOneByte(TSS->NS2->NS2, c);
        if (nextns1.empty() || nextns2.empty() || nextns3.empty())
          continue;
        for (auto nextns1_it : nextns1){
          for (auto nextns2_it : nextns2){
            for (auto nextns3_it : nextns3){
              auto ns = new TernarySimulationState(Normal, nextns1_it, nextns2_it, nextns3_it);
              DumpTernarySimulationState(ns);
              auto itc = SimulationCache.find(*ns);
              if (itc != SimulationCache.end()){
                TernarySimulationSet.insert(itc->second);
              }
              else{
                SimulationCache.insert(std::make_pair(*ns, ns));
                WitnessStr.push_back(c);
                if (TSSET.find(*ns) != TSSET.end()){
                  return true;
                }
                if (DetectABTOFS(ns, TSSET)){
                  return true;
                }
                else
                  WitnessStr.pop_back();
              }
            }
          }
        }
        TSS->byte2state.insert(std::make_pair(ByteMap[c], TernarySimulationSet));  
    }
    return false;
  }

  bool DetectABTNFA::IsABT(TernarySimulationState* TSS){
    std::cout << "witness str: " << InterStr << std::endl;
    DumpTernarySimulationState(TSS);
    for (auto c : Alphabet){
      std::cout << "matching: " << int(c) << " " << std::endl;
      auto nextns1 = F1.StepOneByte(TSS->NS1, c);
      auto nextns2 = F1.StepOneByte(TSS->NS2->NS1, c);
      auto nextns3 = F1.StepOneByte(TSS->NS2->NS2, c);
      if (nextns1.empty() || nextns2.empty() || nextns3.empty())
        continue;
      for (auto nextns1_it : nextns1){
        for (auto nextns2_it : nextns2){
          for (auto nextns3_it : nextns3){
            auto ns = new TernarySimulationState(Normal, nextns1_it, nextns2_it, nextns3_it);
            DumpTernarySimulationState(ns);
            auto itc = DoneCache.find(*ns);
            if (itc != DoneCache.end()){
              continue;
            }
            else{
              DoneCache.insert(std::make_pair(*ns, ns));
              auto TSSET = DTSimulationState(ns);
              InterStr.push_back(c);
              if (!TSSET.empty()){
                SimulationCache.insert(std::make_pair(*ns, ns));
                DumpTernarySimulationState(ns);
                if (DetectABTOFS(ns, TSSET)){
                  InterStr = InterStr + WitnessStr;
                  return true;
                }
                else {
                  SimulationCache.clear();
                  WitnessStr = "";
                }
              }
              if (IsABT(ns)){
                return true;
              }
              else {
                InterStr.pop_back();
              }
            }
          }
        }
      }
    }
    return false;
  }
  
}