mirror of
https://github.com/AutoMaker-Org/automaker.git
synced 2026-02-03 08:53:36 +00:00
Resolves merge conflicts: - apps/server/src/routes/terminal/common.ts: Keep randomBytes import, use @automaker/utils for createLogger - apps/ui/eslint.config.mjs: Use main's explicit globals list with XMLHttpRequest and MediaQueryListEvent additions - apps/ui/src/components/views/terminal-view.tsx: Keep our terminal improvements (killAllSessions, beforeunload, better error handling) - apps/ui/src/config/terminal-themes.ts: Keep our search highlight colors for all themes - apps/ui/src/store/app-store.ts: Keep our terminal settings persistence improvements (merge function) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
93 lines
3.0 KiB
TypeScript
93 lines
3.0 KiB
TypeScript
/**
|
|
* POST /generate endpoint - Generate spec from project definition
|
|
*/
|
|
|
|
import type { Request, Response } from 'express';
|
|
import type { EventEmitter } from '../../../lib/events.js';
|
|
import { createLogger } from '@automaker/utils';
|
|
import {
|
|
getSpecRegenerationStatus,
|
|
setRunningState,
|
|
logAuthStatus,
|
|
logError,
|
|
getErrorMessage,
|
|
} from '../common.js';
|
|
import { generateSpec } from '../generate-spec.js';
|
|
|
|
const logger = createLogger('SpecRegeneration');
|
|
|
|
export function createGenerateHandler(events: EventEmitter) {
|
|
return async (req: Request, res: Response): Promise<void> => {
|
|
logger.info('========== /generate endpoint called ==========');
|
|
logger.debug('Request body:', JSON.stringify(req.body, null, 2));
|
|
|
|
try {
|
|
const { projectPath, projectDefinition, generateFeatures, analyzeProject, maxFeatures } =
|
|
req.body as {
|
|
projectPath: string;
|
|
projectDefinition: string;
|
|
generateFeatures?: boolean;
|
|
analyzeProject?: boolean;
|
|
maxFeatures?: number;
|
|
};
|
|
|
|
logger.debug('Parsed params:');
|
|
logger.debug(' projectPath:', projectPath);
|
|
logger.debug(' projectDefinition length:', `${projectDefinition?.length || 0} chars`);
|
|
logger.debug(' generateFeatures:', generateFeatures);
|
|
logger.debug(' analyzeProject:', analyzeProject);
|
|
logger.debug(' maxFeatures:', maxFeatures);
|
|
|
|
if (!projectPath || !projectDefinition) {
|
|
logger.error('Missing required parameters');
|
|
res.status(400).json({
|
|
success: false,
|
|
error: 'projectPath and projectDefinition required',
|
|
});
|
|
return;
|
|
}
|
|
|
|
const { isRunning } = getSpecRegenerationStatus();
|
|
if (isRunning) {
|
|
logger.warn('Generation already running, rejecting request');
|
|
res.json({ success: false, error: 'Spec generation already running' });
|
|
return;
|
|
}
|
|
|
|
logAuthStatus('Before starting generation');
|
|
|
|
const abortController = new AbortController();
|
|
setRunningState(true, abortController);
|
|
logger.info('Starting background generation task...');
|
|
|
|
generateSpec(
|
|
projectPath,
|
|
projectDefinition,
|
|
events,
|
|
abortController,
|
|
generateFeatures,
|
|
analyzeProject,
|
|
maxFeatures
|
|
)
|
|
.catch((error) => {
|
|
logError(error, 'Generation failed with error');
|
|
events.emit('spec-regeneration:event', {
|
|
type: 'spec_regeneration_error',
|
|
error: getErrorMessage(error),
|
|
projectPath: projectPath,
|
|
});
|
|
})
|
|
.finally(() => {
|
|
logger.info('Generation task finished (success or error)');
|
|
setRunningState(false, null);
|
|
});
|
|
|
|
logger.info('Returning success response (generation running in background)');
|
|
res.json({ success: true });
|
|
} catch (error) {
|
|
logError(error, 'Generate spec route handler failed');
|
|
res.status(500).json({ success: false, error: getErrorMessage(error) });
|
|
}
|
|
};
|
|
}
|