|
1 |
| -import openai |
2 | 1 | import os
|
3 | 2 | import typer
|
4 |
| -import nltk |
5 | 3 |
|
6 |
| -app = typer.Typer() |
7 |
| - |
8 |
| - |
9 |
| -def generate_prompt(refactor_or_edit_instructions, code, language): |
10 |
| - """Generate a prompt from the given instructions and code. |
11 |
| -
|
12 |
| - Args: |
13 |
| - refactor_or_edit_instructions (str): Instructions for refactoring or editing the code. |
14 |
| - code (str): The code to be refactored or edited. |
15 |
| - language (str): The language of the code. |
16 |
| -
|
17 |
| - Returns: |
18 |
| - str: The generated prompt. |
19 |
| - """ |
20 |
| - return f""" |
21 |
| - {'Refactor' if 'refactor' in refactor_or_edit_instructions.lower() else 'Edit'} the following {language} code: {refactor_or_edit_instructions} |
22 |
| -
|
23 |
| - Please provide an extremely succinct human explanation of the changes made to the code |
24 |
| - and return the edited code in a new section, delimited by '==='. Don't use '===' other than |
25 |
| - between the sections (don't remove it if it's present though!), and don't add space between sections. |
26 |
| -
|
27 |
| - Ensure that code is well documented and formatted. |
28 |
| - {" Use google docstrings and black formatting."if language == "python" else ""} |
29 |
| -
|
30 |
| - You must explain what you did, even if you don't make a change. |
31 |
| -
|
32 |
| - Code: |
33 |
| - {code}""".strip() |
34 |
| - |
35 |
| - |
36 |
| -def refactor_or_edit( |
37 |
| - file_path: str, |
38 |
| - refactor_or_edit_instructions: str, |
39 |
| - explanation_file: str = None, |
40 |
| - model: str = "text-davinci-003", |
41 |
| - language: str = "python", |
42 |
| - debug: bool = False, |
43 |
| -): |
44 |
| - """Refactor or edit the given file. |
45 |
| -
|
46 |
| - Args: |
47 |
| - file_path (str): The path to the file to be refactored or edited. |
48 |
| - refactor_or_edit_instructions (str): Instructions for refactoring or editing the code. |
49 |
| - explanation_file (str, optional): The path to the file to save the explanation. Defaults to None. |
50 |
| - model (str, optional): GPT-3 model to use. Defaults to "text-davinci-003". |
51 |
| - language (str, optional): The language of the code. Defaults to "python". |
52 |
| - debug (bool, optional): If True, save the response from the model in a JSON file. Defaults to False. |
53 |
| - """ |
54 |
| - openai.api_key = os.getenv("OPENAI_API_KEY") |
55 |
| - |
56 |
| - # open the file and escape the code as a code block |
57 |
| - with open(file_path, "r") as file: |
58 |
| - code = f"```{language}\n" + file.read() + "\n```" |
59 |
| - |
60 |
| - # specify the prompt |
61 |
| - prompt = generate_prompt(refactor_or_edit_instructions, code, language) |
62 |
| - tokens = nltk.word_tokenize(prompt) |
63 |
| - |
64 |
| - #! Yeah this math is BS, closeish though... |
65 |
| - max_tokens = round(4097 - (7 / 4) * len(tokens)) |
66 |
| - |
67 |
| - typer.confirm( |
68 |
| - f"This prompt is {len(tokens)} tokens, are you sure you want to continue?\nThe most GPT-3 can return in response is {max_tokens}.", |
69 |
| - default=True, |
70 |
| - abort=True, |
71 |
| - ) |
72 |
| - |
73 |
| - # send the prompt to the model |
74 |
| - response = openai.Completion.create( |
75 |
| - max_tokens=max_tokens, |
76 |
| - engine=model, |
77 |
| - prompt=prompt, |
78 |
| - n=1, |
79 |
| - stop=None, |
80 |
| - temperature=0.6, |
81 |
| - ) |
82 |
| - |
83 |
| - if debug: |
84 |
| - import json |
85 |
| - |
86 |
| - # write the response |
87 |
| - with open(file_path + ".resp.json", "w") as file: |
88 |
| - file.write(json.dumps(response)) |
89 |
| - |
90 |
| - # print the response from the model |
91 |
| - refactored_code = response["choices"][0]["text"] |
92 |
| - |
93 |
| - explanation = refactored_code.split("===")[0] |
94 |
| - refactored_code = "".join(refactored_code.split("===")[1:]) |
95 |
| - print(explanation) |
| 4 | +from refactor import edit_file |
96 | 5 |
|
97 |
| - old_file_path = file_path + ".old" |
98 |
| - os.rename(file_path, old_file_path) |
99 |
| - |
100 |
| - # write the refactored code to the original file |
101 |
| - with open(file_path, "w") as file: |
102 |
| - file.write(refactored_code) |
103 |
| - |
104 |
| - # write the refactored code to the original file |
105 |
| - with open( |
106 |
| - explanation_file if explanation_file else file_path + ".explained.txt", "w" |
107 |
| - ) as file: |
108 |
| - file.write(explanation) |
| 6 | +app = typer.Typer() |
109 | 7 |
|
110 | 8 |
|
111 | 9 | @app.command()
|
112 |
| -def refactor( |
| 10 | +def unsafe( |
| 11 | + task: str, |
113 | 12 | file_path: str,
|
114 |
| - refactor_instructions: str, |
115 | 13 | explanation_file: str = None,
|
116 | 14 | model: str = "text-davinci-003",
|
117 |
| - language: str = "python", |
118 | 15 | debug: bool = False,
|
119 | 16 | ):
|
120 |
| - """Refactor the given file according to the given instructions. |
121 |
| -
|
122 |
| - Args: |
123 |
| - file_path (str): The path to the file to be refactored. |
124 |
| - refactor_instructions (str): Instructions for refactoring the code. |
125 |
| - explanation_file (str, optional): The path to the file to save the explanation. Defaults to None. |
126 |
| - model (str, optional): GPT-3 model to use. Defaults to "text-davinci-003". |
127 |
| - language (str, optional): The language of the code. Defaults to "python". |
128 |
| - debug (bool, optional): If True, save the response from the model in a JSON file. Defaults to False. |
129 | 17 | """
|
130 |
| - refactor_or_edit( |
131 |
| - file_path, refactor_instructions, explanation_file, model, language, debug |
132 |
| - ) |
| 18 | + Refactor the given file according to the specified task. |
133 | 19 |
|
| 20 | + :param task: is the most important - The task to perform. |
134 | 21 |
|
135 |
| -@app.command() |
136 |
| -def varnames( |
137 |
| - file_path: str, |
138 |
| - refactor_instructions: str = "In the following code, rename variables as you see appropriate for it to be easier to read. Don't touch any of the code otherwise, other than to update comments.", |
139 |
| - explanation_file: str = None, |
140 |
| - model: str = "text-davinci-003", |
141 |
| - language: str = "python", |
142 |
| - debug: bool = False, |
143 |
| -): |
144 |
| - """Refactor the given file to rename variables as appropriate. |
| 22 | + "varnames" : Changes variable names to be more human readable. |
| 23 | + "comment" : Adds comments to the code, no other changes |
| 24 | + "edit" : A raw interface to send a prompt to gpt3. "Break this up into more functions" or something is fun to try! |
145 | 25 |
|
146 | 26 | Args:
|
| 27 | + task (str): The type of refactoring to perform on the file. Can be "varnames", "comment", or "edit". |
147 | 28 | file_path (str): The path to the file to be refactored.
|
148 |
| - refactor_instructions (str): Instructions for refactoring the code. |
149 | 29 | explanation_file (str, optional): The path to the file to save the explanation. Defaults to None.
|
150 | 30 | model (str, optional): GPT-3 model to use. Defaults to "text-davinci-003".
|
151 |
| - language (str, optional): The language of the code. Defaults to "python". |
152 | 31 | debug (bool, optional): If True, save the response from the model in a JSON file. Defaults to False.
|
153 | 32 | """
|
154 |
| - refactor_or_edit( |
155 |
| - file_path, refactor_instructions, explanation_file, model, language, debug |
156 |
| - ) |
157 |
| - |
158 |
| - |
159 |
| -@app.command() |
160 |
| -def comment( |
161 |
| - file_path: str, |
162 |
| - refactor_instructions: str = "In the following code, make no code changes but add comments. Keep them succinct, but explain everything you can if it's helpful. Add function or class strings where you can.", |
163 |
| - explanation_file: str = None, |
164 |
| - model: str = "text-davinci-003", |
165 |
| - language: str = "python", |
166 |
| - debug: bool = False, |
167 |
| -): |
168 |
| - """Edit the given file to add comments.""" |
169 |
| - refactor_or_edit( |
170 |
| - file_path, refactor_instructions, explanation_file, model, language, debug |
171 |
| - ) |
172 |
| - |
173 |
| - |
174 |
| -@app.command() |
175 |
| -def edit( |
176 |
| - file_path: str, |
177 |
| - edit_instructions: str, |
178 |
| - explanation_file: str = None, |
179 |
| - model: str = "text-davinci-003", |
180 |
| - language: str = "md", |
181 |
| - debug: bool = False, |
182 |
| -): |
183 |
| - """A Generic edit option, meant for editing markdown blog posts. Basically refactor with some extra instructions.""" |
184 |
| - refactor_or_edit( |
185 |
| - file_path, edit_instructions, explanation_file, model, language, debug |
186 |
| - ) |
| 33 | + # Dictionary of instructions for each refactoring task |
| 34 | + refactor_tasks = { |
| 35 | + "varnames": "In the following code, rename variables as you see appropriate for it to be easier to read. Don't touch any of the code otherwise, other than to update comments.", |
| 36 | + "comment": "In the following code, make no code changes but add comments. Keep them succinct, but explain everything you can if it's helpful. Add function or class strings where you can.", |
| 37 | + "edit": "", |
| 38 | + } |
| 39 | + # If the task is "edit", prompt the user for instructions |
| 40 | + if task == "edit": |
| 41 | + refactor_instructions = typer.prompt( |
| 42 | + "Please provide the instructions for editing the file: " |
| 43 | + ) |
| 44 | + # Otherwise, look up the instructions in the dictionary |
| 45 | + else: |
| 46 | + refactor_instructions = refactor_tasks[task] |
| 47 | + # Call the edit_file function with the specified instructions |
| 48 | + edit_file(file_path, refactor_instructions, explanation_file, model, debug) |
187 | 49 |
|
188 | 50 |
|
189 | 51 | @app.command()
|
|
0 commit comments