from pathlib import Path

from lita.core.utils import read_file

from .tools import CodeActTool

_FILE_EDIT_DESCRIPTION = """Edit a file in plain-text format.
* The assistant can edit files by specifying the file path and providing a draft of the new file content.
* The draft content doesn't need to be exactly the same as the existing file; \
the assistant may skip unchanged lines using comments like `# ... existing code ...` to indicate unchanged sections.
* IMPORTANT: For large files (e.g., > 300 lines), \
specify the range of lines to edit using `start` and `end` (1-indexed, inclusive). \
The range should be smaller than 300 lines.
* -1 indicates the last line of the file when used as the `start` or `end` value.
* Keep at least one unchanged line before the changed section and after the changed section wherever possible.
* Make sure to set the `start` and `end` to include all the lines in the original file \
referred to in the draft of the new file content. Failure to do so will result in bad edits.
* To append to a file, set both `start` and `end` to `-1`.
* If the file doesn't exist, a new file will be created with the provided content.
* IMPORTANT: Make sure you include all the required indentations for each line of code in the draft, \
otherwise the edited code will be incorrectly indented.
* IMPORTANT: Make sure that the first line of the draft is also properly indented and has the required whitespaces.
* IMPORTANT: NEVER include or make references to lines from outside the `start` and `end` range in the draft.
* IMPORTANT: Start the content with a comment in the format: #EDIT: Reason for edit
* IMPORTANT: If you are not appending to the file, avoid setting `start` and `end` to the same value.

**Example 1: general edit for short files**
For example, given an existing file `/path/to/file.py` that looks like this:
(this is the beginning of the file)
1|class MyClass:
2|    def __init__(self):
3|        self.x = 1
4|        self.y = 2
5|        self.z = 3
6|
7|print(MyClass().z)
8|print(MyClass().x)
(this is the end of the file)

The assistant wants to edit the file to look like this:
(this is the beginning of the file)
1|class MyClass:
2|    def __init__(self):
3|        self.x = 1
4|        self.y = 2
5|
6|print(MyClass().y)
(this is the end of the file)

The assistant may produce an edit action like this:
path="/path/to/file.txt" start=1 end=-1
content=```
#EDIT: I want to change the value of y to 2
class MyClass:
    def __init__(self):
        # ... existing code ...
        self.y = 2

print(MyClass().y)
```

**Example 2: append to file for short files**
For example, given an existing file `/path/to/file.py` that looks like this:
(this is the beginning of the file)
1|class MyClass:
2|    def __init__(self):
3|        self.x = 1
4|        self.y = 2
5|        self.z = 3
6|
7|print(MyClass().z)
8|print(MyClass().x)
(this is the end of the file)

To append the following lines to the file:
```python
#EDIT: I want to print the value of y
print(MyClass().y)
```

The assistant may produce an edit action like this:
path="/path/to/file.txt" start=-1 end=-1
content=```
print(MyClass().y)
```

**Example 3: edit for long files**

Given an existing file `/path/to/file.py` that looks like this:
(1000 more lines above)
1001|class MyClass:
1002|    def __init__(self):
1003|        self.x = 1
1004|        self.y = 2
1005|        self.z = 3
1006|
1007|print(MyClass().z)
1008|print(MyClass().x)
(2000 more lines below)

The assistant wants to edit the file to look like this:

(1000 more lines above)
1001|class MyClass:
1002|    def __init__(self):
1003|        self.x = 1
1004|        self.y = 2
1005|
1006|print(MyClass().y)
(2000 more lines below)

The assistant may produce an edit action like this:
path="/path/to/file.txt" start=1002 end=1008
content=```
#EDIT: I want to change the value of y to 2
    def __init__(self):
        # no changes before
        self.y = 2
        # self.z is removed

# MyClass().z is removed
print(MyClass().y)
```
"""


# We implement our own version. The original one is here
# openhands/runtime/action_execution_server.py: L560
class LLMBasedFileEditor(CodeActTool):
    name: str = 'edit_file'
    description: str = _FILE_EDIT_DESCRIPTION
    parameters: dict = {
        'type': 'object',
        'properties': {
            'path': {
                'type': 'string',
                'description': 'The absolute path to the file to be edited.',
            },
            'content': {
                'type': 'string',
                'description': 'A draft of the new content for the file being edited. '
                + 'Note that the assistant may skip unchanged lines.',
            },
            'start': {
                'type': 'integer',
                'description': 'The starting line number for the edit (1-indexed, inclusive). Default is 1.',
            },
            'end': {
                'type': 'integer',
                'description': 'The ending line number for the edit (1-indexed, inclusive). '
                + 'Default is -1 (end of file).',
            },
        },
        'required': ['path', 'content'],
    }

    def execute(self, **kwargs) -> str:
        path = kwargs.get("path")
        start = kwargs.get("start", 1)
        end = kwargs.get("end", -1)
        content = kwargs.get("content", "")

        # We only have an `edit` operation here
        result = self.edit(path, content, start, end)

        return result

    def edit(self, path: str, content: str, start: int, end: int) -> str:
        # Early return
        if start < 1:
            return "`start` argument should be 1 or greater."
        if end < -1:
            return "`end` argument should be -1 or greater."
        if end < start:
            return "`end` argument should be no less than `start`."

        _path = Path(path)
        if not _path.exists():
            return f"{self.name}: File not found: {path}"

        try:
            file_content = read_file(_path, splitlines=True)

            num_lines = len(file_content)
            start -= 1
            end -= 1
            if start >= num_lines or end >= num_lines:
                return f"`start` or `end` out of range. The file only has {num_lines} lines."

            # Not intersected
            before = file_content[:start]
            after = file_content[end + 1 :]
            new_content = "".join(before) + content + "".join(after)

            _path.write_text(new_content)
            return "Edited the file."
        except Exception as e:
            return f"{self.name}: Error editing file: {e}"
