Skip to content

Commit 5da2eab

Browse files
Merge pull request #2 from hasura/rob/feature/use-json-format
Feature: use OpenAI JSON mode (closes #1)
2 parents 7150a7b + b9036c8 commit 5da2eab

File tree

7 files changed

+183
-55
lines changed

7 files changed

+183
-55
lines changed

__tests__/openai.test.ts

+10-7
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { testConnection, generatePrompt, testAssertion, writeAnalysis } from '../src/open_ai';
1+
import { testConnection, generatePrompt, testAssertion, writeAnalysis, openAiFeedback } from '../src/open_ai';
22
import { getDiff, getSinglePR, getAssertion, getChangedFiles, getFileContent } from '../src/github';
33

44
describe('OpenAI Functionality', () => {
@@ -27,13 +27,16 @@ describe('OpenAI Functionality', () => {
2727
const file: any = await getFileContent(changedFiles, 'hasura', 'v3-docs');
2828
const prompt: string = generatePrompt(diff, assertion, file);
2929
const response = await testAssertion(prompt);
30+
console.log(response);
3031
expect(response).toBeTruthy();
31-
}, 50000);
32+
}, 100000);
3233
it('Should create a nicely formatted message using the response', async () => {
33-
expect(
34-
writeAnalysis(
35-
`[{"satisfied": "\u2705", "scope": "diff", "feedback": "You did a great job!"}, {"satisfied": "\u2705", "scope": "wholeFile", "feedback": "Look. At. You. Go!"}]`
36-
)
37-
).toContain('You did a great job!');
34+
const feedback: openAiFeedback = {
35+
feedback: [
36+
{ satisfied: '\u2705', scope: 'Diff', feedback: 'You did a great job!' },
37+
{ satisfied: '\u2705', scope: 'Integrated', feedback: 'Look. At. You. Go!' },
38+
],
39+
};
40+
expect(writeAnalysis(feedback)).toContain('You did a great job!');
3841
});
3942
});

bable.config.js babel.config.js

File renamed without changes.

dist/index.js

+53-15
Original file line numberDiff line numberDiff line change
@@ -219,7 +219,7 @@ async function main() {
219219
const file = await (0, github_1.getFileContent)(changedFiles, org, repoName);
220220
const prompt = (0, open_ai_1.generatePrompt)(diff, assertion, file);
221221
const rawAnalysis = await (0, open_ai_1.testAssertion)(prompt);
222-
const analysis = (0, open_ai_1.writeAnalysis)(rawAnalysis?.toString() ?? '');
222+
const analysis = (0, open_ai_1.writeAnalysis)(rawAnalysis);
223223
console.log(analysis);
224224
core.setOutput('analysis', analysis);
225225
return analysis;
@@ -273,7 +273,42 @@ const openai = new openAi({
273273
});
274274
// This wil generate our prompt using the diff, assertion, and whole file
275275
const generatePrompt = (diff, assertion, file) => {
276-
const comboPrompt = `As a senior engineer, you're tasked with reviewing a documentation PR. Your review will be conducted through two distinct lenses, both centered around an assertion related to usability. The first lens will focus on examining the diff itself — providing targeted feedback on what the PR author actually contributed. The second lens will compare the diff to the entire set of changed files, assessing how the contribution fits within the larger context in relation to the usability assertion. For each lens, provide feedback and determine if the usability assertion is satisfied. You should speak directly to the author and refer to them in second person. Your output should be a JSON-formatted array with two objects. Each object should contain the following properties: 'satisfied' (either a ✅ or ❌ to indicate if the assertion is met), 'scope' (either 'Diff' or 'Integrated'), and 'feedback' (a string providing your targeted feedback for that lens). Here's the assertion: ${assertion}\n\nHere's the diff:\n\n${diff}\n\nHere's the original files:\n\n${file}\n\nBear in mind that some of the files may have been renamed. Remember, do not wrap the JSON in a code block.`;
276+
const comboPrompt = `As a senior engineer, you're tasked with reviewing a documentation PR. Your review comprises two distinct perspectives, each focused on a specific aspect of usability.
277+
278+
- **First Perspective**: Examine the PR's diff. Provide targeted feedback on the author's contribution.
279+
- **Second Perspective**: Assess how the diff integrates with the entire set of changed files, evaluating its contribution to the overall usability.
280+
281+
**Usability Assertion**: ${assertion}
282+
283+
**PR Diff**: ${diff}
284+
285+
**Original Files**: ${file}
286+
287+
(Note: Some files may have been renamed.)
288+
289+
**Your Task**: Provide feedback for each perspective. Determine if the usability assertion is met in each context.
290+
291+
**Output Format**: Your response should be a JSON-formatted array containing exactly two objects. Each object must have the following properties:
292+
- 'satisfied': Indicate if the assertion is met (✅ for yes, ❌ for no).
293+
- 'scope': 'Diff' for the first perspective, 'Integrated' for the second.
294+
- 'feedback': A string providing your targeted feedback.
295+
296+
Example Output:
297+
{
298+
"feedback": [
299+
{
300+
"satisfied": "✅",
301+
"scope": "Diff",
302+
"feedback": "Your changes in the PR are clear and enhance the readability of the documentation."
303+
},
304+
{
305+
"satisfied": "❌",
306+
"scope": "Integrated",
307+
"feedback": "The changes do not align well with the overall structure and flow of the existing documentation."
308+
}
309+
]
310+
}
311+
`;
277312
return comboPrompt;
278313
};
279314
exports.generatePrompt = generatePrompt;
@@ -302,10 +337,12 @@ const testAssertion = async (prompt) => {
302337
const chatCompletion = await openai.chat.completions.create({
303338
model: 'gpt-4-1106-preview',
304339
messages: conversation,
340+
response_format: { type: 'json_object' },
305341
});
306342
const analysis = chatCompletion.choices[0].message.content;
307343
console.log(`✅ Got analysis from OpenAI`);
308-
return analysis;
344+
const parsedAnalysis = JSON.parse(analysis);
345+
return parsedAnalysis;
309346
}
310347
catch (error) {
311348
console.error(error);
@@ -315,18 +352,19 @@ const testAssertion = async (prompt) => {
315352
exports.testAssertion = testAssertion;
316353
// We decided to send things back as JSON so we can manipulate the data in the response we'll be sending back to GitHub
317354
const writeAnalysis = (analysis) => {
318-
// We've still got to double-check because ChatGPT will sometimes return a string that's not valid JSON by wrapping it in code blocks
319-
const regex = /^```(json)?/gm;
320-
analysis = analysis.replace(regex, '');
321-
const analysisJSON = JSON.parse(analysis);
322-
let message = `## DX: Assertion Testing\n\n`;
323-
const feedback = analysisJSON.map((item) => {
324-
// we'll create some markdown to make the feedback look nice
325-
return `### ${item.satisfied} ${item.scope}\n\n${item.feedback}\n\n`;
326-
});
327-
feedback.unshift(message);
328-
const feedbackString = feedback.join('');
329-
return feedbackString;
355+
if (analysis === null) {
356+
return `Error testing the assertions. Check the logs.`;
357+
}
358+
else {
359+
let message = `## DX: Assertion Testing\n\n`;
360+
const feedback = analysis.feedback.map((item) => {
361+
// we'll create some markdown to make the feedback look nice
362+
return `### ${item.satisfied} ${item.scope}\n\n${item.feedback}\n\n`;
363+
});
364+
feedback.unshift(message);
365+
const feedbackString = feedback.join('');
366+
return feedbackString;
367+
}
330368
};
331369
exports.writeAnalysis = writeAnalysis;
332370

dist/open_ai/index.js

+52-14
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,42 @@ const openai = new openAi({
3737
});
3838
// This wil generate our prompt using the diff, assertion, and whole file
3939
const generatePrompt = (diff, assertion, file) => {
40-
const comboPrompt = `As a senior engineer, you're tasked with reviewing a documentation PR. Your review will be conducted through two distinct lenses, both centered around an assertion related to usability. The first lens will focus on examining the diff itself — providing targeted feedback on what the PR author actually contributed. The second lens will compare the diff to the entire set of changed files, assessing how the contribution fits within the larger context in relation to the usability assertion. For each lens, provide feedback and determine if the usability assertion is satisfied. You should speak directly to the author and refer to them in second person. Your output should be a JSON-formatted array with two objects. Each object should contain the following properties: 'satisfied' (either a ✅ or ❌ to indicate if the assertion is met), 'scope' (either 'Diff' or 'Integrated'), and 'feedback' (a string providing your targeted feedback for that lens). Here's the assertion: ${assertion}\n\nHere's the diff:\n\n${diff}\n\nHere's the original files:\n\n${file}\n\nBear in mind that some of the files may have been renamed. Remember, do not wrap the JSON in a code block.`;
40+
const comboPrompt = `As a senior engineer, you're tasked with reviewing a documentation PR. Your review comprises two distinct perspectives, each focused on a specific aspect of usability.
41+
42+
- **First Perspective**: Examine the PR's diff. Provide targeted feedback on the author's contribution.
43+
- **Second Perspective**: Assess how the diff integrates with the entire set of changed files, evaluating its contribution to the overall usability.
44+
45+
**Usability Assertion**: ${assertion}
46+
47+
**PR Diff**: ${diff}
48+
49+
**Original Files**: ${file}
50+
51+
(Note: Some files may have been renamed.)
52+
53+
**Your Task**: Provide feedback for each perspective. Determine if the usability assertion is met in each context.
54+
55+
**Output Format**: Your response should be a JSON-formatted array containing exactly two objects. Each object must have the following properties:
56+
- 'satisfied': Indicate if the assertion is met (✅ for yes, ❌ for no).
57+
- 'scope': 'Diff' for the first perspective, 'Integrated' for the second.
58+
- 'feedback': A string providing your targeted feedback.
59+
60+
Example Output:
61+
{
62+
"feedback": [
63+
{
64+
"satisfied": "✅",
65+
"scope": "Diff",
66+
"feedback": "Your changes in the PR are clear and enhance the readability of the documentation."
67+
},
68+
{
69+
"satisfied": "❌",
70+
"scope": "Integrated",
71+
"feedback": "The changes do not align well with the overall structure and flow of the existing documentation."
72+
}
73+
]
74+
}
75+
`;
4176
return comboPrompt;
4277
};
4378
exports.generatePrompt = generatePrompt;
@@ -66,10 +101,12 @@ const testAssertion = async (prompt) => {
66101
const chatCompletion = await openai.chat.completions.create({
67102
model: 'gpt-4-1106-preview',
68103
messages: conversation,
104+
response_format: { type: 'json_object' },
69105
});
70106
const analysis = chatCompletion.choices[0].message.content;
71107
console.log(`✅ Got analysis from OpenAI`);
72-
return analysis;
108+
const parsedAnalysis = JSON.parse(analysis);
109+
return parsedAnalysis;
73110
}
74111
catch (error) {
75112
console.error(error);
@@ -79,17 +116,18 @@ const testAssertion = async (prompt) => {
79116
exports.testAssertion = testAssertion;
80117
// We decided to send things back as JSON so we can manipulate the data in the response we'll be sending back to GitHub
81118
const writeAnalysis = (analysis) => {
82-
// We've still got to double-check because ChatGPT will sometimes return a string that's not valid JSON by wrapping it in code blocks
83-
const regex = /^```(json)?/gm;
84-
analysis = analysis.replace(regex, '');
85-
const analysisJSON = JSON.parse(analysis);
86-
let message = `## DX: Assertion Testing\n\n`;
87-
const feedback = analysisJSON.map((item) => {
88-
// we'll create some markdown to make the feedback look nice
89-
return `### ${item.satisfied} ${item.scope}\n\n${item.feedback}\n\n`;
90-
});
91-
feedback.unshift(message);
92-
const feedbackString = feedback.join('');
93-
return feedbackString;
119+
if (analysis === null) {
120+
return `Error testing the assertions. Check the logs.`;
121+
}
122+
else {
123+
let message = `## DX: Assertion Testing\n\n`;
124+
const feedback = analysis.feedback.map((item) => {
125+
// we'll create some markdown to make the feedback look nice
126+
return `### ${item.satisfied} ${item.scope}\n\n${item.feedback}\n\n`;
127+
});
128+
feedback.unshift(message);
129+
const feedbackString = feedback.join('');
130+
return feedbackString;
131+
}
94132
};
95133
exports.writeAnalysis = writeAnalysis;

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
"build": "tsc && ncc build dist/index.js -o dist",
88
"watch": "tsc -w & nodemon --no-deprecation dist/index.js",
99
"test": "jest --watchAll --verbose --silent",
10+
"test-loud": "jest --watchAll --verbose",
1011
"start": "node --no-deprecation dist/index.js"
1112
},
1213
"keywords": [],

src/index.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import dotenv from 'dotenv';
22
import * as core from '@actions/core';
33
import { getSinglePR, getAssertion, getDiff, getChangedFiles, getFileContent } from './github';
4-
import { generatePrompt, testAssertion, writeAnalysis } from './open_ai';
4+
import { generatePrompt, testAssertion, writeAnalysis, openAiFeedback } from './open_ai';
55

66
dotenv.config();
77

@@ -26,7 +26,7 @@ async function main() {
2626
const file: any = await getFileContent(changedFiles, org, repoName);
2727
const prompt: string = generatePrompt(diff, assertion, file);
2828
const rawAnalysis = await testAssertion(prompt);
29-
const analysis = writeAnalysis(rawAnalysis?.toString() ?? '');
29+
const analysis = writeAnalysis(rawAnalysis);
3030
console.log(analysis);
3131
core.setOutput('analysis', analysis);
3232
return analysis;

src/open_ai/index.ts

+65-17
Original file line numberDiff line numberDiff line change
@@ -10,9 +10,55 @@ const openai = new openAi({
1010
apiKey: api_key,
1111
});
1212

13+
/**
14+
* We're using the JSON export for OpenAI, so we're using this type to dictate how we can
15+
* access properties and iterate over them in the output.
16+
*/
17+
export type openAiFeedback = {
18+
feedback: [
19+
{ satisfied: string; scope: string; feedback: string },
20+
{ satisfied: string; scope: string; feedback: string }
21+
];
22+
};
23+
1324
// This wil generate our prompt using the diff, assertion, and whole file
1425
export const generatePrompt = (diff: string, assertion: string | null, file: string): string => {
15-
const comboPrompt = `As a senior engineer, you're tasked with reviewing a documentation PR. Your review will be conducted through two distinct lenses, both centered around an assertion related to usability. The first lens will focus on examining the diff itself — providing targeted feedback on what the PR author actually contributed. The second lens will compare the diff to the entire set of changed files, assessing how the contribution fits within the larger context in relation to the usability assertion. For each lens, provide feedback and determine if the usability assertion is satisfied. You should speak directly to the author and refer to them in second person. Your output should be a JSON-formatted array with two objects. Each object should contain the following properties: 'satisfied' (either a ✅ or ❌ to indicate if the assertion is met), 'scope' (either 'Diff' or 'Integrated'), and 'feedback' (a string providing your targeted feedback for that lens). Here's the assertion: ${assertion}\n\nHere's the diff:\n\n${diff}\n\nHere's the original files:\n\n${file}\n\nBear in mind that some of the files may have been renamed. Remember, do not wrap the JSON in a code block.`;
26+
const comboPrompt = `As a senior engineer, you're tasked with reviewing a documentation PR. Your review comprises two distinct perspectives, each focused on a specific aspect of usability.
27+
28+
- **First Perspective**: Examine the PR's diff. Provide targeted feedback on the author's contribution.
29+
- **Second Perspective**: Assess how the diff integrates with the entire set of changed files, evaluating its contribution to the overall usability.
30+
31+
**Usability Assertion**: ${assertion}
32+
33+
**PR Diff**: ${diff}
34+
35+
**Original Files**: ${file}
36+
37+
(Note: Some files may have been renamed.)
38+
39+
**Your Task**: Provide feedback for each perspective. Determine if the usability assertion is met in each context.
40+
41+
**Output Format**: Your response should be a JSON-formatted array containing exactly two objects. Each object must have the following properties:
42+
- 'satisfied': Indicate if the assertion is met (✅ for yes, ❌ for no).
43+
- 'scope': 'Diff' for the first perspective, 'Integrated' for the second.
44+
- 'feedback': A string providing your targeted feedback.
45+
46+
Example Output:
47+
{
48+
"feedback": [
49+
{
50+
"satisfied": "✅",
51+
"scope": "Diff",
52+
"feedback": "Your changes in the PR are clear and enhance the readability of the documentation."
53+
},
54+
{
55+
"satisfied": "❌",
56+
"scope": "Integrated",
57+
"feedback": "The changes do not align well with the overall structure and flow of the existing documentation."
58+
}
59+
]
60+
}
61+
`;
1662
return comboPrompt;
1763
};
1864

@@ -29,7 +75,7 @@ export const testConnection = async (): Promise<boolean> => {
2975

3076
// Then, we'll create a function that takes in the diff, the author's assertion(s), and the prompt,
3177
// and returns the analysis from OpenAI
32-
export const testAssertion = async (prompt: string): Promise<string | null> => {
78+
export const testAssertion = async (prompt: string): Promise<openAiFeedback | null> => {
3379
let conversation = [
3480
{
3581
role: 'system',
@@ -41,29 +87,31 @@ export const testAssertion = async (prompt: string): Promise<string | null> => {
4187
const chatCompletion = await openai.chat.completions.create({
4288
model: 'gpt-4-1106-preview',
4389
messages: conversation,
90+
response_format: { type: 'json_object' },
4491
});
4592

46-
const analysis: any = chatCompletion.choices[0].message.content;
93+
const analysis = chatCompletion.choices[0].message.content;
4794
console.log(`✅ Got analysis from OpenAI`);
48-
return analysis;
95+
const parsedAnalysis: openAiFeedback = JSON.parse(analysis);
96+
return parsedAnalysis;
4997
} catch (error) {
5098
console.error(error);
5199
return null;
52100
}
53101
};
54102

55103
// We decided to send things back as JSON so we can manipulate the data in the response we'll be sending back to GitHub
56-
export const writeAnalysis = (analysis: string): string => {
57-
// We've still got to double-check because ChatGPT will sometimes return a string that's not valid JSON by wrapping it in code blocks
58-
const regex = /^```(json)?/gm;
59-
analysis = analysis.replace(regex, '');
60-
const analysisJSON = JSON.parse(analysis);
61-
let message = `## DX: Assertion Testing\n\n`;
62-
const feedback = analysisJSON.map((item: any) => {
63-
// we'll create some markdown to make the feedback look nice
64-
return `### ${item.satisfied} ${item.scope}\n\n${item.feedback}\n\n`;
65-
});
66-
feedback.unshift(message);
67-
const feedbackString = feedback.join('');
68-
return feedbackString;
104+
export const writeAnalysis = (analysis: openAiFeedback | null): string => {
105+
if (analysis === null) {
106+
return `Error testing the assertions. Check the logs.`;
107+
} else {
108+
let message = `## DX: Assertion Testing\n\n`;
109+
const feedback = analysis.feedback.map((item: any) => {
110+
// we'll create some markdown to make the feedback look nice
111+
return `### ${item.satisfied} ${item.scope}\n\n${item.feedback}\n\n`;
112+
});
113+
feedback.unshift(message);
114+
const feedbackString = feedback.join('');
115+
return feedbackString;
116+
}
69117
};

0 commit comments

Comments
 (0)