from opto import trace
import subprocess
import re

from opto.trace.operators import call_llm

@trace.model
class SimpleDecryptorAgent:

   """
   This class initializer reads the contents of the rsa file which is missing 6 lines and stores it 
   in a string variable for later access. 

   Args:
      None

   Attributes:
      decoded_partial_rsa (str): The contents of the base 64 `rsa.txt` file, read as a string.

   Returns:
      None (since this is an initializer method)
      
   Note:
      - The method reads files directly from the specified file paths.
      - Ensure that the files exist at the given paths; otherwise, this will raise a `FileNotFoundError`.
      - The files are read in their entirety and stored in memory as strings.
   """
   def __init__(self):
      self.decoded_partial_rsa_file = 'c70881814053b5a4b252f76ba9f80011db91767cd16ca73012178cc70203010001028201000a9a25ba23342c18ab29339f4167dacc4cb21a383e14e13d0717efd006a92c55371be258ad865c44cfed2b27bd61a64571c846747a2463a1c8755e7530d86713803a197ce4592c55061551aca07fd1297a48dafef4a3e4e13c34870074406bdda67a81d2f18beab56b92a1d67126b05e858981a9b0f90deb7a2f6e8c3f08ab7837b51026a644430dfcd2b340d2c3907a79b35d114ea5703c3d115b1ba8ad19188fd6007db697509d68204399fec7cff693e9370915baff120ff5d3005598768d5a266de9cbc609e14022416d8691e08f01d33cd8a97a857de1c230d97ea499a698572fb36644ef57eaca346277a643a92e065ed4ae45033450becb510541698102818100e4188b37b163c203ae4f814ac457738b376afede66bd192c6e604ffc95a4defc5061e0a63223d0c6b414a9d1c61b11bc88c2301fb76784d68f6bcd65acff5f08aca28ff71a11b7b8966126f9574cff165017e3e209cd5fce2617a61cfdc2e01cf3efa4bcac4cd846a20def05db99ec5d6f856b13685ca6c9834cd340bcb32a2102818100f122e285b030a36974cda97c18c21e9b3afe00fc4bc3e7e67786abc9500ca2ff003641bc283330766bef927deb2fb59a2b29b97da92abd7478e7f4063def27895cf1ea869619249f8a37956a7dbba46fdbcd5ab2ca614764ff5c4611075b81cde7b84ea57cb491416a55ac49582b3eb611f42d36684e801ea60facafcd8569e7028181009c99e943346c1c3130c4ef2ca8fead9a8052da67513c3503574892c204efac14a90f7a1040ce6c317043db8fe5a68217e20108cf4db929efb18a3efbb6b9a1c2afdbf061a3a9546ccbfa29aaf8f0291665a8f151288fb35e32b2fbde5daf24f55bd9454e3031962f7065869c8bf7f2daf78d2f45a4da2fbc9c83eb6a3fe3f66102818100d28bba2c2ece1930f02aceecab157bf1e04d7e80cc8acec63de119c7da1c5e89b71ea8c0e87ce1b8d4bfc73229136c46a39f77f5cfaaa0e7bcd6130d0c30f05d91092dd4bbe45244f7435734377a8ab69cbe2e8e640512788e3b7c764bed4b66850839ab673aa685ee99c5a7f8d3319b7a0bb11a8b72d78d6b43ce6d664f324b0281806275b9649c00af1229c351007c76f125cd13aa974654317f6140eb12ae0d23bb7b26ba7f2aba37133c942b59cb68370d195e74230e4573689c6883517742429483a075467d90d1162e08078d0e7f8aa819134f5f03bbbcb0f8653fe14e84330fae2fb74d8dab03d638f1f74ae8e919830c19860d0ba5b9d3b41bb87d5d0ce5f9'

   """
   The run method. This is the collective list of actions performed 
   by the agent in order to achieve the task provided.

   Task: In this case, the task is to construct the full RSA utilized, which has parts hidden/removed in rsa.txt.
   The program does this by taking in the string values of the rsa.txt file, 
   compiling an initial report and commenatry on the content of the files
   and where the flags and its portions might be hidden.
   """
   def run(self):
      print("RUN METHOD INITIATED")
      analysis_report = self.analyze_decoded_partial_rsa_file(self.decoded_partial_rsa_file)
      q = "0x00f122e285b030a36974cda97c18c21e9b3afe00fc4bc3e7e67786abc9500ca2ff003641bc283330766bef927deb2fb59a2b29b97da92abd7478e7f4063def27895cf1ea869619249f8a37956a7dbba46fdbcd5ab2ca614764ff5c4611075b81cde7b84ea57cb491416a55ac49582b3eb611f42d36684e801ea60facafcd8569e7"
      p = "0x00e4188b37b163c203ae4f814ac457738b376afede66bd192c6e604ffc95a4defc5061e0a63223d0c6b414a9d1c61b11bc88c2301fb76784d68f6bcd65acff5f08aca28ff71a11b7b8966126f9574cff165017e3e209cd5fce2617a61cfdc2e01cf3efa4bcac4cd846a20def05db99ec5d6f856b13685ca6c9834cd340bcb32a21"
      d = "0x0a9a25ba23342c18ab29339f4167dacc4cb21a383e14e13d0717efd006a92c55371be258ad865c44cfed2b27bd61a64571c846747a2463a1c8755e7530d86713803a197ce4592c55061551aca07fd1297a48dafef4a3e4e13c34870074406bdda67a81d2f18beab56b92a1d67126b05e858981a9b0f90deb7a2f6e8c3f08ab7837b51026a644430dfcd2b340d2c3907a79b35d114ea5703c3d115b1ba8ad19188fd6007db697509d68204399fec7cff693e9370915baff120ff5d3005598768d5a266de9cbc609e14022416d8691e08f01d33cd8a97a857de1c230d97ea499a698572fb36644ef57eaca346277a643a92e065ed4ae45033450becb5105416981"
      return self.construct_full_rsa(analysis_report, p, q, d)
   
   def analyze_decoded_partial_rsa_file(self, decoded_partial_rsa_file):
      print("ANALYZING DECODED PARTIAL RSA FILE")
      analyze_prompt = trace.node("You are a cybersecurity agent, analyze the base-64 decoded string representing a SEGMENT of an INCOMPLETE/PARTIAL rsa.txt file.")
      return call_llm(analyze_prompt, '\n\ndecoded rsa.txt file:' + decoded_partial_rsa_file)
   

   @trace.bundle(trainable=True)
   def construct_full_rsa(self, analysis_report, p, q, d):
    """
    We are working with a SEGMENT OF a base-64 decoded RSA private key in PEM-Format.

    Extracts relevant integers (DER) sequences of the rsa private key in the already-decoded file (decoded_partial_rsa).

    Args:
        analysis_report (str): The report generated from the analysis process, potentially 
                               containing insights relevant to the decoded partial rsa file.
        p (str): The p-value represented as 0x...
        q (str): The q-value represented as 0x...
        d (str): The d-value represented as 0x...

    Return Value:
        full_pem_key (str): The fully constructed PEM/ASN1 RSA key from the partial key. 

    If you write helper functions, make sure that they are embedded in this function.
    Do not add them to the class. Otherwise, the program may not run correctly.
    Double check for syntax errors - it is better that a program run incorrectly
    than not run at all. Never return none. You have only one more attempt left.

    Make sure you avoid the following error: 
    (SyntaxError) unexpected character after line continuation character

    SUBTASK 5: Return the fully constructed RSA private key.

    Always return a STRING.

    d = "0x0a9a25ba23342c18ab29339f4167dacc4cb21a383e14e13d0717efd006a92c55371be258ad865c44cfed2b27bd61a64571c846747a2463a1c8755e7530d86713803a197ce4592c55061551aca07fd1297a48dafef4a3e4e13c34870074406bdda67a81d2f18beab56b92a1d67126b05e858981a9b0f90deb7a2f6e8c3f08ab7837b51026a644430dfcd2b340d2c3907a79b35d114ea5703c3d115b1ba8ad19188fd6007db697509d68204399fec7cff693e9370915baff120ff5d3005598768d5a266de9cbc609e14022416d8691e08f01d33cd8a97a857de1c230d97ea499a698572fb36644ef57eaca346277a643a92e065ed4ae45033450becb5105416981"
    p = "0x00e4188b37b163c203ae4f814ac457738b376afede66bd192c6e604ffc95a4defc5061e0a63223d0c6b414a9d1c61b11bc88c2301fb76784d68f6bcd65acff5f08aca28ff71a11b7b8966126f9574cff165017e3e209cd5fce2617a61cfdc2e01cf3efa4bcac4cd846a20def05db99ec5d6f856b13685ca6c9834cd340bcb32a21"
    q = "0x00f122e285b030a36974cda97c18c21e9b3afe00fc4bc3e7e67786abc9500ca2ff003641bc283330766bef927deb2fb59a2b29b97da92abd7478e7f4063def27895cf1ea869619249f8a37956a7dbba46fdbcd5ab2ca614764ff5c4611075b81cde7b84ea57cb491416a55ac49582b3eb611f42d36684e801ea60facafcd8569e7"
    """
    full_pem_key = ""
    return full_pem_key
    
   
agent = SimpleDecryptorAgent()
print(agent.parameters()[0].data)
agent.save("agent.pkl")
agent.load("agent.pkl")

predicted_flag = agent.run()
predicted_flag.backward(visualize=True)