-
Notifications
You must be signed in to change notification settings - Fork 1
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat(backend): adding CodeValidation and TestCLI to help BuildingSystem to generate correct version code #117
Merged
Merged
Changes from all commits
Commits
Show all changes
18 commits
Select commit
Hold shift + click to select a range
ce65c55
init auto fix
ZHallen122 61c103c
delete unused function
ZHallen122 183d960
init llm operation
ZHallen122 36f2f2e
update to use json
ZHallen122 b1d2bbf
update template
ZHallen122 59af234
Fix bug
ZHallen122 8f7db08
[autofix.ci] apply automated fixes
autofix-ci[bot] 0bfcf37
refector to make code easier to read
ZHallen122 0be6af5
add read operation
ZHallen122 af42fff
add rename map to help match dependencies file name
ZHallen122 7266731
add rename map path
ZHallen122 998c609
udpate template
ZHallen122 cf7e3c1
reduce folder scope
ZHallen122 c88df24
fix error
ZHallen122 64782b4
update file fix and frontend generate
ZHallen122 373f11c
update test case
ZHallen122 62015a5
refect code structure and add comment
ZHallen122 d3cbd03
update file name
ZHallen122 File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
355 changes: 355 additions & 0 deletions
355
backend/src/build-system/handlers/frontend-code-generate/CodeReview.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,355 @@ | ||
/** | ||
* FrontendQueueProcessor.ts | ||
* | ||
* This file defines: | ||
* 1) The `FileTask` interface, which describes the basic structure | ||
* of tasks that the queue will handle. | ||
* 2) The `CodeTaskQueue` class, a simple FIFO queue for tasks. | ||
* 3) The `FrontendQueueProcessor` class, which orchestrates: | ||
* - Writing files | ||
* - Validating builds via `FrontendCodeValidator` | ||
* - Attempting fixes (via LLM) if builds fail | ||
*/ | ||
|
||
import { Logger } from '@nestjs/common'; | ||
import { FrontendCodeValidator } from './CodeValidator'; | ||
import { readFileSync } from 'fs'; | ||
import { chatSyncWithClocker } from 'src/build-system/utils/handler-helper'; | ||
import { createFileWithRetries } from 'src/build-system/utils/files'; | ||
import { BuilderContext } from 'src/build-system/context'; | ||
import { removeCodeBlockFences } from 'src/build-system/utils/strings'; | ||
import { | ||
generateCommonErrorPrompt, | ||
generateFileOperationPrompt, | ||
} from './prompt'; | ||
import { FileOperationManager } from './FileOperationManager'; | ||
|
||
import normalizePath from 'normalize-path'; | ||
import path from 'path'; | ||
|
||
/** | ||
* Describes a single file task that the queue will process. | ||
* | ||
* @property filePath - The relative path to the file (e.g. "src/components/MyComp.ts"). | ||
* @property fileContents - The code content for that file (usually generated by an LLM). | ||
* @property dependenciesPath? - Optional: references to any internal dependency file(s). | ||
*/ | ||
export interface FileTask { | ||
filePath: string; // e.g. "src/components/MyComponent.ts" | ||
fileContents: string; // the code you got from the LLM | ||
dependenciesPath?: string; | ||
} | ||
|
||
export class CodeTaskQueue { | ||
private tasks: FileTask[] = []; | ||
|
||
enqueue(task: FileTask) { | ||
this.tasks.push(task); | ||
} | ||
|
||
dequeue(): FileTask | undefined { | ||
return this.tasks.shift(); | ||
} | ||
|
||
get size(): number { | ||
return this.tasks.length; | ||
} | ||
} | ||
|
||
/** | ||
* FrontendQueueProcessor | ||
* | ||
* This class orchestrates how each `FileTask` is handled: | ||
* 1. Writes the file content to disk. | ||
* 2. Runs a build/validation step (via `FrontendCodeValidator`). | ||
* 3. If there is a build error, attempts to fix the file by | ||
* invoking an LLM and performing the suggested operations. | ||
* 4. Repeats until build succeeds or max attempts are reached. | ||
*/ | ||
export class FrontendQueueProcessor { | ||
private logger = new Logger('FrontendQueueProcessor'); | ||
|
||
constructor( | ||
private validator: FrontendCodeValidator, // Path to your frontend project | ||
private queue: CodeTaskQueue, // The queue of files to process | ||
private context: BuilderContext, | ||
private frontendPath: string, | ||
private renameMap: Map<string, string>, | ||
) {} | ||
|
||
/** | ||
* Process the entire queue, one file at a time. | ||
*/ | ||
public async processAllTasks(): Promise<void> { | ||
while (this.queue.size > 0) { | ||
const task = this.queue.dequeue(); | ||
if (!task) break; | ||
|
||
await this.processSingleTask(task); | ||
|
||
this.logger.log(`Remaining tasks in queue: ${this.queue.size}`); | ||
} | ||
|
||
// maybe need to requeue | ||
|
||
this.logger.log('All tasks processed successfully!'); | ||
} | ||
|
||
/** | ||
* For a single file: | ||
* 1. Write it to disk | ||
* 2. Run "npm run build" (through the validator) | ||
* 3. If error -> try to fix -> repeat until success or max attempts. | ||
*/ | ||
private async processSingleTask(task: FileTask): Promise<void> { | ||
this.logger.log(`Processing file task: ${task.filePath}`); | ||
|
||
let currentFullFilePath = normalizePath( | ||
path.resolve(this.frontendPath, task.filePath), | ||
); | ||
|
||
// 1. Write the file to disk | ||
createFileWithRetries(currentFullFilePath, task.fileContents); | ||
|
||
const maxFixAttempts = 2; | ||
|
||
for (let attempt = 1; attempt <= maxFixAttempts; attempt++) { | ||
const validationResult = await this.validator.validate(); | ||
|
||
if (validationResult.success) { | ||
this.logger.log( | ||
`File ${task.filePath} build succeeded on attempt #${attempt}.`, | ||
); | ||
return; // done, move on | ||
} | ||
|
||
// Build failed. We'll feed the entire `validationResult.error` back to GPT | ||
// this.logger.warn( | ||
// `Build failed on attempt #${attempt} for file ${task.filePath}. Error:\n${validationResult.error}`, | ||
// ); | ||
|
||
this.logger.warn( | ||
`Build failed on attempt #${attempt} for file ${task.filePath}.`, | ||
); | ||
|
||
// 3. Fix the file | ||
try { | ||
const newFilePath = await this.fixFileGeneric( | ||
currentFullFilePath, | ||
task, | ||
validationResult.error ?? '', | ||
); | ||
|
||
if (newFilePath !== null) { | ||
this.logger.log( | ||
`File was renamed: ${task.filePath} → ${newFilePath}`, | ||
); | ||
task.filePath = newFilePath; | ||
currentFullFilePath = normalizePath( | ||
path.resolve(this.frontendPath, newFilePath), | ||
); | ||
this.logger.log( | ||
`Updated currentFullFilePath: ${currentFullFilePath}`, | ||
); | ||
} | ||
} catch (error) { | ||
this.logger.error( | ||
'Fix File Generic failed, get error: ' + error.messages, | ||
); | ||
} | ||
} | ||
|
||
// If we reached here, we failed all attempts | ||
// if we want to end all generate | ||
// throw new Error( | ||
// `Failed to fix build for file ${task.filePath} after ${maxFixAttempts} attempts.`, | ||
// ); | ||
|
||
// if we dont want to end the process | ||
this.logger.error( | ||
`Failed to fix build for file ${task.filePath} after ${maxFixAttempts} attempts.`, | ||
); | ||
} | ||
|
||
/** | ||
* Fallback fix method if no structured error details are available. | ||
* This method interacts with an LLM to propose fixes and then | ||
* executes those fixes via FileOperationManager. | ||
* | ||
* @param currentFullFilePath - The absolute path to the file being fixed. | ||
* @param task - The original FileTask with code and path info. | ||
* @param rawErrorText - The error text from the validator, fed to the LLM. | ||
* | ||
* @returns The new file path if the file was renamed, or null otherwise. | ||
*/ | ||
private async fixFileGeneric( | ||
currentFullFilePath: string, | ||
task: FileTask, | ||
rawErrorText: string, | ||
): Promise<string | null> { | ||
try { | ||
this.logger.log(`Generic fix attempt for file: ${currentFullFilePath}`); | ||
const originalContent = readFileSync(currentFullFilePath, 'utf-8'); | ||
|
||
this.logger.debug('raw error: ' + rawErrorText); | ||
|
||
const fixPrompt = generateFileOperationPrompt(task.filePath); | ||
const commonIssuePrompt = generateCommonErrorPrompt(); | ||
|
||
const fileOperationManager = new FileOperationManager( | ||
this.frontendPath, | ||
this.renameMap, | ||
); | ||
|
||
//this.logger.log(fixPrompt); | ||
|
||
// Use model for a fix | ||
let fixResponse = await chatSyncWithClocker( | ||
this.context, | ||
{ | ||
model: 'gpt-4o', | ||
messages: [ | ||
{ role: 'system', content: fixPrompt }, | ||
{ | ||
role: 'user', | ||
content: ` ## Current file: \n ${task.filePath} ## Current Code \n ${originalContent}\n `, | ||
}, | ||
{ | ||
role: 'user', | ||
content: ` ## Error Messages: \n ${rawErrorText}`, | ||
}, | ||
{ | ||
role: 'assistant', | ||
content: | ||
"Good, now provider your dependencies, it's okay dependencies are empty, which means you don't have any dependencies", | ||
}, | ||
{ | ||
role: 'user', | ||
content: `## Overview of The Internal Dependencies file you may need: \n ${task.dependenciesPath}`, | ||
}, | ||
{ | ||
role: 'assistant', | ||
content: `Let me analysis the current file. Why error message occour? | ||
Let me check some common issue to make sure my thinking is correct ${commonIssuePrompt}. | ||
I must follow the output format`, | ||
}, | ||
{ | ||
role: 'user', | ||
content: `Now you should start fix the current code error.`, | ||
}, | ||
{ | ||
role: 'assistant', | ||
content: `Let me check my result and I must follow the output format.`, | ||
}, | ||
], | ||
}, | ||
'fix code (generic)', | ||
'FrontendQueueProcessor', | ||
); | ||
|
||
this.logger.debug('Fix Response: ' + fixResponse); | ||
this.logger.debug('dependency file Paths ' + task.dependenciesPath); | ||
const parsed_fixResponse = removeCodeBlockFences(fixResponse); | ||
|
||
let operations = fileOperationManager.parse( | ||
parsed_fixResponse, | ||
task.filePath, | ||
); | ||
|
||
// **If LLM requested additional files, read them** | ||
if (operations.some((op) => op.action === 'read')) { | ||
this.logger.log( | ||
`LLM requested additional context. Reading dependencies...`, | ||
); | ||
|
||
for (const op of operations) { | ||
if (op.action === 'read' && op.originalPath) { | ||
try { | ||
op.code = readFileSync( | ||
path.resolve(this.frontendPath, op.originalPath), | ||
'utf-8', | ||
); | ||
this.logger.log(`Read file: ${op.originalPath}`); | ||
} catch (error) { | ||
this.logger.warn( | ||
`Failed to read file: ${op.originalPath}. Error: ${error.message}`, | ||
); | ||
} | ||
} | ||
} | ||
|
||
// **Second Attempt: Retry fix with additional file content** | ||
fixResponse = await chatSyncWithClocker( | ||
this.context, | ||
{ | ||
model: 'gpt-4o', | ||
messages: [ | ||
{ role: 'system', content: fixPrompt }, | ||
{ | ||
role: 'user', | ||
content: `## Current Code \n ${originalContent}\n `, | ||
}, | ||
{ | ||
role: 'user', | ||
content: `## Error messages: \n ${rawErrorText}`, | ||
}, | ||
{ | ||
role: 'assistant', | ||
content: | ||
"Good, now provider your Internal dependencies, it's okay dependencies are empty, which means you don't have any dependencies", | ||
}, | ||
{ | ||
role: 'user', | ||
content: `## Overview of Internal Dependencies files: \n ${task.dependenciesPath}\n | ||
## Internal Dependencies files content:\n ${operations | ||
.filter((op) => op.action === 'read' && op.code) | ||
.map((op) => `File: ${op.originalPath}\nContent:\n${op.code}`) | ||
.join('\n\n')}`, | ||
}, | ||
{ | ||
role: 'assistant', | ||
content: `Let me analysis the current file. Why error message occour | ||
This time I shouldn't use the read tool because previous context already use it. | ||
Let me check some common issue to make sure my thinking is correct ${commonIssuePrompt}. | ||
I must follow the output format`, | ||
}, | ||
{ | ||
role: 'user', | ||
content: `Now you should start fix the current code error.`, | ||
}, | ||
{ | ||
role: 'assistant', | ||
content: `Let me check my result and I must follow the output format`, | ||
}, | ||
], | ||
}, | ||
'fix code (generic)', | ||
'FrontendQueueProcessor', | ||
); | ||
this.logger.debug( | ||
'Updated Fix Response with extra context: ' + fixResponse, | ||
); | ||
const updated_fixResponse = removeCodeBlockFences(fixResponse); | ||
operations = await fileOperationManager.parse( | ||
updated_fixResponse, | ||
task.filePath, | ||
); | ||
} | ||
|
||
const newFilePath = | ||
await fileOperationManager.executeOperations(operations); | ||
|
||
this.logger.log(`Generic fix applied to file: ${task.filePath}`); | ||
|
||
if (newFilePath) { | ||
this.logger.log(`File was renamed: ${task.filePath} → ${newFilePath}`); | ||
return newFilePath; | ||
} | ||
|
||
return null; | ||
} catch (error) { | ||
this.logger.error('Generic Fix file: ' + error.message); | ||
return null; | ||
} | ||
} | ||
} |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Ensure consistent error handling
The code only logs a final error when all fix attempts fail. Decide whether the process should halt or continue for subsequent files to prevent partial corruption.