ManimCat / src /services /code-edit.ts
Bin29's picture
Sync from main: b9996d0 feat: refine studio workspace and plot canvas ui
14ea677
import OpenAI from 'openai'
import { createLogger } from '../utils/logger'
import { generateCodeEditPrompt, getRoleSystemPrompt, getSharedModule } from '../prompts'
import type { CustomApiConfig, OutputMode, PromptOverrides } from '../types'
import { createCustomOpenAIClient } from './openai-client-factory'
import { createChatCompletionText } from './openai-stream'
import { buildTokenParams } from '../utils/reasoning-model'
const logger = createLogger('CodeEditService')
const CODER_TEMPERATURE = parseFloat(process.env.AI_TEMPERATURE || '0.7')
const MAX_TOKENS = parseInt(process.env.AI_MAX_TOKENS || '12000', 10)
const THINKING_TOKENS = parseInt(process.env.AI_THINKING_TOKENS || '20000', 10)
function createCustomClient(config: CustomApiConfig): OpenAI {
return createCustomOpenAIClient(config)
}
function applyPromptTemplate(
template: string,
values: Record<string, string>,
promptOverrides?: PromptOverrides
): string {
let output = template
// 替换共享模块占位符
output = output.replace(/\{\{apiIndexModule\}\}/g, getSharedModule('apiIndex', promptOverrides))
output = output.replace(/\{\{sharedSpecification\}\}/g, getSharedModule('specification', promptOverrides))
// 替换变量占位符
for (const [key, value] of Object.entries(values)) {
output = output.replace(new RegExp(`\\{\\{\\s*${key}\\s*\\}\\}`, 'g'), value || '')
}
return output
}
function extractCodeFromResponse(text: string, outputMode: OutputMode): string {
if (!text) return ''
const sanitized = text.replace(/<think>[\s\S]*?<\/think>/gi, '')
if (outputMode === 'image') {
return sanitized.trim()
}
const anchorMatch = sanitized.match(/### START ###([\s\S]*?)### END ###/)
if (anchorMatch) {
return anchorMatch[1].trim()
}
const codeMatch = sanitized.match(/```(?:python)?\n([\s\S]*?)```/i)
if (codeMatch) {
return codeMatch[1].trim()
}
return sanitized.trim()
}
export async function generateEditedManimCode(
concept: string,
instructions: string,
code: string,
outputMode: OutputMode,
customApiConfig?: CustomApiConfig,
promptOverrides?: PromptOverrides
): Promise<string> {
if (!customApiConfig) {
throw new Error('No upstream AI is configured for this request')
}
const model = customApiConfig.model?.trim() || ''
if (!model) {
throw new Error('No model available')
}
const client = createCustomClient(customApiConfig)
try {
const baseSystemPrompt = getRoleSystemPrompt('codeEdit', promptOverrides)
const userPromptOverride = promptOverrides?.roles?.codeEdit?.user
const baseUserPrompt = userPromptOverride
? applyPromptTemplate(userPromptOverride, { concept, instructions, code, outputMode }, promptOverrides)
: generateCodeEditPrompt(concept, instructions, code, outputMode)
const systemPrompt = baseSystemPrompt
const userPrompt = baseUserPrompt
logger.info('开始 AI 修改代码', { concept, outputMode })
const { content, mode } = await createChatCompletionText(
client,
{
model,
messages: [
{ role: 'system', content: systemPrompt },
{ role: 'user', content: userPrompt }
],
temperature: CODER_TEMPERATURE,
...buildTokenParams(THINKING_TOKENS, MAX_TOKENS)
},
{ fallbackToNonStream: true, usageLabel: 'code-edit' }
)
if (!content) {
logger.warn('AI 修改返回空内容')
return ''
}
const extracted = extractCodeFromResponse(content, outputMode)
logger.info('AI 修改完成', {
concept,
outputMode,
mode,
length: extracted.length,
codePreview: extracted.slice(0, 500)
})
return extracted
} catch (error) {
if (error instanceof OpenAI.APIError) {
logger.error('AI 修改 API 错误', {
concept,
status: error.status,
code: error.code,
type: error.type,
message: error.message
})
} else if (error instanceof Error) {
logger.error('AI 修改失败', { concept, errorName: error.name, errorMessage: error.message })
} else {
logger.error('AI 修改失败,未知错误', { concept, error: String(error) })
}
return ''
}
}
export function isCodeEditAvailable(): boolean {
return true
}