Skip to content

Commit b4ac468

Browse files
committed
Close #2, break whole api, put unsafe stuff in unsafe route
1 parent a039c60 commit b4ac468

File tree

4 files changed

+147
-170
lines changed

4 files changed

+147
-170
lines changed

README.md

+13-5
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,10 @@
22

33
A tool for using GPT just a little quicker. A nearly truly automated footgun. Learn how to revert with git before trying please.
44

5+
Posting about progress here:
6+
7+
[![Twitter Follow](https://img.shields.io/twitter/follow/_JohnPartee?style=social)](https://twitter.com/_JohnPartee)
8+
59
# Getting Started
610

711
`pip install codegpt`
@@ -15,9 +19,13 @@ Windows users can also use `setx` like:
1519

1620
from an admin console.
1721

18-
Then find a file you hate (Back it up! Don't do it live!) and give it a shot.
22+
## Be careful! But let's go.
23+
24+
The fun stuff is in the `unsafe` command.
25+
26+
Find a file you hate (Back it up! Don't do it live!) and give it a shot.
1927

20-
`codegpt refactor .\helper.py "Break this up into smaller functions where you can. Add google style docstrings. Feel free to rewrite any code doesn't make sense."`
28+
`codegpt unsafe edit .\helper.py "Break this up into smaller functions where you can. Add google style docstrings. Feel free to rewrite any code doesn't make sense."`
2129

2230
You'll see something like:
2331

@@ -32,9 +40,9 @@ Explanation: The code has been refactored into smaller functions to improve read
3240

3341
Other things to try:
3442

35-
- `codegpt edit` - For editing markdown files, including code blocks. Hello, blog editor!
36-
- `codegpt varnames` - Changes variable names (and supposed to only be variable names...) to be readable
37-
- `codegpt comment` - Automatically add comments to a file.
43+
- `codegpt unsafe edit` - Try it with anything. Markdown blog posts, js, yaml, python, whatever.
44+
- `codegpt unsafe varnames` - Changes variable names (and supposed to only be variable names...) to be readable
45+
- `codegpt unsafe comment` - Automatically add comments to a file.
3846

3947
Propose endpoints as issues, I've got a few ideas:
4048

codegpt/codegpt.py

+26-164
Original file line numberDiff line numberDiff line change
@@ -1,189 +1,51 @@
1-
import openai
21
import os
32
import typer
4-
import nltk
53

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
965

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()
1097

1108

1119
@app.command()
112-
def refactor(
10+
def unsafe(
11+
task: str,
11312
file_path: str,
114-
refactor_instructions: str,
11513
explanation_file: str = None,
11614
model: str = "text-davinci-003",
117-
language: str = "python",
11815
debug: bool = False,
11916
):
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.
12917
"""
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.
13319
20+
:param task: is the most important - The task to perform.
13421
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!
14525
14626
Args:
27+
task (str): The type of refactoring to perform on the file. Can be "varnames", "comment", or "edit".
14728
file_path (str): The path to the file to be refactored.
148-
refactor_instructions (str): Instructions for refactoring the code.
14929
explanation_file (str, optional): The path to the file to save the explanation. Defaults to None.
15030
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".
15231
debug (bool, optional): If True, save the response from the model in a JSON file. Defaults to False.
15332
"""
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)
18749

18850

18951
@app.command()

codegpt/refactor.py

+107
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,107 @@
1+
import nltk
2+
import openai
3+
import os
4+
import typer
5+
6+
7+
def edit_file(
8+
file_path: str,
9+
refactor_or_edit_instructions: str,
10+
explanation_file: str = None,
11+
model: str = "text-davinci-003",
12+
debug: bool = False,
13+
):
14+
"""Edit the given file.
15+
16+
Args:
17+
file_path (str): The path to the file to be refactored or edited.
18+
refactor_or_edit_instructions (str): Instructions for refactoring or editing the code.
19+
explanation_file (str, optional): The path to the file to save the explanation. Defaults to None.
20+
model (str, optional): GPT-3 model to use. Defaults to "text-davinci-003".
21+
language (str, optional): The language of the code. Defaults to "python".
22+
debug (bool, optional): If True, save the response from the model in a JSON file. Defaults to False.
23+
"""
24+
openai.api_key = os.getenv("OPENAI_API_KEY")
25+
26+
language = file_path.split(".")[-1]
27+
28+
# open the file and escape the code as a code block
29+
with open(file_path, "r") as file:
30+
code = f"###{file_path}\n```{language}\n" + file.read() + "\n```"
31+
32+
# specify the prompt
33+
prompt = generate_prompt(refactor_or_edit_instructions, code, language)
34+
tokens = nltk.word_tokenize(prompt)
35+
36+
#! Yeah this math is BS, closeish though...
37+
max_tokens = round(4097 - (7 / 4) * len(tokens))
38+
39+
typer.confirm(
40+
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}.",
41+
default=True,
42+
abort=True,
43+
)
44+
45+
# send the prompt to the model
46+
response = openai.Completion.create(
47+
max_tokens=max_tokens,
48+
engine=model,
49+
prompt=prompt,
50+
n=1,
51+
stop=None,
52+
temperature=0.6,
53+
)
54+
55+
if debug:
56+
import json
57+
58+
# write the response
59+
with open(file_path + ".resp.json", "w") as file:
60+
file.write(json.dumps(response))
61+
62+
# print the response from the model
63+
refactored_code = response["choices"][0]["text"]
64+
65+
explanation = refactored_code.split("===")[0]
66+
refactored_code = "".join(refactored_code.split("===")[1:])
67+
print(explanation)
68+
69+
old_file_path = file_path + ".old"
70+
os.rename(file_path, old_file_path)
71+
72+
# write the refactored code to the original file
73+
with open(file_path, "w") as file:
74+
file.write(refactored_code)
75+
76+
# write the refactored code to the original file
77+
with open(
78+
explanation_file if explanation_file else file_path + ".explained.txt", "w"
79+
) as file:
80+
file.write(explanation)
81+
82+
83+
def generate_prompt(refactor_or_edit_instructions, code, language):
84+
"""Generate a prompt from the given instructions and code.
85+
86+
Args:
87+
refactor_or_edit_instructions (str): Instructions for refactoring or editing the code.
88+
code (str): The code to be refactored or edited.
89+
language (str): The language of the code.
90+
91+
Returns:
92+
str: The generated prompt.
93+
"""
94+
return f"""
95+
{'Refactor' if 'refactor' in refactor_or_edit_instructions.lower() else 'Edit'} the code below, to: {refactor_or_edit_instructions}
96+
97+
Write a short summary of the changes made to the code and return the edited code in a new section,
98+
delimited by '==='. Don't use '===' other than between the sections, and don't add space between sections.
99+
100+
You are an expert, ensure that code is technically correct, well documented and formatted.
101+
{"Use google docstrings and black formatting."if language == "py" else ""}
102+
{"This is likely an article or blog post, maintain the authors tone and voice while you edit." if language == "md" else ""}
103+
104+
You must explain what you did, even if you don't make a change.
105+
106+
Code:
107+
{code}""".strip()

pyproject.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
[tool.poetry]
22
name = "codegpt"
3-
version = "0.0.7"
3+
version = "0.0.7-8"
44
license = "GPL-3.0-or-later"
55
description = "A CLI tool for refactoring Python code using OpenAI's text-davinci-003 model"
66
authors = ["John Partee"]

0 commit comments

Comments
 (0)