/*
This is adapted from contents in Dominique Unruh's package scala-isabelle
The package can be found on github at https://github.com/dominique-unruh/scala-isabelle
This particular file is adapted from https://github.com/dominique-unruh/scala-isabelle/blob/master/src/test/scala/de/unruh/isabelle/experiments/ExecuteIsar.scala
 */

package pisa.server

import io.grpc.Status
import zio.{ZEnv, ZIO}
import pisa.server.ZioServer.ZServer
import de.unruh.isabelle.pure.{Theory, ToplevelState}
import de.unruh.isabelle.control.IsabelleMLException
import de.unruh.isabelle.mlvalue.MLValue
import de.unruh.isabelle.control.Isabelle
import de.unruh.isabelle.pure.Implicits._
import de.unruh.isabelle.mlvalue.Implicits._

import scala.concurrent.{ExecutionContext, TimeoutException}
import java.io.PrintWriter
import scala.util.control.Breaks

class OneStageBody extends ZServer[ZEnv, Any] {
  var pisaos: PisaOS = null
  var stand_in_thy: MLValue[Theory] = null
  var stand_in_tls: MLValue[ToplevelState] = null
  var isaPath: String = null
  var isaWorkingDirectory: String = null

  def initialiseIsabelle(
      isa_path: IsaPath
  ): ZIO[zio.ZEnv, Status, IsaMessage] = {
    isaPath = isa_path.path
    ZIO.succeed(
      IsaMessage(
        s"----------Path to Isabelle source----------\n${isa_path.path}"
      )
    )
  }

  def isabelleWorkingDirectory(
      isa_working_directory: IsaPath
  ): zio.ZIO[zio.ZEnv, Status, IsaMessage] = {
    isaWorkingDirectory = isa_working_directory.path
    ZIO.succeed(
      IsaMessage(
        s"----------Path to Isabelle working directory----------\n${isaWorkingDirectory}"
      )
    )
  }

  def isabelleContext(
      path_to_file: IsaContext
  ): ZIO[zio.ZEnv, Status, IsaMessage] = {
    try {
    pisaos = new PisaOS(
      path_to_isa_bin = isaPath,
      path_to_file = path_to_file.context,
      working_directory = isaWorkingDirectory,
      debug = true
    )
    } catch {
      case e: Throwable => {
        println("An error ocurred in PisaOS initialization: " + e.getMessage())
        println("Stacktrace: " + e.printStackTrace())
        return ZIO.succeed(IsaMessage("Failed to initialize with error: " + e.getMessage()))
      }
    }
    stand_in_thy = pisaos.thy1.mlValue
    stand_in_tls = pisaos.copy_tls
    ZIO.succeed(
      IsaMessage(
        s"----------Path to Isabelle theory file----------\n${path_to_file.context}"
      )
    )
  }

  def reset_problem(): Unit = {
    implicit val isabelle: Isabelle = pisaos.isabelle
    implicit val ec: ExecutionContext = pisaos.ec
    pisaos.thy1 = stand_in_thy.force.retrieveNow
    pisaos.toplevel = stand_in_tls.force.retrieveNow
    pisaos.reset_map()
    pisaos.register_tls("default", pisaos.toplevel)
  }

  def deal_with_reset_problem(): String = {
    reset_problem()
    "The problem is reset."
  }

  def deal_with_extraction(): String = {
    try{
      pisaos.step("PISA extract data")
    } catch {
      case t: Throwable => {println(t); t.toString()}
    }
  }

  def deal_with_extraction_with_hammer(): String =
    pisaos.step("PISA extract data with hammer")

  def deal_with_list_states(): String =
    pisaos.top_level_state_map.keys.mkString(" | ")

  def deal_with_initialise(): String = {
    // print("Intialising the problem...")
    pisaos.top_level_state_map += ("default" -> pisaos.copy_tls)
    // println(" Intialised!")
    "Toplevel state 'default' is ready"
  }

  def deal_with_get_state(toplevel_state_name: String): String = {
    if (pisaos.top_level_state_map.contains(toplevel_state_name))
      pisaos.getStateString(pisaos.retrieve_tls(toplevel_state_name))
    else s"Didn't find top level state of given name: ${toplevel_state_name}"
  }

  def deal_with_is_finished(toplevel_state_name: String): String = {
    if (pisaos.top_level_state_map.contains(toplevel_state_name)) {
      if (pisaos.getProofLevel(pisaos.retrieve_tls(toplevel_state_name)) == 0)
        "true"
      else "false"
    } else s"Didn't find top level state of given name: ${toplevel_state_name}"
  }

  val TRY_STRING: String = "Try this:"
  val FOUND_PROOF_STRING: String = "found a proof:"
  val ERROR_MSG: String = "error"
  val SMT_HAMMER: String = "sledgehammer"
  val METIS_HAMMER: String = "metishammer"
  val NORMAL_HAMMER: String = "normalhammer"
  val DEL_HAMMER: String = "delhammer"
  val TIME_STRING1: String = " ms)"
  val TIME_STRING2: String = " s)"
  val ALLOW_MORE_TIME: String = "<allow more time>"

  def process_hammer_strings(hammer_string_list: List[String]): String = {
    var found = false
    for (attempt_string <- hammer_string_list) {
      if (!found && (attempt_string contains TRY_STRING)) {
        found = true
        val parsed = attempt_string.split(TRY_STRING).drop(1).mkString("").trim
        if ((parsed contains TIME_STRING1) || (parsed contains TIME_STRING2)) {
          return parsed.split('(').dropRight(1).mkString("(").trim
        }
        return parsed
      } else if (!found && (attempt_string contains FOUND_PROOF_STRING)) {
        found = true
        val parsed =
          attempt_string.split(FOUND_PROOF_STRING).drop(1).mkString("").trim
        if ((parsed contains TIME_STRING1) || (parsed contains TIME_STRING2)) {
          return parsed.split('(').dropRight(1).mkString("(").trim
        }
        return parsed
      }
    }
    ""
  }

  def hammer_actual_step(
      old_state: ToplevelState,
      new_name: String,
      hammer_method: (ToplevelState, Int) => (Boolean, List[String])
  ): String = {
    // If found a sledgehammer step, execute it differently
    var raw_hammer_strings = List[String]()
    val actual_step: String =
      try {
        val total_result = hammer_method(old_state, 40000)
        // println(total_result)
        val success = total_result._1
        if (success) {
          // println("Hammer string list: " + total_result._2.mkString(" ||| "))
          val tentative_step = process_hammer_strings(total_result._2)
          // println("actual_step: " + tentative_step)
          tentative_step
        } else {
          ERROR_MSG
        }
      } catch {
        case e: Exception => {
          println("Exception while trying to run sledgehammer: " + e.getMessage)
          e.getMessage
        }
      }
    // println(actual_step)
    assert(actual_step.trim.nonEmpty)
    actual_step
  }

  def deal_with_apply_to_tls(
      toplevel_state_name: String,
      action: String,
      new_name: String,
      tactic_time: Int = 10000,
  ): String = {
    if (pisaos.top_level_state_map.contains(toplevel_state_name)) {
      var actual_timeout = tactic_time
      val old_state: ToplevelState = pisaos.retrieve_tls(toplevel_state_name)
      var actual_step: String = "Gibberish"
      var hammered: Boolean = false

      if (action.startsWith(NORMAL_HAMMER)) {
        val additional_arguments_string: String =
          action.split(NORMAL_HAMMER).drop(1).mkString("").trim
        val add_names: List[String] = {
          if (additional_arguments_string contains "<add>") {
            additional_arguments_string.split("<add>")(1).split(",").toList
          } else List[String]()
        }
        val del_names: List[String] = {
          if (additional_arguments_string contains "<del>") {
            additional_arguments_string.split("<del>")(1).split(",").toList
          } else List[String]()
        }
        val partial_hammer = (state: ToplevelState, timeout: Int) =>
          pisaos.normal_with_hammer(state, add_names, del_names, timeout)
        actual_step = hammer_actual_step(old_state, new_name, partial_hammer)
        hammered = true
      } else if (action.startsWith(ALLOW_MORE_TIME)) {
        actual_step = action.split(ALLOW_MORE_TIME).drop(1).mkString("").trim
        actual_timeout = 30000
      } else {
        actual_step = action
      }
      // println("Actual step: " + actual_step)

      val new_state: ToplevelState =
        pisaos.step(actual_step, old_state, actual_timeout)
      val new_level: Int = pisaos.getProofLevel(new_state)
      var elaborated_string: String = ""
      if (new_level != 0) {
        elaborated_string = pisaos.getElaborateStateString(new_state)
      }else{
        elaborated_string = ""
      }
      // println("Application successful")
      // println("New state: " + pisaos.getStateString(new_state))

      pisaos.register_tls(name = new_name, tls = new_state)
      val state_string: String = if (hammered) {
        s"$actual_step <hammer> ${pisaos.getStateString(new_state)} <elaborated> ${elaborated_string}"
      } else s"${pisaos.getStateString(new_state)} <elaborated> ${elaborated_string}"
      return s"${state_string} <prooflevel> ${new_level}"
    } else s"Didn't find top level state of given name: ${toplevel_state_name}"
  }

  def deal_with_proof_level(toplevel_state_name: String): String = {
    if (pisaos.top_level_state_map.contains(toplevel_state_name)) {
      val tls: ToplevelState = pisaos.retrieve_tls(toplevel_state_name)
      s"${pisaos.getProofLevel(tls)}"
    } else s"Didn't find top level state of given name: ${toplevel_state_name}"
  }

  def deal_with_proceed_to_line(true_command: String, before_after: String, counter:Int): String = {
    val after = before_after == "after"
    pisaos.step_to_transition_text(true_command, after = after, counter=counter)
  }

  def deal_with_proceed_before(true_command: String): String =
    pisaos.step_to_transition_text(true_command, after = false)

  def deal_with_proceed_after(true_command: String): String =
    pisaos.step_to_transition_text(true_command, after = true)

  def deal_with_exit(command: String): String = {
    pisaos.step(command)
    pisaos = null
    "Exited"
  }

  def deal_with_clone(old_name: String, new_name: String): String = {
    pisaos.clone_tls(old_name, new_name)
    "Successfully copied top level state named: " + new_name
  }

  def deal_with_delete(toplevel_state_name: String): String = {
    if (pisaos.top_level_state_map.contains(toplevel_state_name)) {
      pisaos.top_level_state_map -= toplevel_state_name
      s"Successfully deleted state named: ${toplevel_state_name}"
    } else s"Didn't find top level state of given name: ${toplevel_state_name}"
  }

  def deal_with_local_facts_and_defs(toplevel_state_name: String): String = {
    if (pisaos.top_level_state_map.contains(toplevel_state_name)) {
      pisaos.local_facts_and_defs_string(toplevel_state_name)
    } else s"Didn't find top level state of given name: ${toplevel_state_name}"
  }

  def deal_with_global_facts_and_defs(toplevel_state_name: String): String = {
    if (pisaos.top_level_state_map.contains(toplevel_state_name)) {
      pisaos.global_facts_and_defs_string(toplevel_state_name)
    } else s"Didn't find top level state of given name: ${toplevel_state_name}"
  }

  def deal_with_total_facts_and_defs(toplevel_state_name: String): String = {
    if (pisaos.top_level_state_map.contains(toplevel_state_name)) {
      pisaos.total_facts_and_defs_string(toplevel_state_name)
    } else s"Didn't find top level state of given name: ${toplevel_state_name}"
  }

  def deal_with_get_all_defs(theorem_string: String): String = {
    val tls_name = "default"
    pisaos.get_all_definitions(tls_name, theorem_string).mkString("\n")
  }

  def deal_with_global_facts_from_file: String = {
    implicit val isabelle: Isabelle = pisaos.isabelle
    implicit val ec: ExecutionContext = pisaos.ec
    val continue = new Breaks
    val transition_and_index_list = pisaos
      .parse_text(pisaos.thy1, pisaos.fileContentCopy.trim)
      .force
      .retrieveNow
      .zipWithIndex
    for (((transition, text), i) <- transition_and_index_list) {
      continue.breakable {
        if (text.trim.isEmpty) continue.break
        else if (
          text.trim == "end" && (i == transition_and_index_list.length - 1)
        ) continue.break
        else {
          pisaos.singleTransition(transition)
        }
      }
    }
    pisaos.global_facts_and_defs_string(pisaos.toplevel)
  }

  def deal_with_parse_text(text: String): String = {
    implicit val isabelle: Isabelle = pisaos.isabelle
    implicit val ec: ExecutionContext = pisaos.ec
    val parsed = pisaos.parse_text(pisaos.thy1, text.trim).force.retrieveNow
    parsed.map(x => x._2).filter(_.trim.nonEmpty).mkString("<SEP>")
  }

  def deal_with_accummulative_step_before(text: String) = {
    pisaos.accumulative_step_to_before_transition_starting(text)
  }

  def deal_with_get_dependent_theorems(tls_name: String, theorem_name: String, separator: String) = {
    // println("Start dealing with dependent theorems")
    val dependent_theorem_list = pisaos.get_dependent_theorems(tls_name, theorem_name)
    dependent_theorem_list.mkString(separator)
  }

  def deal_with_fact_definition(tls_name: String, fact_name: String) = {
    pisaos.fact_definition(tls_name, fact_name)
  }

  def deal_with_parse_entire_thy : String = {
    val entire_thy_parsed = pisaos.parse_entire_thy
    entire_thy_parsed.filter(_.trim.nonEmpty).mkString("<SEP>")
  }

  def deal_with_accumulative_step_to_theorem_end(theorem_name: String) = {
    pisaos.accumulative_step_to_theorem_end(theorem_name)
    "<SUCCESS>"
  }

  def deal_with_accumulative_step_before_theorem_starts(theorem_name: String) = {
    pisaos.accumulative_step_before_theorem_starts(theorem_name)
    "<SUCCESS>"
  }

  def deal_with_accumulative_step_through_a_theorem = {
    pisaos.accumulative_step_through_a_theorem
    "<SUCCESS>"
  }

  def isabelleCommand(
      isa_command: IsaCommand
  ): ZIO[zio.ZEnv, Status, IsaState] = {
    val proof_state: String = {
      if (isa_command.command.trim == "PISA extract data")
        deal_with_extraction()
      else if (isa_command.command.trim == "PISA extract data with hammer")
        deal_with_extraction_with_hammer()
      else if (isa_command.command.startsWith("<accumulative step before>")) {
        val text = isa_command.command.stripPrefix("<accumulative step before>")
        deal_with_accummulative_step_before(text)
      } else if (isa_command.command.trim.startsWith("<parse text>")) {
        val text = isa_command.command.trim.stripPrefix("<parse text>")
        deal_with_parse_text(text)
      } else if (isa_command.command.trim.startsWith("<get all definitions>")) {
        val theorem_string: String =
          isa_command.command.stripPrefix("<get all definitions>").trim
        deal_with_get_all_defs(theorem_string)
      } else if (isa_command.command.startsWith("<local facts and defs>")) {
        val tls_name: String =
          isa_command.command.stripPrefix("<local facts and defs>").trim
        deal_with_local_facts_and_defs(tls_name)
      } else if (isa_command.command.startsWith("<global facts and defs>")) {
        val tls_name: String =
          isa_command.command.stripPrefix("<global facts and defs>").trim
        deal_with_global_facts_and_defs(tls_name)
      } else if (isa_command.command.startsWith("<total facts and defs>")) {
        val tls_name: String =
          isa_command.command.stripPrefix("<total facts and defs>").trim
        deal_with_total_facts_and_defs(tls_name)
      } else if (
        isa_command.command.startsWith("<get global facts from file>")
      ) {
        deal_with_global_facts_from_file
      } else if (isa_command.command.startsWith("<list states>"))
        deal_with_list_states()
      else if (isa_command.command.startsWith("<initialise>"))
        deal_with_initialise()
      else if (isa_command.command.startsWith("<get state>")) {
        val tls_name: String =
          isa_command.command.stripPrefix("<get state>").trim
        deal_with_get_state(tls_name)
      } else if (isa_command.command.startsWith("<is finished>")) {
        val tls_name: String =
          isa_command.command.split("<is finished>").last.trim
        deal_with_is_finished(tls_name)
      } else if (isa_command.command.startsWith("<apply to top level state>")) {
        val tls_name: String =
          isa_command.command.split("<apply to top level state>")(1).trim
        val action: String =
          isa_command.command.split("<apply to top level state>")(2).trim
        val new_name: String =
          isa_command.command.split("<apply to top level state>")(3).trim
        val tactic_time: Int = 
          isa_command.command.split("<apply to top level state>")(4).trim.toInt
        try {
          // println(s"Start applying action '${action}' to top level state '${tls_name}'")
          deal_with_apply_to_tls(tls_name, action, new_name, tactic_time)
        } catch {
          case e: IsabelleMLException => {
            println("Action: " + action)
            println("IsabelleException: " + e.getMessage + "\n")
            "Step error: " + e.getMessage
          }
          case f: Throwable => {
            println("Action: " + action)
            println("Unknown error: " + f.getMessage + "\n")
            "Unknown error: " + f.getMessage
          }
        }
      } else if (isa_command.command.startsWith("<get_proof_level>")) {
        val tls_name: String =
          isa_command.command.stripPrefix("<get_proof_level>").trim
        deal_with_proof_level(tls_name)
      // } else if (isa_command.command.startsWith("<proceed before>")) {
      //   val true_command: String =
      //     isa_command.command.stripPrefix("<proceed before>").trim
      //   deal_with_proceed_before(true_command)
      // } else if (isa_command.command.startsWith("<proceed after>")) {
      //   val true_command: String =
      //     isa_command.command.stripPrefix("<proceed after>").trim
      //   deal_with_proceed_after(true_command)
      } else if (isa_command.command.trim.startsWith("<proceed to line>")) {
        val true_string: String = isa_command.command.split("<proceed to line>")(1).trim
        val before_after: String = isa_command.command.split("<proceed to line>")(2).trim
        val counter: Int = isa_command.command.split("<proceed to line>")(3).trim.toInt
        deal_with_proceed_to_line(true_string, before_after, counter)
      } else if (isa_command.command.trim.startsWith("<clone>")) {
        val old_name: String = isa_command.command.trim.split("<clone>")(1).trim
        val new_name: String = isa_command.command.trim.split("<clone>")(2).trim
        deal_with_clone(old_name, new_name)
      } else if (isa_command.command.trim.startsWith("<delete>")) {
        val tls_name: String =
          isa_command.command.trim.stripPrefix("<delete>").trim
        deal_with_delete(tls_name)
      } else if (isa_command.command.trim == "<get_ancestors>") {
        val ancestors_names_list: List[String] =
          pisaos.get_theory_ancestors_names(pisaos.thy1)
        ancestors_names_list.mkString(",")
      } else if (isa_command.command == "exit")
        deal_with_exit(isa_command.command)
      else if (isa_command.command.startsWith("<get dependent theorems>")) {
        val tls_name: String = isa_command.command.split("<get dependent theorems>")(1).trim
        val theorem_name: String = isa_command.command.split("<get dependent theorems>")(2).trim
        val separator: String = isa_command.command.split("<get dependent theorems>")(3).trim
        deal_with_get_dependent_theorems(tls_name, theorem_name, separator)
      } else if (isa_command.command.startsWith("<get fact definition>")) {
        val tls_name: String = isa_command.command.split("<get fact definition>")(1).trim
        val fact_name: String = isa_command.command.split("<get fact definition>")(2).trim
        deal_with_fact_definition(tls_name, fact_name)
      } else if (isa_command.command.trim == "<parse entire thy>") {
        deal_with_parse_entire_thy
      } else if (isa_command.command.trim.startsWith("<accumulative_step_to_theorem_end>")) {
        val theorem_name = isa_command.command.trim.split("<accumulative_step_to_theorem_end>")(1).trim
        deal_with_accumulative_step_to_theorem_end(theorem_name)
      } else if (isa_command.command.trim.startsWith("<accumulative_step_before_theorem_starts>")) {
        val theorem_name = isa_command.command.trim.split("<accumulative_step_before_theorem_starts>")(1).trim
        deal_with_accumulative_step_before_theorem_starts(theorem_name)
      } else if (isa_command.command.trim.startsWith("<accumulative_step_through_a_theorem>")) {
        deal_with_accumulative_step_through_a_theorem
      } else if (isa_command.command.trim.startsWith("<reset problem>")) {
        deal_with_reset_problem
      }
      else "Unrecognised operation."
    }
    ZIO.succeed(IsaState(proof_state))
  }

  def isabelleSetSearchWidth(
      request: IsaSearchWidth
  ): ZIO[zio.ZEnv with Any, Status, IsaMessage] = {
    ZIO.succeed(IsaMessage(s"This shouldn't be used here."))
  }

  def isabelleSearchIndexCommand(
      request: IsaSearchIndexCommand
  ): ZIO[zio.ZEnv with Any, Status, IsaState] =
    ZIO.succeed(IsaState(s"This shouldn't be used here."))
}

object PisaOneStage {
  val path_to_isa_bin: String = "/home/qj213/Isabelle2021"
  val path_to_afp: String = "/home/qj213/afp-2021-10-22"

  def main(args: Array[String]): Unit = {
    //    val path_to_file: String = s"$path_to_afp/thys/Game_Based_Crypto/Guessing_Many_One.thy"
    val path_to_file: String =
      "/home/qj213/miniF2F/isabelle/valid/amc12_2000_p15.thy"
    //    val working_directory: String = s"$path_to_afp/thys/Game_Based_Crypto"
    val working_directory: String = "/home/qj213/miniF2F/isabelle/valid"
    val pisaos = new PisaOS(
      path_to_isa_bin = path_to_isa_bin,
      path_to_file = path_to_file,
      working_directory = working_directory
    )
    //    val theorem_name = """lemma accepts_conv_steps: "accepts A w = (\<exists>q. (start A,q) \<in> steps A w \<and> fin A q)"""".stripMargin
    //    val parsed : String = pisaos.step("PISA extract data")
    val parsed: String = pisaos.step("PISA extract data")
    //    val parsed : String = pisaos.step_to_transition_text(theorem_name)
    // println(parsed)
    //    println(pisaos.step("by(simp add: delta_conv_steps accepts_def)"))
  }
}

object PisaMini {
  val path_to_isa_bin: String = "/home/qj213/Isabelle2021"
  val path_to_mini: String = "/home/qj213/miniF2F/isabelle"

  def main(args: Array[String]): Unit = {
    val path_to_file: String = s"$path_to_mini/valid/aime_1983_p9.thy"
    val working_directory: String = s"$path_to_mini/valid"
    val pisaos = new PisaOS(
      path_to_isa_bin = path_to_isa_bin,
      path_to_file = path_to_file,
      working_directory = working_directory
    )
    val theorem_name =
      """theorem aime_1983_p9: fixes x::real assumes "0<x" "x<pi" shows "12 \<le> ((9 * (x^2 * (sin x)^2)) + 4) / (x * sin x)"""".stripMargin
    val parsed: String = pisaos.step("PISA extract data")
    //    val parsed : String = pisaos.step_to_transition_text(theorem_name)
    // println(parsed)
    //    println(pisaos.step("by(simp add: delta_conv_steps accepts_def)"))
  }
}

object PisaExtraction {
  val path_to_isa_bin: String = "/home/qj213/Isabelle2021"
  val path_to_afp: String = "/home/qj213/afp-2021-10-22"

  def main(args: Array[String]): Unit = {
    val path_to_file: String = args(0)
    val working_directory: String = args(1)
    val dump_path: String = args(2)
    val pisaos = new PisaOS(
      path_to_isa_bin = path_to_isa_bin,
      path_to_file = path_to_file,
      working_directory = working_directory
    )
    new PrintWriter(dump_path) {
      close()
    }
  }
}
