mirror of
https://github.com/Aider-AI/aider.git
synced 2025-06-02 10:45:00 +00:00
196 lines
5.7 KiB
Python
196 lines
5.7 KiB
Python
import os
|
|
from pygments.lexers import guess_lexer_for_filename
|
|
from pygments.token import Token
|
|
from prompt_toolkit.styles import Style
|
|
from pygments.util import ClassNotFound
|
|
from prompt_toolkit import prompt
|
|
from prompt_toolkit.completion import Completer, Completion
|
|
from prompt_toolkit.history import FileHistory
|
|
from prompt_toolkit.shortcuts import CompleteStyle
|
|
from rich.console import Console
|
|
from rich.text import Text
|
|
import sys
|
|
import time
|
|
import random
|
|
from pathlib import Path
|
|
|
|
|
|
class FileContentCompleter(Completer):
|
|
def __init__(self, fnames, commands):
|
|
self.commands = commands
|
|
|
|
self.words = set()
|
|
for fname in fnames:
|
|
with open(fname, "r") as f:
|
|
content = f.read()
|
|
try:
|
|
lexer = guess_lexer_for_filename(fname, content)
|
|
except ClassNotFound:
|
|
continue
|
|
tokens = list(lexer.get_tokens(content))
|
|
self.words.update(token[1] for token in tokens if token[0] in Token.Name)
|
|
|
|
def get_completions(self, document, complete_event):
|
|
text = document.text_before_cursor
|
|
words = text.split()
|
|
if not words:
|
|
return
|
|
|
|
if text[0] == "/":
|
|
if len(words) == 1 and not text[-1].isspace():
|
|
candidates = self.commands.get_commands()
|
|
else:
|
|
for completion in self.commands.get_command_completions(words[0][1:], words[-1]):
|
|
yield completion
|
|
return
|
|
else:
|
|
candidates = self.words
|
|
|
|
last_word = words[-1]
|
|
for word in candidates:
|
|
if word.lower().startswith(last_word.lower()):
|
|
yield Completion(word, start_position=-len(last_word))
|
|
|
|
|
|
class InputOutput:
|
|
def __init__(self, pretty, yes, input_history_file, chat_history_file):
|
|
self.pretty = pretty
|
|
self.yes = yes
|
|
self.input_history_file = input_history_file
|
|
self.chat_history_file = Path(chat_history_file)
|
|
|
|
if pretty:
|
|
self.console = Console()
|
|
else:
|
|
self.console = Console(force_terminal=True, no_color=True)
|
|
|
|
self.append_chat_history('# New session started', False)
|
|
|
|
def canned_input(self, show_prompt):
|
|
console = Console()
|
|
|
|
input_line = input()
|
|
|
|
console.print(show_prompt, end="", style="green")
|
|
for char in input_line:
|
|
console.print(char, end="", style="green")
|
|
time.sleep(random.uniform(0.01, 0.15))
|
|
console.print()
|
|
console.print()
|
|
return input_line
|
|
|
|
def get_input(self, fnames, commands):
|
|
if self.pretty:
|
|
self.console.rule()
|
|
else:
|
|
print()
|
|
|
|
fnames = list(fnames)
|
|
if len(fnames) > 1:
|
|
common_prefix = os.path.commonpath(fnames)
|
|
if not common_prefix.endswith(os.path.sep):
|
|
common_prefix += os.path.sep
|
|
short_fnames = [fname.replace(common_prefix, "", 1) for fname in fnames]
|
|
elif len(fnames):
|
|
short_fnames = [os.path.basename(fnames[0])]
|
|
else:
|
|
short_fnames = []
|
|
|
|
show = " ".join(short_fnames)
|
|
if len(show) > 10:
|
|
show += "\n"
|
|
show += "> "
|
|
|
|
if not sys.stdin.isatty():
|
|
return self.canned_input(show)
|
|
|
|
inp = ""
|
|
multiline_input = False
|
|
|
|
style = Style.from_dict({"": "green"})
|
|
|
|
while True:
|
|
completer_instance = FileContentCompleter(fnames, commands)
|
|
if multiline_input:
|
|
show = ". "
|
|
|
|
line = prompt(
|
|
show,
|
|
completer=completer_instance,
|
|
history=FileHistory(self.input_history_file),
|
|
style=style,
|
|
reserve_space_for_menu=4,
|
|
complete_style=CompleteStyle.MULTI_COLUMN,
|
|
)
|
|
if line.strip() == "{" and not multiline_input:
|
|
multiline_input = True
|
|
continue
|
|
elif line.strip() == "}" and multiline_input:
|
|
break
|
|
elif multiline_input:
|
|
inp += line + "\n"
|
|
else:
|
|
inp = line
|
|
break
|
|
|
|
print()
|
|
|
|
prefix = "####"
|
|
hist = inp.splitlines()
|
|
hist = f" \n{prefix} ".join(hist)
|
|
|
|
hist = f"""---
|
|
{prefix} {hist}"""
|
|
self.append_chat_history(hist, True)
|
|
|
|
return inp
|
|
|
|
# OUTPUT
|
|
|
|
def ai_output(self, content):
|
|
hist = '\n' + content.strip() + '\n\n'
|
|
self.append_chat_history(hist, False)
|
|
|
|
def confirm_ask(self, question, default="y"):
|
|
if self.yes:
|
|
res = "yes"
|
|
else:
|
|
res = prompt(question + " ", default=default)
|
|
|
|
hist = f"*{question.strip()} {res}*"
|
|
self.append_chat_history(hist, True)
|
|
|
|
if not res:
|
|
return
|
|
return res.lower().startswith("y")
|
|
|
|
def prompt_ask(self, question, default=None):
|
|
if self.yes:
|
|
return True
|
|
return prompt(question + " ", default=default)
|
|
|
|
def tool_error(self, message):
|
|
if message.strip():
|
|
hist = f"*{message.strip()}*"
|
|
self.append_chat_history(hist, True)
|
|
|
|
message = Text(message)
|
|
self.console.print(message, style="red")
|
|
|
|
def tool(self, *messages):
|
|
if messages:
|
|
hist = " ".join(messages)
|
|
hist = f"*{hist.strip()}*"
|
|
self.append_chat_history(hist, True)
|
|
|
|
messages = list(map(Text, messages))
|
|
self.console.print(*messages)
|
|
|
|
def append_chat_history(self, text, linebreak):
|
|
if linebreak:
|
|
text = text.rstrip()
|
|
text = text + " \n"
|
|
if not text.endswith('\n'):
|
|
text += '\n'
|
|
with self.chat_history_file.open("a") as f:
|
|
f.write(text)
|