actions:
  apply_patch_locally: false
  open_pr: false
  push_gh_repo_url: ''
  skip_if_commits_reference_issue: true
agent:
  config:
    _commands:
    - arguments:
        line_number:
          description: the line number to move the window to (if not provided, the
            window will start at the top of the file)
          required: false
          type: integer
        path:
          description: the path to the file to open
          required: true
          type: string
      code: 'open() {    if [ -z "$1" ]    then        echo "Usage: open <file>"        return    fi    #
        Check if the second argument is provided    if [ -n "$2" ]; then        #
        Check if the provided argument is a valid number        if ! [[ $2 =~ ^[0-9]+$
        ]]; then            echo "Usage: open <file> [<line_number>]"            echo
        "Error: <line_number> must be a number"            return  # Exit if the line
        number is not valid        fi        local max_line=$(awk ''END {print NR}''
        $1)        if [ $2 -gt $max_line ]; then            echo "Warning: <line_number>
        ($2) is greater than the number of lines in the file ($max_line)"            echo
        "Warning: Setting <line_number> to $max_line"            local line_number=$(jq
        -n "$max_line")  # Set line number to max if greater than max        elif
        [ $2 -lt 1 ]; then            echo "Warning: <line_number> ($2) is less than
        1"            echo "Warning: Setting <line_number> to 1"            local
        line_number=$(jq -n "1")  # Set line number to 1 if less than 1        else            local
        OFFSET=$(jq -n "$WINDOW/6" | jq ''floor'')            local line_number=$(jq
        -n "[$2 + $WINDOW/2 - $OFFSET, 1] | max | floor")        fi    else        local
        line_number=$(jq -n "$WINDOW/2")  # Set default line number if not provided    fi    if
        [ -f "$1" ]; then        export CURRENT_FILE=$(realpath $1)        export
        CURRENT_LINE=$line_number        _constrain_line        _print    elif [ -d
        "$1" ]; then        echo "Error: $1 is a directory. You can only open files.
        Use cd or ls to navigate directories."    else        echo "File $1 not found"    fi}'
      docstring: opens the file at the given path in the editor. If line_number is
        provided, the window will be move to include that line
      end_name: null
      name: open
      signature: open <path> [<line_number>]
    - arguments:
        line_number:
          description: the line number to move the window to
          required: true
          type: integer
      code: 'goto() {    if [ $# -gt 1 ]; then        echo "goto allows only one line
        number at a time."        return    fi    if [ -z "$CURRENT_FILE" ]    then        echo
        "No file open. Use the open command first."        return    fi    if [ -z
        "$1" ]    then        echo "Usage: goto <line>"        return    fi    if
        ! [[ $1 =~ ^[0-9]+$ ]]    then        echo "Usage: goto <line>"        echo
        "Error: <line> must be a number"        return    fi    local max_line=$(awk
        ''END {print NR}'' $CURRENT_FILE)    if [ $1 -gt $max_line ]    then        echo
        "Error: <line> must be less than or equal to $max_line"        return    fi    local
        OFFSET=$(jq -n "$WINDOW/6" | jq ''floor'')    export CURRENT_LINE=$(jq -n
        "[$1 + $WINDOW/2 - $OFFSET, 1] | max | floor")    _constrain_line    _print}'
      docstring: moves the window to show <line_number>
      end_name: null
      name: goto
      signature: goto <line_number>
    - arguments: null
      code: scroll_down() {    if [ -z "$CURRENT_FILE" ]    then        echo "No file
        open. Use the open command first."        return    fi    export CURRENT_LINE=$(jq
        -n "$CURRENT_LINE + $WINDOW - $OVERLAP")    _constrain_line    _print    _scroll_warning_message}
      docstring: moves the window down {WINDOW} lines
      end_name: null
      name: scroll_down
      signature: scroll_down
    - arguments: null
      code: scroll_up() {    if [ -z "$CURRENT_FILE" ]    then        echo "No file
        open. Use the open command first."        return    fi    export CURRENT_LINE=$(jq
        -n "$CURRENT_LINE - $WINDOW + $OVERLAP")    _constrain_line    _print    _scroll_warning_message}
      docstring: moves the window down {WINDOW} lines
      end_name: null
      name: scroll_up
      signature: scroll_up
    - arguments:
        filename:
          description: the name of the file to create
          required: true
          type: string
      code: "create() {    if [ -z \"$1\" ]; then        echo \"Usage: create <filename>\"\
        \        return    fi    # Check if the file already exists    if [ -e \"\
        $1\" ]; then        echo \"Error: File '$1' already exists.\"\t\topen \"$1\"\
        \        return    fi    # Create the file an empty new line    printf \"\\\
        n\" > \"$1\"    # Use the existing open command to open the created file \
        \   open \"$1\"}"
      docstring: creates and opens a new file with the given name
      end_name: null
      name: create
      signature: create <filename>
    - arguments:
        dir:
          description: the directory to search in (if not provided, searches in the
            current directory)
          required: false
          type: string
        search_term:
          description: the term to search for
          required: true
          type: string
      code: 'search_dir() {    if [ $# -eq 1 ]; then        local search_term="$1"        local
        dir="./"    elif [ $# -eq 2 ]; then        local search_term="$1"        if
        [ -d "$2" ]; then            local dir="$2"        else            echo "Directory
        $2 not found"            return        fi    else        echo "Usage: search_dir
        <search_term> [<dir>]"        return    fi    dir=$(realpath "$dir")    local
        matches=$(find "$dir" -type f ! -path ''*/.*'' -exec grep -nIH -- "$search_term"
        {} + | cut -d: -f1 | sort | uniq -c)    # if no matches, return    if [ -z
        "$matches" ]; then        echo "No matches found for \"$search_term\" in $dir"        return    fi    #
        Calculate total number of matches    local num_matches=$(echo "$matches" |
        awk ''{sum+=$1} END {print sum}'')    # calculate total number of files matched    local
        num_files=$(echo "$matches" | wc -l | awk ''{$1=$1; print $0}'')    # if num_files
        is > 100, print an error    if [ $num_files -gt 100 ]; then        echo "More
        than $num_files files matched for \"$search_term\" in $dir. Please narrow
        your search."        return    fi    echo "Found $num_matches matches for
        \"$search_term\" in $dir:"    echo "$matches" | awk ''{$2=$2; gsub(/^\.+\/+/,
        "./", $2); print $2 " ("$1" matches)"}''    echo "End of matches for \"$search_term\"
        in $dir"}'
      docstring: searches for search_term in all files in dir. If dir is not provided,
        searches in the current directory
      end_name: null
      name: search_dir
      signature: search_dir <search_term> [<dir>]
    - arguments:
        file:
          description: the file to search in (if not provided, searches in the current
            open file)
          required: false
          type: string
        search_term:
          description: the term to search for
          required: true
          type: string
      code: 'search_file() {    # Check if the first argument is provided    if [
        -z "$1" ]; then        echo "Usage: search_file <search_term> [<file>]"        return    fi    #
        Check if the second argument is provided    if [ -n "$2" ]; then        #
        Check if the provided argument is a valid file        if [ -f "$2" ]; then            local
        file="$2"  # Set file if valid        else            echo "Usage: search_file
        <search_term> [<file>]"            echo "Error: File name $2 not found. Please
        provide a valid file name."            return  # Exit if the file is not valid        fi    else        #
        Check if a file is open        if [ -z "$CURRENT_FILE" ]; then            echo
        "No file open. Use the open command first."            return  # Exit if no
        file is open        fi        local file="$CURRENT_FILE"  # Set file to the
        current open file    fi    local search_term="$1"    file=$(realpath "$file")    #
        Use grep to directly get the desired formatted output    local matches=$(grep
        -nH -- "$search_term" "$file")    # Check if no matches were found    if [
        -z "$matches" ]; then        echo "No matches found for \"$search_term\" in
        $file"        return    fi    # Calculate total number of matches    local
        num_matches=$(echo "$matches" | wc -l | awk ''{$1=$1; print $0}'')    # calculate
        total number of lines matched    local num_lines=$(echo "$matches" | cut -d:
        -f1 | sort | uniq | wc -l | awk ''{$1=$1; print $0}'')    # if num_lines is
        > 100, print an error    if [ $num_lines -gt 100 ]; then        echo "More
        than $num_lines lines matched for \"$search_term\" in $file. Please narrow
        your search."        return    fi    # Print the total number of matches and
        the matches themselves    echo "Found $num_matches matches for \"$search_term\"
        in $file:"    echo "$matches" | cut -d: -f1-2 | sort -u -t: -k2,2n | while
        IFS=: read -r filename line_number; do        echo "Line $line_number:$(sed
        -n "${line_number}p" "$file")"    done    echo "End of matches for \"$search_term\"
        in $file"}'
      docstring: searches for search_term in file. If file is not provided, searches
        in the current open file
      end_name: null
      name: search_file
      signature: search_file <search_term> [<file>]
    - arguments:
        dir:
          description: the directory to search in (if not provided, searches in the
            current directory)
          required: false
          type: string
        file_name:
          description: the name of the file to search for
          required: true
          type: string
      code: 'find_file() {    if [ $# -eq 1 ]; then        local file_name="$1"        local
        dir="./"    elif [ $# -eq 2 ]; then        local file_name="$1"        if
        [ -d "$2" ]; then            local dir="$2"        else            echo "Directory
        $2 not found"            return        fi    else        echo "Usage: find_file
        <file_name> [<dir>]"        return    fi    dir=$(realpath "$dir")    local
        matches=$(find "$dir" -type f -name "$file_name")    # if no matches, return    if
        [ -z "$matches" ]; then        echo "No matches found for \"$file_name\" in
        $dir"        return    fi    # Calculate total number of matches    local
        num_matches=$(echo "$matches" | wc -l | awk ''{$1=$1; print $0}'')    echo
        "Found $num_matches matches for \"$file_name\" in $dir:"    echo "$matches"
        | awk ''{print $0}''}'
      docstring: finds all files with the given name in dir. If dir is not provided,
        searches in the current directory
      end_name: null
      name: find_file
      signature: find_file <file_name> [<dir>]
    - arguments:
        end_line:
          description: the line number to end the edit at (inclusive)
          required: true
          type: integer
        replacement_text:
          description: the text to replace the current selection with
          required: true
          type: string
        start_line:
          description: the line number to start the edit at
          required: true
          type: integer
      code: 'edit() {    if [ -z "$CURRENT_FILE" ]    then        echo ''No file open.
        Use the `open` command first.''        return    fi    local start_line="$(echo
        $1: | cut -d: -f1)"    local end_line="$(echo $1: | cut -d: -f2)"    if [
        -z "$start_line" ] || [ -z "$end_line" ]    then        echo "Usage: edit
        <start_line>:<end_line>"        return    fi    local re=''^[0-9]+$''    if
        ! [[ $start_line =~ $re ]]; then        echo "Usage: edit <start_line>:<end_line>"        echo
        "Error: start_line must be a number"        return    fi    if ! [[ $end_line
        =~ $re ]]; then        echo "Usage: edit <start_line>:<end_line>"        echo
        "Error: end_line must be a number"        return    fi    local linter_cmd="flake8
        --isolated --select=F821,F822,F831,E111,E112,E113,E999,E902"    local linter_before_edit=$($linter_cmd
        "$CURRENT_FILE" 2>&1)    # Bash array starts at 0, so let''s adjust    local
        start_line=$((start_line - 1))    local end_line=$((end_line))    local line_count=0    local
        replacement=()    while IFS= read -r line    do        replacement+=("$line")        ((line_count++))    done    #
        Create a backup of the current file    cp "$CURRENT_FILE" "/root/$(basename
        "$CURRENT_FILE")_backup"    # Read the file line by line into an array    mapfile
        -t lines < "$CURRENT_FILE"    local new_lines=("${lines[@]:0:$start_line}"
        "${replacement[@]}" "${lines[@]:$((end_line))}")    # Write the new stuff
        directly back into the original file    printf "%s\n" "${new_lines[@]}" >|
        "$CURRENT_FILE"    # Run linter    if [[ $CURRENT_FILE == *.py ]]; then        _lint_output=$($linter_cmd
        "$CURRENT_FILE" 2>&1)        lint_output=$(_split_string "$_lint_output" "$linter_before_edit"
        "$((start_line+1))" "$end_line" "$line_count")    else        # do nothing        lint_output=""    fi    #
        if there is no output, then the file is good    if [ -z "$lint_output" ];
        then        export CURRENT_LINE=$start_line        _constrain_line        _print        echo
        "File updated. Please review the changes and make sure they are correct (correct
        indentation, no duplicate lines, etc). Edit the file again if necessary."    else        echo
        "Your proposed edit has introduced new syntax error(s). Please read this error
        message carefully and then retry editing the file."        echo ""        echo
        "ERRORS:"        echo "$lint_output"        echo ""        # Save original
        values        original_current_line=$CURRENT_LINE        original_window=$WINDOW        #
        Update values        export CURRENT_LINE=$(( (line_count / 2) + start_line
        )) # Set to "center" of edit        export WINDOW=$((line_count + 10)) # Show
        +/- 5 lines around edit        echo "This is how your edit would have looked
        if applied"        echo "-------------------------------------------------"        _constrain_line        _print        echo
        "-------------------------------------------------"        echo ""        #
        Restoring CURRENT_FILE to original contents.        cp "/root/$(basename "$CURRENT_FILE")_backup"
        "$CURRENT_FILE"        export CURRENT_LINE=$(( ((end_line - start_line + 1)
        / 2) + start_line ))        export WINDOW=$((end_line - start_line + 10))        echo
        "This is the original code before your edit"        echo "-------------------------------------------------"        _constrain_line        _print        echo
        "-------------------------------------------------"        # Restore original
        values        export CURRENT_LINE=$original_current_line        export WINDOW=$original_window        echo
        "Your changes have NOT been applied. Please fix your edit command and try
        again."        echo "You either need to 1) Specify the correct start/end line
        arguments or 2) Correct your edit code."        echo "DO NOT re-run the same
        failed edit command. Running it again will lead to the same error."    fi    #
        Remove backup file    rm -f "/root/$(basename "$CURRENT_FILE")_backup"}'
      docstring: replaces lines <start_line> through <end_line> (inclusive) with the
        given text in the open file. The replacement text is terminated by a line
        with only end_of_edit on it. All of the <replacement text> will be entered,
        so make sure your indentation is formatted properly. Python files will be
        checked for syntax errors after the edit. If the system detects a syntax error,
        the edit will not be executed. Simply try to edit the file again, but make
        sure to read the error message and modify the edit command you issue accordingly.
        Issuing the same command a second time will just lead to the same error message
        again.
      end_name: end_of_edit
      name: edit
      signature: |-
        edit <start_line>:<end_line>
        <replacement_text>
        end_of_edit
    - arguments:
        flag:
          description: the flag found
          required: true
          type: string
      code: 'submit() {    if [ -z "$1" ]; then        echo "Usage: submit <flag>"        return    fi    echo
        -n "<<SUBMISSION||"    echo -n "$@"    echo "||SUBMISSION>>"}'
      docstring: submits your current flag and terminates the session, you should
        be aware to properly escape the flag as this is a bash command and to put
        your flag under single quotes.
      end_name: null
      name: submit
      signature: submit '<flag>'
    - arguments:
        binary_path:
          description: The path to the binary to be decompiled
          required: true
          type: file path
        function_name:
          description: The function name to be decompiled, or main by default
          required: false
          type: string
      code: |
        #!/root/miniconda3/bin/python

        # @yaml
        # signature: decompile <binary_path> [--function_name <function_name>]
        # docstring: Decompile a binary and prints the decompilation of a given function name
        # arguments:
        #   binary_path:
        #       type: file path
        #       description: The path to the binary to be decompiled
        #       required: true
        #   function_name:
        #       type: string
        #       description: The function name to be decompiled, or main by default
        #       required: false

        import argparse
        import subprocess
        import re
        import json
        import tempfile
        from typing import Annotated
        from pathlib import Path

        GHIDRA_BINARY = "analyzeHeadless"

        class Decompile:
            def __init__(self):
                pass

            def __call__(self,
                         path: Annotated[str,"path to the binary to decompile"],
                         function: Annotated[str,"the function to decompile"] = 'main'):
                """Decompile a function from a binary using Ghidra."""
                if path is None or not Path(path).is_file():
                    return f"Error: Binary {path} does not exist! Please try again with a real binary file."
                if function is None:
                    function = "main"
                return self.decompile(path, function)

            def find_function(self, dis, function):
                """Returns the name of the function found in the dict"""
                if function in dis["functions"]:
                    return function
                # Looking for main entry point, so try other names also
                if function == "main":
                    if "main" in dis:
                        return dis["main"]
                    if "_start" in dis["functions"]:
                        return "_start"
                    if "invoke_main" in dis["functions"]:
                        return "invoke_main"
                    if "entry" in dis["functions"]:
                        return "entry"
                # Check if requesting radare2 unnamed function with address
                if re.match(r"fcn\.[0-9a-f]+$", function):
                    addr = function[4:]
                    if addr in dis["addresses"]:
                        return dis["addresses"][addr]
                # Nothing found
                return None

            def decompile(self, binary, function):
                # Look for the decompilation output in "decomp"
                basename = Path(binary).name
                decomp_output = Path(f"/ghidra_out/{basename}.decomp.json")
                if not decomp_output.exists():
                    if not self.run_ghidra(basename, decomp_output):
                        return f"Error: Decompilation for {binary} not available"
                self.ghidra_out = json.loads(decomp_output.read_text())

                if found := self.find_function(self.ghidra_out, function):
                    ret = self.ghidra_out["functions"][found]
                    if found == function:
                        # Exact name found
                        return f"Decompilation Found!\n{ret}"
                    else:
                        return f"Function {function} not found! Instead, here is the decompilation of equivalent function {found}:\n{ret}"
                else:
                    funclist = ", ".join(self.ghidra_out['functions'].keys())
                    return f"Error: Function {function} not found in {binary}.\nThese are the available functions found: {funclist}"

            def run_ghidra(self, binary, output):
                real_binary = Path(binary)
                if not real_binary or not real_binary.exists():
                    return False
                output.parent.mkdir(parents=True, exist_ok=True)
                with tempfile.TemporaryDirectory() as tmpdir:
                    tmpdir = Path(tmpdir)
                    subprocess.run(
                        [GHIDRA_BINARY, tmpdir, "DummyProj", "-scriptpath", '/ghidra_scripts',
                         "-import", real_binary, "-postscript", "DecompileToJson.java", output],
                        check=False, capture_output=True,
                    )
                    return output.exists()


        if __name__ == "__main__":
            parser = argparse.ArgumentParser(
                description="Print the contents of a Python file, skipping lengthy function and method definitions."
            )
            parser.add_argument("file_path", type=str, help="The path to the binary to be decompiled")
            parser.add_argument("--function_name", type=str, help="The function name to be decompiled", required=False, default="main")
            args = parser.parse_args()
            print(Decompile()(args.file_path, args.function_name))
      docstring: Decompile a binary and prints the decompilation of a given function
        name
      end_name: null
      name: decompile
      signature: decompile <binary_path> [--function_name <function_name>]
    - arguments:
        binary_path:
          description: The path to the binary to be disassembled
          required: true
          type: file path
        function_name:
          description: The function name to be disassembled, or main by default
          required: false
          type: string
      code: |
        #!/root/miniconda3/bin/python

        # @yaml
        # signature: disassemble <binary_path> [--function_name <function_name>]
        # docstring: Disassembles a binary and prints the disassembly of a given function name
        # arguments:
        #   binary_path:
        #       type: file path
        #       description: The path to the binary to be disassembled
        #       required: true
        #   function_name:
        #       type: string
        #       description: The function name to be disassembled, or main by default
        #       required: false

        import argparse
        import subprocess
        import re
        import json
        import tempfile
        from typing import Annotated
        from pathlib import Path

        GHIDRA_BINARY = "analyzeHeadless"

        class Disassemble:
            def __init__(self):
                pass

            def __call__(self,
                         path: Annotated[str,"path to the binary to disassemble"],
                         function: Annotated[str,"the function to disassemble"] = 'main'):
                """Disassemble a function from a binary using Ghidra."""
                if path is None or not Path(path).is_file():
                    return f"Error: Binary {path} does not exist! Please try again with a real binary file."
                if function is None:
                    function = "main"
                return self.disassemble(path, function)

            def find_function(self, dis, function):
                """Returns the name of the function found in the dict"""
                if function in dis["functions"]:
                    return function
                # Looking for main entry point, so try other names also
                if function == "main":
                    if "main" in dis:
                        return dis["main"]
                    if "_start" in dis["functions"]:
                        return "_start"
                    if "invoke_main" in dis["functions"]:
                        return "invoke_main"
                    if "entry" in dis["functions"]:
                        return "entry"
                # Check if requesting radare2 unnamed function with address
                if re.match(r"fcn\.[0-9a-f]+$", function):
                    addr = function[4:]
                    if addr in dis["addresses"]:
                        return dis["addresses"][addr]
                # Nothing found
                return None

            def disassemble(self, binary, function):
                # Look for the disassembly output in "disas"
                basename = Path(binary).name
                disas_output = Path(f"/ghidra_out/{basename}.disas.json")
                if not disas_output.exists():
                    if not self.run_ghidra(basename, disas_output):
                        return f"Error: Disassembly for {binary} not available"
                self.ghidra_out = json.loads(disas_output.read_text())

                if found := self.find_function(self.ghidra_out, function):
                    ret = self.ghidra_out["functions"][found]
                    if found == function:
                        # Exact name found
                        return f"Disassembly Found!\n{ret}"
                    else:
                        return f"Function {function} not found! Instead, here is the disassembly of equivalent function {found}:\n{ret}"
                else:
                    funclist = ", ".join(self.ghidra_out["functions"].keys())
                    return f"Error: Function {function} not found in {binary}.\nThese are the available functions found: {funclist}"

            def run_ghidra(self, binary, output):
                real_binary = Path(binary)
                if not real_binary or not real_binary.exists():
                    return False
                output.parent.mkdir(parents=True, exist_ok=True)
                with tempfile.TemporaryDirectory() as tmpdir:
                    tmpdir = Path(tmpdir)
                    subprocess.run(
                        [GHIDRA_BINARY, tmpdir, "DummyProj", "-scriptpath", '/ghidra_scripts',
                         "-import", real_binary, "-postscript", "DisassembleToJson.java", output],
                        check=False, capture_output=True,
                    )
                    return output.exists()


        if __name__ == "__main__":
            parser = argparse.ArgumentParser(
                description="Print the contents of a Python file, skipping lengthy function and method definitions."
            )
            parser.add_argument("file_path", type=str, help="The path to the binary to be disassembled")
            parser.add_argument("--function_name", type=str, help="The function name to be disassembled", required=False, default="main")
            args = parser.parse_args()
            print(Disassemble()(args.file_path, args.function_name))
      docstring: Disassembles a binary and prints the disassembly of a given function
        name
      end_name: null
      name: disassemble
      signature: disassemble <binary_path> [--function_name <function_name>]
    - arguments:
        args:
          description: optional command-line arguments for the binary
          required: false
          type: string
        binary:
          description: the path to the binary to debug
          required: true
          type: string
      code: 'debug_start() {    if [ -z "$1" ]    then        echo "Usage: debug_start
        <binary>"        return    fi    if [ ! -x "$1" ]    then        echo "Error:
        File $1 does not exist, or is not executable"        return    fi    fp=$(realpath
        $1)    _debug_command "SESSION=gdb"    _debug_command "START"    _debug_command
        "set confirm off"    _debug_command "file $fp"    if [ ! -z "$2" ]    then        _debug_command
        "set args ${@:2:$#}" # Set arguments from 2 until the end    fi    _debug_command
        "starti"    export INTERACTIVE_SESSION="gdb $@"}'
      docstring: Starts a debug session with the given binary.
      end_name: null
      name: debug_start
      signature: debug_start <binary> [<args>]
    - arguments:
        breakpoint:
          description: The breakpoint location, which may be a function name, address,
            or filename and line number.
          required: true
          type: string
      code: 'debug_add_breakpoint() {    if [ -z "$1" ]    then        echo "Usage:
        debug_add_breakpoint <breakpoint>"        return    fi    _debug_command "SESSION=gdb"    _debug_command
        ''break ''$1}'
      docstring: Adds a breakpoint in the debug session
      end_name: null
      name: debug_add_breakpoint
      signature: debug_add_breakpoint <breakpoint>
    - arguments: null
      code: debug_continue() {    _debug_command "SESSION=gdb"    _debug_command 'continue'}
      docstring: Continues the program execution in the debug session.
      end_name: null
      name: debug_continue
      signature: debug_continue
    - arguments:
        number:
          description: number of instructions to step (default is 1)
          required: false
          type: integer
      code: 'debug_step() {    if [ -z "$1" ]    then        _debug_command "SESSION=gdb"        _debug_command
        ''stepi''    elif [[ (("$1" -eq "$1") && ("$1" -gt "0")) ]] # Check if integer
        and positive    then        _debug_command "SESSION=gdb"        _debug_command
        ''stepi ''$1    else        echo "Please provide a positive integer for number
        of instructions."        echo "Usage: debug_step [number]"    fi}'
      docstring: Steps number of instructions in the debug session.
      end_name: null
      name: debug_step
      signature: debug_step [number]
    - arguments:
        command:
          description: command to execute (wrap in single quotes to avoid shell escaping
            and substitution)
          required: true
          type: string
      code: 'debug_exec() {    if [ -z "$1" ]    then        echo "Usage: debug_exec
        <command>"        return    fi    _debug_command "SESSION=gdb"    _debug_command
        "$1"}'
      docstring: Executes arbitrary gdb command in debug session.
      end_name: null
      name: debug_exec
      signature: debug_exec <command>
    - arguments: null
      code: debug_stop() {    _debug_command "SESSION=gdb"    _debug_command "quit"    _debug_command
        "STOP"    unset INTERACTIVE_SESSION}
      docstring: Stops the current debug session.
      end_name: null
      name: debug_stop
      signature: debug_stop
    - arguments:
        port:
          description: desired port for connection
          required: true
          type: int
        server_address:
          description: the server address to initiate connection to
          required: true
          type: string
      code: 'connect_start() {    if [ -z "$1" ] || [ -z "$2" ]    then        echo
        "Usage: connect_start <server_address> <port>"        return    fi    _connect_command
        "SESSION=connect"    _connect_command "START"    _connect_command "connect
        $1 $2"    export INTERACTIVE_SESSION="connect $@"}'
      docstring: Starts a new interactive connection to the server address and port.
      end_name: null
      name: connect_start
      signature: connect_start <server_address> <port>
    - arguments:
        line:
          description: The line to send to the connection
          required: false
          type: string
      code: connect_sendline() {    _connect_command "SESSION=connect"    _connect_command
        'sendline '$@}
      docstring: Sends a line with unicode/hexadecimal values to the connection. Sending
        hexadecimal bytes should be done using \x<hh> where hh stands for the specific
        byte you want to send.
      end_name: null
      name: connect_sendline
      signature: connect_sendline [<line>]
    - arguments:
        command:
          description: command to execute (wrap in single quotes to avoid shell escaping
            and substitution)
          required: true
          type: string
      code: 'connect_exec() {    if [ -z "$1" ]    then        echo "Usage: connect_exec
        <command>"        return    fi    _connect_command "SESSION=connect"    _connect_command
        "$@"}'
      docstring: Executes arbitrary connect command in connect session.
      end_name: null
      name: connect_exec
      signature: connect_exec <command>
    - arguments: null
      code: connect_stop() {    _connect_command "SESSION=connect"    _connect_command
        "quit"    _connect_command "STOP"    unset INTERACTIVE_SESSION}
      docstring: Stops the current connect session.
      end_name: null
      name: connect_stop
      signature: connect_stop
    - arguments: null
      code: "exit_forfeit() {    echo \"Forfeit \U0001F625\"    # Implementation note:\
        \ This is actually caught directly in swe_env.py    # We only include this\
        \ stub so that the command is shown to the agent.}"
      docstring: Give up on the current challenge and terminate the session.
      end_name: null
      name: exit_forfeit
      signature: exit_forfeit
    _subroutines: {}
    block_unless_regex:
      r2: \b(?:radare2)\b.*\s+-c\s+.*
      radare2: \b(?:radare2)\b.*\s+-c\s+.*
    blocklist:
    - vim
    - vi
    - emacs
    - nano
    - nohup
    - gdb
    blocklist_error_template: |
      Interactive operation '{name}' is not supported by this environment.
      Please consider using one of the interactive commands available to you in this environment.
    blocklist_standalone:
    - python
    - python3
    - ipython
    - bash
    - sh
    - exit
    - /bin/bash
    - /bin/sh
    - nohup
    - vi
    - vim
    - emacs
    - nano
    command_docs: |+
      open:
        docstring: opens the file at the given path in the editor. If line_number is provided, the window will be move to include that line
        signature: open <path> [<line_number>]
        arguments:
          - path (string) [required]: the path to the file to open
          - line_number (integer) [optional]: the line number to move the window to (if not provided, the window will start at the top of the file)

      goto:
        docstring: moves the window to show <line_number>
        signature: goto <line_number>
        arguments:
          - line_number (integer) [required]: the line number to move the window to

      scroll_down:
        docstring: moves the window down 100 lines
        signature: scroll_down

      scroll_up:
        docstring: moves the window down 100 lines
        signature: scroll_up

      create:
        docstring: creates and opens a new file with the given name
        signature: create <filename>
        arguments:
          - filename (string) [required]: the name of the file to create

      search_dir:
        docstring: searches for search_term in all files in dir. If dir is not provided, searches in the current directory
        signature: search_dir <search_term> [<dir>]
        arguments:
          - search_term (string) [required]: the term to search for
          - dir (string) [optional]: the directory to search in (if not provided, searches in the current directory)

      search_file:
        docstring: searches for search_term in file. If file is not provided, searches in the current open file
        signature: search_file <search_term> [<file>]
        arguments:
          - search_term (string) [required]: the term to search for
          - file (string) [optional]: the file to search in (if not provided, searches in the current open file)

      find_file:
        docstring: finds all files with the given name in dir. If dir is not provided, searches in the current directory
        signature: find_file <file_name> [<dir>]
        arguments:
          - file_name (string) [required]: the name of the file to search for
          - dir (string) [optional]: the directory to search in (if not provided, searches in the current directory)

      edit:
        docstring: replaces lines <start_line> through <end_line> (inclusive) with the given text in the open file. The replacement text is terminated by a line with only end_of_edit on it. All of the <replacement text> will be entered, so make sure your indentation is formatted properly. Python files will be checked for syntax errors after the edit. If the system detects a syntax error, the edit will not be executed. Simply try to edit the file again, but make sure to read the error message and modify the edit command you issue accordingly. Issuing the same command a second time will just lead to the same error message again.
        signature: edit <start_line>:<end_line>
      <replacement_text>
      end_of_edit
        arguments:
          - start_line (integer) [required]: the line number to start the edit at
          - end_line (integer) [required]: the line number to end the edit at (inclusive)
          - replacement_text (string) [required]: the text to replace the current selection with

      submit:
        docstring: submits your current flag and terminates the session, you should be aware to properly escape the flag as this is a bash command and to put your flag under single quotes.
        signature: submit '<flag>'
        arguments:
          - flag (string) [required]: the flag found

      decompile:
        docstring: Decompile a binary and prints the decompilation of a given function name
        signature: decompile <binary_path> [--function_name <function_name>]
        arguments:
          - binary_path (file path) [required]: The path to the binary to be decompiled
          - function_name (string) [optional]: The function name to be decompiled, or main by default

      disassemble:
        docstring: Disassembles a binary and prints the disassembly of a given function name
        signature: disassemble <binary_path> [--function_name <function_name>]
        arguments:
          - binary_path (file path) [required]: The path to the binary to be disassembled
          - function_name (string) [optional]: The function name to be disassembled, or main by default

      debug_start:
        docstring: Starts a debug session with the given binary.
        signature: debug_start <binary> [<args>]
        arguments:
          - binary (string) [required]: the path to the binary to debug
          - args (string) [optional]: optional command-line arguments for the binary

      debug_add_breakpoint:
        docstring: Adds a breakpoint in the debug session
        signature: debug_add_breakpoint <breakpoint>
        arguments:
          - breakpoint (string) [required]: The breakpoint location, which may be a function name, address, or filename and line number.

      debug_continue:
        docstring: Continues the program execution in the debug session.
        signature: debug_continue

      debug_step:
        docstring: Steps number of instructions in the debug session.
        signature: debug_step [number]
        arguments:
          - number (integer) [optional]: number of instructions to step (default is 1)

      debug_exec:
        docstring: Executes arbitrary gdb command in debug session.
        signature: debug_exec <command>
        arguments:
          - command (string) [required]: command to execute (wrap in single quotes to avoid shell escaping and substitution)

      debug_stop:
        docstring: Stops the current debug session.
        signature: debug_stop

      connect_start:
        docstring: Starts a new interactive connection to the server address and port.
        signature: connect_start <server_address> <port>
        arguments:
          - server_address (string) [required]: the server address to initiate connection to
          - port (int) [required]: desired port for connection

      connect_sendline:
        docstring: Sends a line with unicode/hexadecimal values to the connection. Sending hexadecimal bytes should be done using \x<hh> where hh stands for the specific byte you want to send.
        signature: connect_sendline [<line>]
        arguments:
          - line (string) [optional]: The line to send to the connection

      connect_exec:
        docstring: Executes arbitrary connect command in connect session.
        signature: connect_exec <command>
        arguments:
          - command (string) [required]: command to execute (wrap in single quotes to avoid shell escaping and substitution)

      connect_stop:
        docstring: Stops the current connect session.
        signature: connect_stop

      exit_forfeit:
        docstring: Give up on the current challenge and terminate the session.
        signature: exit_forfeit

    command_files:
    - /home/talorabr/SWE-agent-ctf/config/commands/defaults.sh
    - /home/talorabr/SWE-agent-ctf/config/commands/search.sh
    - /home/talorabr/SWE-agent-ctf/config/commands/edit_linting.sh
    - /home/talorabr/SWE-agent-ctf/config/commands/_split_string.py
    - /home/talorabr/SWE-agent-ctf/config/commands/submit_ctf.sh
    - /home/talorabr/SWE-agent-ctf/config/commands/decompile.py
    - /home/talorabr/SWE-agent-ctf/config/commands/disassemble.py
    - /home/talorabr/SWE-agent-ctf/config/commands/debug.sh
    - /home/talorabr/SWE-agent-ctf/config/commands/_connect.py
    - /home/talorabr/SWE-agent-ctf/config/commands/server_connection.sh
    - /home/talorabr/SWE-agent-ctf/config/commands/forfeit.sh
    demonstration_template: |
      Here is a demonstration of how to correctly accomplish this task.
      It is included to show you how to correctly use the interface.
      You do not need to follow exactly what is done in the demonstration.
      --- DEMONSTRATION ---
      {demonstration}
      --- END OF DEMONSTRATION ---
    demonstrations:
    - /home/talorabr/SWE-agent-ctf/trajectories/demonstrations/ctf/crypto/katy.traj
    - /home/talorabr/SWE-agent-ctf/trajectories/demonstrations/ctf/crypto/BabyTimeCapsule.traj
    env_variables:
      CURRENT_FILE: ''
      CURRENT_LINE: '0'
      OVERLAP: '2'
      SEARCH_FILES: ()
      SEARCH_INDEX: '0'
      SEARCH_RESULTS: ()
      WINDOW: '100'
    format_error_template: |
      Your output was not formatted correctly. You must always include one discussion and one command as part of your response. Make sure you do not have multiple discussion/command tags.
      Please make sure your output precisely matches the following format:
      DISCUSSION
      Discuss here with yourself about what your planning and what you're going to do in this step.

      ```
      command(s) that you're going to run
      ```
    history_processor: {}
    history_processor_args: {}
    instance_template: |-
      We're currently solving the following CTF challenge. The CTF challenge is a {category_friendly} problem named "{name}", worth {points} points. The description is:
      {description}
      Files included in the challenge: {files}

      {server_description}

      INSTRUCTIONS:
      Now, you're going to solve this challenge on your own. Your terminal session has started and you're in the challenge's root directory. You can use any bash commands or the special interface to help you.
      Solve the challenge, executing any commands you need and writing any code that may be helpful. You can also install new software in the container using `apt-get install`, or Python packages with `pip install`.
      Remember, YOU CAN ONLY ENTER ONE COMMAND AT A TIME. You should always wait for feedback after every command.
      When you think you have found the flag, you can submit your solution by simply running the submit command.
      Note however that you cannot use any interactive session commands (e.g. python, vim) in this environment, but you can write scripts and run them. E.g. you can write a python script and then run it with `python <script_name>.py`.

      NOTE ABOUT THE EDIT COMMAND: Indentation really matters! When editing a file, make sure to insert appropriate indentation before each line!

      IMPORTANT TIPS:
      1. When you edit an existing file, try to minimize the changes you make to the file.

      2. Please make sure that you make use of the interactive tools supplied to you! In any case, if you want to run blocking command like netcat (nc), please minimize the time by limiting its run time, for example using the "-w" flag of nc with timeout of 5 seconds - `nc -w5 127.0.0.1 80`.

      3. Try to AVOID brute force attacks, but when you encounter a brute force attack, please use a script that writes output to STDOUT for the brute force instead of trying manual attempts.

      4. Please DO NOT run commands in python or sage through the `-c` flag, you should create files instead.

      5. Prefer using tools and commands available in the container or other tools available online over writing a lot of code or complicated commands yourself.
         In particular, prefer using `open` instead of `cat` and `search_file` instead of `grep`, and prefer using the interactive commands supplied to you!

      6. DO NOT run any command that produces binary output! Instead try to use `hexdump` or `xxd`.

      7. Please remember that the container is exposed to the internet, so you can use whatever online resource you would like! Use it wisely!

      8. Do not use any interactive commands AT ALL! Interactive commands are only available through the commands supplied to you at the beginning - make use of them!

      9. YOU CAN SOLVE THE CHALLENGES BY YOURSELF COMPLETELY - I TRUST IN YOU!

      (Open file: {open_file})
      (Current directory: {working_dir})
      (Interactive session: {interactive_session})
      bash-$
    next_step_no_output_template: |-
      Your command ran successfully and did not produce any output.
      (Open file: {open_file})
      (Current directory: {working_dir})
      (Interactive session: {interactive_session})
      bash-$
    next_step_template: |-
      {observation}
      (Open file: {open_file})
      (Current directory: {working_dir})
      (Interactive session: {interactive_session})
      bash-$
    parse_command: {}
    parse_function: {}
    put_demos_in_history: false
    review_loop_config: null
    state_command:
      arguments: null
      code: |
        state() {
          local working_dir="$PWD";
          local open_file="n/a";
          local interactive_session="${INTERACTIVE_SESSION:-n/a}";
          if [ ! -z $CURRENT_FILE ]; then
              open_file=$(realpath $CURRENT_FILE);
          fi

          echo '{"open_file": "'$open_file'", "working_dir": "'$working_dir'", "interactive_session": "'$interactive_session'"}'
        };
      docstring: null
      end_name: null
      name: state
      signature: null
    strategy_template: null
    submit_command: submit
    subroutine_types: []
    summarizer_config:
      function: {}
      instance_template: |-
        We're currently solving the following CTF challenge. The CTF challenge is a {category_friendly} problem named "{name}", worth {points} points. The description is:
        {description}

        The last command that was run is:
        ```
        {command}
        ```

        The output of the command is given hereby:
        ```
        {observation}
        ```

        INSTRUCTIONS:
        You need to give a short summary of the output so that your team member who run this command and is solving the challenge will have insights on how to solve the challenge using your summary.
        Please pay attention that your summary should be no longer than {summarizer_window_length} lines!
      model: null
      system_template: |-
        SETTING: You are a skilled cybersecurity professional and top CTF player.
        You are currently playing as part of a team in a CTF competition.

        Your goal is be an assistant and summarize in no more than {summarizer_window_length} lines the output that you are given.
        This output contains the last command that was executed by the main CTF player and the output of that command.

        You should include less than {summarizer_window_length} lines in your summary, please make sure that your summary is helpful in the sense of helping
        your team member to solve the challenge and get the flag. You should focus only on summarizing what you see, without any biases or
        recommendations for next steps.
      template: null
      window_length: 105
    system_template: |-
      SETTING: You are a skilled cybersecurity professional and top CTF player.
      Your goal is to solve the CTF challenge and find the flag. The flag format for this CTF is {flag_format}.

      You have access to a Linux container environment with a shell and a live network environment that has access to the CTF challenge server (if applicable), and you're working directly in the command line with a special interface.
      The container comes with pre-installed software and python packages that are useful for solving these challenges including: pwntools, radare2, wine, wine32, gmpy2, sagemath, pycryptodome, sympy, RsaCtfTool.py, tshark, sqlmap and nikto.

      The special interface consists of a file editor that shows you {WINDOW} lines of a file at a time.
      In addition to typical bash commands, you can also use the following commands to help you navigate and edit files.

      COMMANDS:
      {command_docs}

      Please note that THE EDIT COMMAND REQUIRES PROPER INDENTATION.
      If you'd like to add the line '        print(x)' you must fully write that out, with all those spaces before the code! Indentation is important and code that is not indented correctly will fail and require fixing before it can be run.

      RESPONSE FORMAT:
      Your shell prompt is formatted as follows:
      (Open file: <path>) <cwd> $

      You need to format your output using two fields; discussion and command.
      Your output should always include _one_ discussion and _one_ command field EXACTLY as in the following example:
      DISCUSSION
      First I'll start by using ls to see what files are in the current directory. Then maybe we can look at some relevant files to see what they look like.
      ```
      ls -a
      ```

      You should only include a *SINGLE* command in the command section and then wait for a response from the shell before continuing with more discussion and commands. Everything you include in the DISCUSSION section will be saved for future reference.
      If you'd like to issue two commands at once, PLEASE DO NOT DO THAT! Please instead first submit just the first command, and then after receiving a response you'll be able to issue the second command.
      You're free to use any other bash commands you want (e.g. find, grep, cat, ls, cd) in addition to the special commands listed above.
      However, the environment does NOT support interactive session commands (e.g. python, vim), so please do not invoke them.
    util_functions:
    - arguments: null
      code: '_print() {    local total_lines=$(awk ''END {print NR}'' $CURRENT_FILE)    echo
        "[File: $(realpath $CURRENT_FILE) ($total_lines lines total)]"    lines_above=$(jq
        -n "$CURRENT_LINE - $WINDOW/2" | jq ''[0, .] | max | floor'')    lines_below=$(jq
        -n "$total_lines - $CURRENT_LINE - $WINDOW/2" | jq ''[0, .] | max | round'')    if
        [ $lines_above -gt 0 ]; then        echo "($lines_above more lines above)"    fi    cat
        $CURRENT_FILE | grep -n $ | head -n $(jq -n "[$CURRENT_LINE + $WINDOW/2, $WINDOW/2]
        | max | floor") | tail -n $(jq -n "$WINDOW")    if [ $lines_below -gt 0 ];
        then        echo "($lines_below more lines below)"    fi}'
      docstring: null
      end_name: null
      name: _print
      signature: _print
    - arguments: null
      code: _constrain_line() {    if [ -z "$CURRENT_FILE" ]    then        echo "No
        file open. Use the open command first."        return    fi    local max_line=$(awk
        'END {print NR}' $CURRENT_FILE)    local half_window=$(jq -n "$WINDOW/2" |
        jq 'floor')    export CURRENT_LINE=$(jq -n "[$CURRENT_LINE, $max_line - $half_window]
        | min")    export CURRENT_LINE=$(jq -n "[$CURRENT_LINE, $half_window] | max")}
      docstring: null
      end_name: null
      name: _constrain_line
      signature: _constrain_line
    - arguments: null
      code: '_scroll_warning_message() {    # Warn the agent if we scroll too many
        times    # Message will be shown if scroll is called more than WARN_AFTER_SCROLLING_TIMES
        (default 3) times    # Initialize variable if it''s not set    export SCROLL_COUNT=${SCROLL_COUNT:-0}    #
        Reset if the last command wasn''t about scrolling    if [ "$LAST_ACTION" !=
        "scroll_up" ] && [ "$LAST_ACTION" != "scroll_down" ]; then        export SCROLL_COUNT=0    fi    #
        Increment because we''re definitely scrolling now    export SCROLL_COUNT=$((SCROLL_COUNT
        + 1))    if [ $SCROLL_COUNT -ge ${WARN_AFTER_SCROLLING_TIMES:-3} ]; then        echo
        ""        echo "WARNING: Scrolling many times in a row is very inefficient."        echo
        "If you know what you are looking for, use \`search_file <pattern>\` instead."        echo
        ""    fi}'
      docstring: null
      end_name: null
      name: _scroll_warning_message
      signature: _scroll_warning_message
    - arguments: null
      code: _debug_command() {    echo "<<INTERACTIVE||$@||INTERACTIVE>>"}
      docstring: null
      end_name: null
      name: _debug_command
      signature: _debug_command
    - arguments: null
      code: _connect_command() {    echo "<<INTERACTIVE||$@||INTERACTIVE>>"}
      docstring: null
      end_name: null
      name: _connect_command
      signature: _connect_command
  config_file: config/ctf_crypto.yaml
  model:
    host_url: localhost:11434
    model_name: azure:gpt4
    per_instance_cost_limit: 3.0
    replay_path: null
    temperature: 0.0
    top_p: 0.95
    total_cost_limit: 0.0
environment:
  base_commit: null
  cache_task_images: false
  container_mounts:
  - ghidra_out
  container_name: null
  data_path: tests/test_data/data_sources/ctf/forensics/flash/challenge.json
  environment_setup: null
  image_name: sweagent/swe-ctf:latest
  install_environment: true
  interactive_sessions_config:
    connect:
      cmdline: /root/commands/_connect
      exit_command: connect_stop
      quit_commands_in_session:
      - quit
      signal_for_interrupt_limit: 3
      start_command: connect_start
      terminal_prompt_pattern: '(nc) '
      timeout_duration_on_interrupt: 5
    gdb:
      cmdline: gdb
      exit_command: debug_stop
      quit_commands_in_session:
      - quit
      signal_for_interrupt_limit: 3
      start_command: debug_start
      terminal_prompt_pattern: '(gdb) '
      timeout_duration_on_interrupt: 5
  no_mirror: false
  repo_path: tests/test_data/data_sources/ctf/forensics/flash
  split: dev
  timeout: null
  verbose: true
instance_filter: .*
print_config: false
raise_exceptions: false
skip_existing: false
