api.php 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096
  1. <?php
  2. use App\Http\Controllers\Api\IntelligentExamController;
  3. use App\Http\Controllers\Api\PreQuestionApiController;
  4. use App\Http\Controllers\Api\TextbookApiController;
  5. use App\Http\Controllers\Api\MistakeBookController;
  6. use App\Http\Controllers\Api\QuestionSearchController;
  7. use App\Http\Controllers\Api\QuestionRandomController;
  8. use App\Http\Controllers\Api\PaperAssembleController;
  9. use App\Http\Controllers\Api\PaperJsonController;
  10. use App\Http\Controllers\Api\QuestionSolutionController;
  11. use App\Http\Controllers\Api\KnowledgeRecommendController;
  12. use App\Http\Controllers\Api\AbilityEvaluateController;
  13. use App\Services\QuestionServiceApi;
  14. use Illuminate\Support\Facades\Log;
  15. use Illuminate\Support\Facades\Route;
  16. use App\Events\QuestionGenerationCompleted;
  17. use App\Events\QuestionGenerationFailed;
  18. use Illuminate\Auth\Middleware\Authenticate;
  19. use App\Http\Controllers\Api\ExamAnalysisApiController;
  20. use App\Http\Controllers\Api\StudentAnswerAnalysisController;
  21. /*
  22. |--------------------------------------------------------------------------
  23. | 题库管理 API 路由
  24. |--------------------------------------------------------------------------
  25. */
  26. // 给 Python/内部服务消费的“筛选题库”API(需要 X-Internal-Token)
  27. Route::middleware('internal.token')->get('/pre-questions', [PreQuestionApiController::class, 'index'])
  28. ->name('api.pre-questions.index');
  29. Route::get('/questions/search', QuestionSearchController::class)->name('api.questions.search');
  30. Route::get('/questions/random', QuestionRandomController::class)->name('api.questions.random');
  31. Route::post('/papers/assemble', PaperAssembleController::class)->name('api.papers.assemble');
  32. Route::get('/papers/{paperId}/json', [PaperJsonController::class, 'show'])->name('api.papers.json');
  33. Route::get('/questions/{id}/solution', QuestionSolutionController::class)->name('api.questions.solution');
  34. Route::get('/knowledge/recommend', KnowledgeRecommendController::class)->name('api.knowledge.recommend');
  35. Route::post('/abilities/evaluate', AbilityEvaluateController::class)->name('api.abilities.evaluate');
  36. // 接收题目生成回调
  37. Route::post('/questions/callback', function () {
  38. try {
  39. $data = request()->all();
  40. Log::info('Received question generation callback', $data);
  41. // 验证回调数据
  42. if (!isset($data['task_id']) || !isset($data['status'])) {
  43. return response()->json(['error' => 'Invalid callback data'], 400);
  44. }
  45. // 处理回调数据并存储通知到session
  46. if ($data['status'] === 'completed') {
  47. $result = $data['result'] ?? [];
  48. $total = $result['total'] ?? $data['total'] ?? ($result['saved'] ?? 0);
  49. $kpCode = $result['kp_code'] ?? $data['kp_code'] ?? '';
  50. // 将成功通知存储到session,供下次页面刷新时显示
  51. session()->flash('notification', [
  52. 'type' => 'success',
  53. 'title' => '✅ 题目生成完成',
  54. 'body' => "任务 ID: {$data['task_id']}\n生成题目: {$total} 道" . ($kpCode ? "\n知识点: {$kpCode}" : ''),
  55. 'color' => 'success'
  56. ]);
  57. Log::info("题目生成成功通知已存储", [
  58. 'task_id' => $data['task_id'],
  59. 'total' => $total,
  60. 'kp_code' => $kpCode
  61. ]);
  62. } elseif ($data['status'] === 'failed') {
  63. $error = $data['error'] ?? '未知错误';
  64. // 将失败通知存储到session
  65. session()->flash('notification', [
  66. 'type' => 'error',
  67. 'title' => '❌ 题目生成失败',
  68. 'body' => "任务 ID: {$data['task_id']}\n错误: {$error}",
  69. 'color' => 'danger'
  70. ]);
  71. Log::error("题目生成失败通知已存储", [
  72. 'task_id' => $data['task_id'],
  73. 'error' => $error
  74. ]);
  75. }
  76. return response()->json([
  77. 'success' => true,
  78. 'message' => 'Callback received and notification stored',
  79. 'status' => $data['status']
  80. ]);
  81. } catch (\Exception $e) {
  82. Log::error('Callback processing failed: ' . $e->getMessage());
  83. return response()->json(['error' => $e->getMessage()], 500);
  84. }
  85. })->name('api.questions.callback');
  86. // 接收OCR题目生成回调
  87. Route::post('/ocr-question-callback', function () {
  88. try {
  89. $data = request()->all();
  90. Log::info('Received OCR question generation callback', $data);
  91. // 验证必要的回调数据
  92. if (!isset($data['task_id']) || !isset($data['status']) || !isset($data['ocr_record_id'])) {
  93. Log::error('OCR callback missing required fields', $data);
  94. return response()->json([
  95. 'success' => false,
  96. 'error' => 'Missing required fields: task_id, status, ocr_record_id'
  97. ], 400);
  98. }
  99. $taskId = $data['task_id'];
  100. $ocrRecordId = $data['ocr_record_id'];
  101. $status = $data['status'];
  102. // 将回调结果存储到缓存中,供前端查询(保留30秒)
  103. $cacheKey = "ocr_callback_{$ocrRecordId}_{$taskId}";
  104. cache([$cacheKey => $data], now()->addSeconds(30));
  105. Log::info("OCR callback cached with key: {$cacheKey}", [
  106. 'ocr_record_id' => $ocrRecordId,
  107. 'task_id' => $taskId,
  108. 'status' => $status,
  109. 'total_generated' => $data['result']['total_generated'] ?? 0,
  110. 'total_saved' => $data['result']['total_saved'] ?? 0
  111. ]);
  112. // 处理题目关联逻辑
  113. if ($status === 'completed') {
  114. $updatedCount = 0;
  115. // 从result中提取question_mappings(QuestionBank API将它放在result字段中)
  116. $mappings = $data['result']['question_mappings'] ?? $data['question_mappings'] ?? [];
  117. Log::info("Processing OCR question associations", [
  118. 'ocr_record_id' => $ocrRecordId,
  119. 'task_id' => $taskId,
  120. 'mappings_count' => count($mappings)
  121. ]);
  122. // 更新ocr_question_results表中的关联关系
  123. foreach ($mappings as $mapping) {
  124. try {
  125. $ocrQuestionNumber = $mapping['ocr_question_number'] ?? null;
  126. $questionBankId = $mapping['question_bank_id'] ?? null;
  127. $questionCode = $mapping['question_code'] ?? null;
  128. if ($ocrQuestionNumber && $questionBankId) {
  129. // 查找对应的OCR题目结果并更新
  130. $updated = DB::table('ocr_question_results')
  131. ->where('ocr_record_id', $ocrRecordId)
  132. ->where('question_number', $ocrQuestionNumber)
  133. ->update([
  134. 'question_bank_id' => $questionBankId,
  135. 'generation_status' => 'completed',
  136. 'generation_task_id' => $taskId,
  137. 'generation_error' => null,
  138. ]);
  139. if ($updated) {
  140. $updatedCount++;
  141. Log::info("Updated OCR question association", [
  142. 'ocr_record_id' => $ocrRecordId,
  143. 'question_number' => $ocrQuestionNumber,
  144. 'question_bank_id' => $questionBankId,
  145. 'question_code' => $questionCode
  146. ]);
  147. } else {
  148. Log::warning("No OCR question result found for association", [
  149. 'ocr_record_id' => $ocrRecordId,
  150. 'question_number' => $ocrQuestionNumber
  151. ]);
  152. }
  153. }
  154. } catch (\Exception $e) {
  155. Log::error("Failed to update OCR question association", [
  156. 'mapping' => $mapping,
  157. 'error' => $e->getMessage()
  158. ]);
  159. }
  160. }
  161. Log::info("OCR question association completed", [
  162. 'ocr_record_id' => $ocrRecordId,
  163. 'task_id' => $taskId,
  164. 'total_mappings' => count($mappings),
  165. 'updated_count' => $updatedCount
  166. ]);
  167. // 更新OCR记录的整体状态为已完成
  168. try {
  169. DB::table('ocr_records')
  170. ->where('id', $ocrRecordId)
  171. ->update([
  172. 'status' => 'completed',
  173. 'processed_at' => now(),
  174. 'updated_at' => now()
  175. ]);
  176. Log::info("Updated OCR record status to completed", [
  177. 'ocr_record_id' => $ocrRecordId,
  178. 'task_id' => $taskId
  179. ]);
  180. } catch (\Exception $e) {
  181. Log::error("Failed to update OCR record status", [
  182. 'ocr_record_id' => $ocrRecordId,
  183. 'error' => $e->getMessage()
  184. ]);
  185. }
  186. } elseif ($status === 'failed') {
  187. // 更新所有相关的OCR题目结果为失败状态
  188. try {
  189. $updated = DB::table('ocr_question_results')
  190. ->where('ocr_record_id', $ocrRecordId)
  191. ->where('generation_status', 'pending') // 只更新待处理的
  192. ->update([
  193. 'generation_status' => 'failed',
  194. 'generation_task_id' => $taskId,
  195. 'generation_error' => $data['error'] ?? 'Unknown error',
  196. ]);
  197. Log::info("Updated OCR questions to failed status", [
  198. 'ocr_record_id' => $ocrRecordId,
  199. 'task_id' => $taskId,
  200. 'updated_count' => $updated,
  201. 'error' => $data['error'] ?? 'Unknown error'
  202. ]);
  203. // 更新OCR记录的状态为失败
  204. DB::table('ocr_records')
  205. ->where('id', $ocrRecordId)
  206. ->update([
  207. 'status' => 'failed',
  208. 'error_message' => $data['error'] ?? 'Question generation failed',
  209. 'updated_at' => now()
  210. ]);
  211. Log::info("Updated OCR record status to failed", [
  212. 'ocr_record_id' => $ocrRecordId,
  213. 'task_id' => $taskId,
  214. 'error' => $data['error'] ?? 'Unknown error'
  215. ]);
  216. } catch (\Exception $e) {
  217. Log::error("Failed to update OCR questions to failed status", [
  218. 'ocr_record_id' => $ocrRecordId,
  219. 'error' => $e->getMessage()
  220. ]);
  221. }
  222. }
  223. return response()->json([
  224. 'success' => true,
  225. 'message' => 'OCR callback received and processed',
  226. 'data' => [
  227. 'task_id' => $taskId,
  228. 'ocr_record_id' => $ocrRecordId,
  229. 'status' => $status,
  230. 'cache_key' => $cacheKey,
  231. 'associations_processed' => $status === 'completed' ? count($data['question_mappings'] ?? []) : 0
  232. ]
  233. ]);
  234. } catch (\Exception $e) {
  235. Log::error('OCR callback processing failed: ' . $e->getMessage());
  236. Log::error('Exception details: ' . $e->getTraceAsString());
  237. return response()->json([
  238. 'success' => false,
  239. 'error' => 'Callback processing failed: ' . $e->getMessage()
  240. ], 500);
  241. }
  242. })->name('api.ocr.callback');
  243. // 获取题目生成回调结果
  244. Route::get('/questions/callback/{taskId}', function (string $taskId) {
  245. // ✅ 优先从缓存读取(跨域友好)
  246. $callbackData = cache($taskId);
  247. if ($callbackData) {
  248. // 清除已读取的回调数据
  249. cache()->forget($taskId);
  250. session()->forget('question_gen_callback_' . $taskId);
  251. return response()->json($callbackData);
  252. }
  253. // 备选:从session读取
  254. $sessionData = session('question_gen_callback_' . $taskId);
  255. if ($sessionData) {
  256. // 清除已读取的回调数据
  257. session()->forget('question_gen_callback_' . $taskId);
  258. return response()->json($sessionData);
  259. }
  260. // 未收到回调
  261. return response()->json(['status' => 'pending'], 202);
  262. })->name('api.questions.callback.get');
  263. // 获取OCR题目生成回调结果
  264. Route::get('/ocr-question-callback/{ocrRecordId}/{taskId}', function (int $ocrRecordId, string $taskId) {
  265. $cacheKey = "ocr_callback_{$ocrRecordId}_{$taskId}";
  266. $callbackData = cache($cacheKey);
  267. if ($callbackData) {
  268. // 清除已读取的回调数据
  269. cache()->forget($cacheKey);
  270. return response()->json([
  271. 'success' => true,
  272. 'data' => $callbackData
  273. ]);
  274. }
  275. return response()->json([
  276. 'success' => false,
  277. 'status' => 'pending',
  278. 'message' => 'OCR callback not received yet'
  279. ], 202);
  280. })->name('api.ocr.callback.get');
  281. // 题目相关 API
  282. Route::get('/questions', function (QuestionServiceApi $service) {
  283. try {
  284. $page = (int) request()->get('page', 1);
  285. $perPage = (int) request()->get('per_page', 25);
  286. $filters = [
  287. 'kp_code' => request()->get('kp_code'),
  288. 'difficulty' => request()->get('difficulty'),
  289. 'search' => request()->get('search'),
  290. ];
  291. $response = $service->listQuestions($page, $perPage, $filters);
  292. return response()->json($response);
  293. } catch (\Exception $e) {
  294. \Log::error('Failed to fetch questions: ' . $e->getMessage());
  295. return response()->json([
  296. 'data' => [],
  297. 'meta' => [
  298. 'page' => 1,
  299. 'per_page' => 25,
  300. 'total' => 0,
  301. 'total_pages' => 0,
  302. ],
  303. 'error' => $e->getMessage(),
  304. ], 500);
  305. }
  306. });
  307. // 获取题目统计信息
  308. Route::get('/questions/statistics', function (QuestionServiceApi $service) {
  309. try {
  310. $stats = $service->getStatistics();
  311. return response()->json($stats);
  312. } catch (\Exception $e) {
  313. \Log::error('Failed to get question statistics: ' . $e->getMessage());
  314. return response()->json(['error' => $e->getMessage()], 500);
  315. }
  316. });
  317. // 语义搜索题目
  318. Route::post('/questions/search', function (QuestionServiceApi $service) {
  319. try {
  320. $data = request()->only(['query', 'limit']);
  321. $results = $service->searchQuestions($data['query'], $data['limit'] ?? 20);
  322. return response()->json($results);
  323. } catch (\Exception $e) {
  324. \Log::error('Question search failed: ' . $e->getMessage());
  325. return response()->json(['error' => $e->getMessage()], 500);
  326. }
  327. });
  328. // 获取单个题目详情
  329. Route::get('/questions/{id}', function (int $id, QuestionServiceApi $service) {
  330. try {
  331. $question = $service->getQuestionById($id);
  332. if (!$question) {
  333. return response()->json(['error' => 'Question not found'], 404);
  334. }
  335. return response()->json($question);
  336. } catch (\Exception $e) {
  337. \Log::error("Failed to get question {$id}: " . $e->getMessage());
  338. return response()->json(['error' => $e->getMessage()], 500);
  339. }
  340. });
  341. // AI 生成题目
  342. Route::post('/questions/generate', function (QuestionServiceApi $service) {
  343. try {
  344. $data = request()->only(['kp_code', 'keyword', 'count', 'strategy']);
  345. $result = $service->generateQuestions($data);
  346. return response()->json($result);
  347. } catch (\Exception $e) {
  348. \Log::error('Question generation failed: ' . $e->getMessage());
  349. return response()->json([
  350. 'success' => false,
  351. 'message' => $e->getMessage(),
  352. ], 500);
  353. }
  354. });
  355. // 删除题目
  356. Route::delete('/questions/{id}', function (int $id, QuestionServiceApi $service) {
  357. try {
  358. $deleted = $service->deleteQuestion($id);
  359. return response()->json([
  360. 'success' => $deleted,
  361. 'message' => $deleted ? 'Question deleted' : 'Failed to delete',
  362. ]);
  363. } catch (\Exception $e) {
  364. \Log::error("Failed to delete question {$id}: " . $e->getMessage());
  365. return response()->json([
  366. 'success' => false,
  367. 'message' => $e->getMessage(),
  368. ], 500);
  369. }
  370. });
  371. // 获取知识点选项
  372. Route::get('/knowledge-points', function (QuestionServiceApi $service) {
  373. try {
  374. $points = $service->getKnowledgePointOptions();
  375. return response()->json($points);
  376. } catch (\Exception $e) {
  377. \Log::error('Failed to get knowledge points: ' . $e->getMessage());
  378. return response()->json([], 500);
  379. }
  380. });
  381. // 智能出卷对外接口:生成试卷并返回PDF/判卷地址
  382. Route::post('/intelligent-exams', [IntelligentExamController::class, 'store'])
  383. ->withoutMiddleware([
  384. Authenticate::class,
  385. 'auth',
  386. 'auth:sanctum',
  387. 'auth:api',
  388. ])
  389. ->name('api.intelligent-exams.store');
  390. // 智能出卷任务状态查询
  391. Route::get('/intelligent-exams/status/{taskId}', [IntelligentExamController::class, 'status'])
  392. ->withoutMiddleware([
  393. Authenticate::class,
  394. 'auth',
  395. 'auth:sanctum',
  396. 'auth:api',
  397. ])
  398. ->name('api.intelligent-exams.status');
  399. // 学情报告对外接口:生成并返回学情报告 PDF
  400. Route::post('/exam-analysis/report', [ExamAnalysisApiController::class, 'store'])
  401. ->withoutMiddleware([
  402. Authenticate::class,
  403. 'auth',
  404. 'auth:sanctum',
  405. 'auth:api',
  406. ])
  407. ->name('api.exam-analysis.report');
  408. // 学情报告任务状态查询
  409. Route::get('/exam-analysis/status/{taskId}', [ExamAnalysisApiController::class, 'status'])
  410. ->withoutMiddleware([
  411. Authenticate::class,
  412. 'auth',
  413. 'auth:sanctum',
  414. 'auth:api',
  415. ])
  416. ->name('api.exam-analysis.status');
  417. /*
  418. |--------------------------------------------------------------------------
  419. | 错题本 API 路由
  420. |--------------------------------------------------------------------------
  421. */
  422. // 获取错题列表
  423. Route::get('/mistake-book', [MistakeBookController::class, 'listMistakes'])
  424. ->withoutMiddleware([
  425. Authenticate::class,
  426. 'auth',
  427. 'auth:sanctum',
  428. 'auth:api',
  429. ])
  430. ->name('api.mistake-book.list');
  431. // 新增错题
  432. Route::post('/mistake-book', [MistakeBookController::class, 'addMistake'])
  433. ->withoutMiddleware([
  434. Authenticate::class,
  435. 'auth',
  436. 'auth:sanctum',
  437. 'auth:api',
  438. ])
  439. ->name('api.mistake-book.create');
  440. // 获取单条错题详情
  441. Route::get('/mistake-book/{mistakeId}', [MistakeBookController::class, 'getMistakeDetail'])
  442. ->withoutMiddleware([
  443. Authenticate::class,
  444. 'auth',
  445. 'auth:sanctum',
  446. 'auth:api',
  447. ])
  448. ->whereNumber('mistakeId')
  449. ->name('api.mistake-book.detail');
  450. // 获取错题统计概要
  451. Route::get('/mistake-book/summary', [MistakeBookController::class, 'getSummary'])
  452. ->withoutMiddleware([
  453. Authenticate::class,
  454. 'auth',
  455. 'auth:sanctum',
  456. 'auth:api',
  457. ])
  458. ->name('api.mistake-book.summary');
  459. // 获取错误模式分析
  460. Route::get('/mistake-book/analytics/mistake-pattern', [MistakeBookController::class, 'getMistakePatterns'])
  461. ->withoutMiddleware([
  462. Authenticate::class,
  463. 'auth',
  464. 'auth:sanctum',
  465. 'auth:api',
  466. ])
  467. ->name('api.mistake-book.patterns');
  468. // 收藏/取消收藏错题
  469. Route::post('/mistake-book/{mistakeId}/favorite', [MistakeBookController::class, 'toggleFavorite'])
  470. ->withoutMiddleware([
  471. Authenticate::class,
  472. 'auth',
  473. 'auth:sanctum',
  474. 'auth:api',
  475. ])
  476. ->name('api.mistake-book.favorite');
  477. // 标记错题已复习
  478. Route::post('/mistake-book/{mistakeId}/review', [MistakeBookController::class, 'markReviewed'])
  479. ->withoutMiddleware([
  480. Authenticate::class,
  481. 'auth',
  482. 'auth:sanctum',
  483. 'auth:api',
  484. ])
  485. ->name('api.mistake-book.review');
  486. // 加入重练清单
  487. Route::post('/mistake-book/{mistakeId}/retry-list', [MistakeBookController::class, 'addToRetryList'])
  488. ->withoutMiddleware([
  489. Authenticate::class,
  490. 'auth',
  491. 'auth:sanctum',
  492. 'auth:api',
  493. ])
  494. ->name('api.mistake-book.retry-list');
  495. // 推荐练习题
  496. Route::post('/mistake-book/recommend-practice', [MistakeBookController::class, 'recommendPractice'])
  497. ->withoutMiddleware([
  498. Authenticate::class,
  499. 'auth',
  500. 'auth:sanctum',
  501. 'auth:api',
  502. ])
  503. ->name('api.mistake-book.recommend-practice');
  504. // 获取错题本快照数据(仪表板用)
  505. Route::get('/mistake-book/snapshot', [MistakeBookController::class, 'getSnapshot'])
  506. ->withoutMiddleware([
  507. Authenticate::class,
  508. 'auth',
  509. 'auth:sanctum',
  510. 'auth:api',
  511. ])
  512. ->name('api.mistake-book.snapshot');
  513. /*
  514. |--------------------------------------------------------------------------
  515. | 错题复习状态管理 API 路由
  516. |--------------------------------------------------------------------------
  517. */
  518. Route::post('/mistake-book/{mistakeId}/review-status', [MistakeBookController::class, 'updateReviewStatus'])
  519. ->withoutMiddleware([
  520. Authenticate::class,
  521. 'auth',
  522. 'auth:sanctum',
  523. 'auth:api',
  524. ])
  525. ->name('api.mistake-book.review-status.update');
  526. Route::get('/mistake-book/{mistakeId}/review-status', [MistakeBookController::class, 'getReviewStatus'])
  527. ->withoutMiddleware([
  528. Authenticate::class,
  529. 'auth',
  530. 'auth:sanctum',
  531. 'auth:api',
  532. ])
  533. ->name('api.mistake-book.review-status.get');
  534. Route::post('/mistake-book/{mistakeId}/increment-review', [MistakeBookController::class, 'incrementReview'])
  535. ->withoutMiddleware([
  536. Authenticate::class,
  537. 'auth',
  538. 'auth:sanctum',
  539. 'auth:api',
  540. ])
  541. ->name('api.mistake-book.increment-review');
  542. Route::post('/mistake-book/{mistakeId}/reset-review', [MistakeBookController::class, 'resetReview'])
  543. ->withoutMiddleware([
  544. Authenticate::class,
  545. 'auth',
  546. 'auth:sanctum',
  547. 'auth:api',
  548. ])
  549. ->name('api.mistake-book.reset-review');
  550. /*
  551. |--------------------------------------------------------------------------
  552. | 错题批量操作 API 路由
  553. |--------------------------------------------------------------------------
  554. */
  555. Route::post('/mistake-book/batch-operation', [MistakeBookController::class, 'batchOperation'])
  556. ->withoutMiddleware([
  557. Authenticate::class,
  558. 'auth',
  559. 'auth:sanctum',
  560. 'auth:api',
  561. ])
  562. ->name('api.mistake-book.batch-operation');
  563. Route::post('/mistake-book/batch/mark-reviewed', [MistakeBookController::class, 'batchMarkReviewed'])
  564. ->withoutMiddleware([
  565. Authenticate::class,
  566. 'auth',
  567. 'auth:sanctum',
  568. 'auth:api',
  569. ])
  570. ->name('api.mistake-book.batch.mark-reviewed');
  571. Route::post('/mistake-book/batch/mark-mastered', [MistakeBookController::class, 'batchMarkMastered'])
  572. ->withoutMiddleware([
  573. Authenticate::class,
  574. 'auth',
  575. 'auth:sanctum',
  576. 'auth:api',
  577. ])
  578. ->name('api.mistake-book.batch.mark-mastered');
  579. Route::post('/mistake-book/batch/add-to-retry-list', [MistakeBookController::class, 'batchAddToRetryList'])
  580. ->withoutMiddleware([
  581. Authenticate::class,
  582. 'auth',
  583. 'auth:sanctum',
  584. 'auth:api',
  585. ])
  586. ->name('api.mistake-book.batch.add-to-retry-list');
  587. Route::post('/mistake-book/batch/remove-from-retry-list', [MistakeBookController::class, 'batchRemoveFromRetryList'])
  588. ->withoutMiddleware([
  589. Authenticate::class,
  590. 'auth',
  591. 'auth:sanctum',
  592. 'auth:api',
  593. ])
  594. ->name('api.mistake-book.batch.remove-from-retry-list');
  595. Route::post('/mistake-book/batch/set-error-type', [MistakeBookController::class, 'batchSetErrorType'])
  596. ->withoutMiddleware([
  597. Authenticate::class,
  598. 'auth',
  599. 'auth:sanctum',
  600. 'auth:api',
  601. ])
  602. ->name('api.mistake-book.batch.set-error-type');
  603. Route::post('/mistake-book/batch/set-importance', [MistakeBookController::class, 'batchSetImportance'])
  604. ->withoutMiddleware([
  605. Authenticate::class,
  606. 'auth',
  607. 'auth:sanctum',
  608. 'auth:api',
  609. ])
  610. ->name('api.mistake-book.batch.set-importance');
  611. Route::post('/mistake-book/batch/toggle-favorite', [MistakeBookController::class, 'batchToggleFavorite'])
  612. ->withoutMiddleware([
  613. Authenticate::class,
  614. 'auth',
  615. 'auth:sanctum',
  616. 'auth:api',
  617. ])
  618. ->name('api.mistake-book.batch.toggle-favorite');
  619. /*
  620. |--------------------------------------------------------------------------
  621. | 知识点掌握情况 API 路由
  622. |--------------------------------------------------------------------------
  623. */
  624. use App\Http\Controllers\Api\KnowledgeMasteryController;
  625. // 获取学生知识点掌握情况统计
  626. Route::get('/knowledge-mastery/stats/{studentId}', [KnowledgeMasteryController::class, 'stats'])
  627. ->where('studentId', '[0-9]+') // 限制为数字
  628. ->withoutMiddleware([
  629. Authenticate::class,
  630. 'auth',
  631. 'auth:sanctum',
  632. 'auth:api',
  633. ])
  634. ->name('api.knowledge-mastery.stats');
  635. // 获取学生知识点掌握摘要
  636. Route::get('/knowledge-mastery/summary/{studentId}', [KnowledgeMasteryController::class, 'summary'])
  637. ->where('studentId', '[0-9]+') // 限制为数字
  638. ->withoutMiddleware([
  639. Authenticate::class,
  640. 'auth',
  641. 'auth:sanctum',
  642. 'auth:api',
  643. ])
  644. ->name('api.knowledge-mastery.summary');
  645. // 获取学生知识点图谱数据
  646. Route::get('/knowledge-mastery/graph/{studentId}', [KnowledgeMasteryController::class, 'graph'])
  647. ->where('studentId', '[0-9]+') // 限制为数字
  648. ->withoutMiddleware([
  649. Authenticate::class,
  650. 'auth',
  651. 'auth:sanctum',
  652. 'auth:api',
  653. ])
  654. ->name('api.knowledge-mastery.graph');
  655. // 获取学生知识点图谱快照列表
  656. Route::get('/knowledge-mastery/graph/snapshots/{studentId}', [KnowledgeMasteryController::class, 'graphSnapshots'])
  657. ->where('studentId', '[0-9]+') // 限制为数字
  658. ->withoutMiddleware([
  659. Authenticate::class,
  660. 'auth',
  661. 'auth:sanctum',
  662. 'auth:api',
  663. ])
  664. ->name('api.knowledge-mastery.graph.snapshots');
  665. // 获取学生知识点快照列表(简化路径)
  666. Route::get('/knowledge-mastery/snapshots/{studentId}', [KnowledgeMasteryController::class, 'snapshots'])
  667. ->where('studentId', '[0-9]+') // 限制为数字
  668. ->withoutMiddleware([
  669. Authenticate::class,
  670. 'auth',
  671. 'auth:sanctum',
  672. 'auth:api',
  673. ])
  674. ->name('api.knowledge-mastery.snapshots');
  675. // 创建知识点掌握度快照
  676. Route::post('/knowledge-mastery/snapshot/{studentId}', [KnowledgeMasteryController::class, 'createSnapshot'])
  677. ->where('studentId', '[0-9]+') // 限制为数字
  678. ->withoutMiddleware([
  679. Authenticate::class,
  680. 'auth',
  681. 'auth:sanctum',
  682. 'auth:api',
  683. ])
  684. ->name('api.knowledge-mastery.snapshot.create');
  685. /*
  686. |--------------------------------------------------------------------------
  687. | 教材管理 API 路由
  688. |--------------------------------------------------------------------------
  689. */
  690. // 获取教材列表(按年级排序)
  691. Route::get('/textbooks', [TextbookApiController::class, 'index'])
  692. ->name('api.textbooks.index');
  693. // 根据年级获取教材
  694. Route::get('/textbooks/grade/{grade}', [TextbookApiController::class, 'getByGrade'])
  695. ->name('api.textbooks.by-grade');
  696. // 获取教材系列列表(必须在 {id} 路由之前定义)
  697. Route::get('/textbooks/series', [TextbookApiController::class, 'getSeries'])
  698. ->name('api.textbooks.series');
  699. // 获取年级枚举
  700. Route::get('/textbooks/grades', [TextbookApiController::class, 'getGradeEnums'])
  701. ->name('api.textbooks.grades');
  702. // 获取单个教材详情
  703. Route::get('/textbooks/{id}', [TextbookApiController::class, 'show'])
  704. ->name('api.textbooks.show');
  705. // 获取教材目录
  706. Route::get('/textbooks/{id}/catalog', [TextbookApiController::class, 'getCatalog'])
  707. ->name('api.textbooks.catalog');
  708. /*
  709. |--------------------------------------------------------------------------
  710. | MathRecSys 集成 API 路由
  711. |--------------------------------------------------------------------------
  712. */
  713. use App\Http\Controllers\Api\StudentController;
  714. // 健康检查
  715. Route::get('/mathrecsys/health', [StudentController::class, 'checkServiceHealth'])->name('api.mathrecsys.health');
  716. // 学生相关 API
  717. Route::prefix('mathrecsys/students')->name('api.mathrecsys.students.')->group(function () {
  718. // 获取学生完整信息
  719. Route::get('{studentId}', [StudentController::class, 'show'])
  720. ->where('studentId', '[0-9]+') // 限制为数字
  721. ->name('show');
  722. // 获取个性化推荐
  723. Route::get('{studentId}/recommendations', [StudentController::class, 'getRecommendations'])
  724. ->where('studentId', '[0-9]+') // 限制为数字
  725. ->name('recommendations');
  726. // 获取学习轨迹
  727. Route::get('{studentId}/trajectory', [StudentController::class, 'getTrajectory'])
  728. ->where('studentId', '[0-9]+') // 限制为数字
  729. ->name('trajectory');
  730. // 获取学习建议
  731. Route::get('{studentId}/suggestions', [StudentController::class, 'getSuggestions'])
  732. ->where('studentId', '[0-9]+') // 限制为数字
  733. ->name('suggestions');
  734. // 智能分析题目
  735. Route::post('{studentId}/analyze', [StudentController::class, 'analyzeQuestion'])
  736. ->where('studentId', '[0-9]+') // 限制为数字
  737. ->name('analyze');
  738. // 更新掌握度
  739. Route::put('{studentId}/mastery', [StudentController::class, 'updateMastery'])
  740. ->where('studentId', '[0-9]+') // 限制为数字
  741. ->name('update-mastery');
  742. });
  743. // 班级分析 API
  744. Route::prefix('mathrecsys/classes')->name('api.mathrecsys.classes.')->group(function () {
  745. Route::get('{classId}/analysis', [StudentController::class, 'classAnalysis'])
  746. ->where('classId', '[0-9]+') // 限制为数字
  747. ->name('analysis');
  748. });
  749. // 测试 API
  750. Route::get('/mathrecsys/test', function () {
  751. return response()->json([
  752. 'success' => true,
  753. 'message' => 'MathRecSys API integration is working',
  754. 'timestamp' => now()->toISOString()
  755. ]);
  756. })->name('api.mathrecsys.test');
  757. // 测试OCR题目生成API调用
  758. Route::post('/test-ocr-generation', function () {
  759. try {
  760. $service = new \App\Services\QuestionBankService();
  761. // 模拟前端传递的OCR题目数据
  762. $questions = [
  763. [
  764. 'id' => 1,
  765. 'content' => '计算:2+3-4'
  766. ],
  767. [
  768. 'id' => 2,
  769. 'content' => '解方程:x+5=10'
  770. ]
  771. ];
  772. Log::info('开始测试OCR题目生成', [
  773. 'questions_count' => count($questions),
  774. 'ocr_record_id' => 12
  775. ]);
  776. // 使用异步API,系统自动生成回调URL
  777. $response = $service->generateQuestionsFromOcrAsync(
  778. $questions,
  779. '高一',
  780. '数学',
  781. 12, // OCR记录ID
  782. null, // 让系统自动生成回调URL
  783. 'api.ocr.callback' // 回调路由名称
  784. );
  785. Log::info('OCR题目生成响应', [
  786. 'response' => $response,
  787. 'status' => $response['status'] ?? 'unknown',
  788. 'task_id' => $response['task_id'] ?? 'N/A'
  789. ]);
  790. return response()->json([
  791. 'success' => true,
  792. 'message' => 'OCR题目生成测试完成',
  793. 'data' => $response
  794. ]);
  795. } catch (\Exception $e) {
  796. Log::error('测试OCR题目生成失败', [
  797. 'error' => $e->getMessage(),
  798. 'trace' => $e->getTraceAsString()
  799. ]);
  800. return response()->json([
  801. 'success' => false,
  802. 'error' => $e->getMessage()
  803. ], 500);
  804. }
  805. })->name('api.test.ocr.generation');
  806. /*
  807. |--------------------------------------------------------------------------
  808. | 学生作答分析 API 路由
  809. |--------------------------------------------------------------------------
  810. */
  811. // 提交学生作答结果
  812. Route::post('/student-answers/analyze', [StudentAnswerAnalysisController::class, 'submitAnswers'])
  813. ->withoutMiddleware([
  814. Authenticate::class,
  815. 'auth',
  816. 'auth:sanctum',
  817. 'auth:api',
  818. ])
  819. ->name('api.student-answers.analyze');
  820. // 查询分析任务状态
  821. Route::get('/student-answers/analysis/status/{taskId}', [StudentAnswerAnalysisController::class, 'getAnalysisStatus'])
  822. ->withoutMiddleware([
  823. Authenticate::class,
  824. 'auth',
  825. 'auth:sanctum',
  826. 'auth:api',
  827. ])
  828. ->name('api.student-answers.analysis.status');
  829. // 获取学生学习历史
  830. Route::get('/student-answers/history/{studentId}', [StudentAnswerAnalysisController::class, 'getStudentLearningHistory'])
  831. ->withoutMiddleware([
  832. Authenticate::class,
  833. 'auth',
  834. 'auth:sanctum',
  835. 'auth:api',
  836. ])
  837. ->name('api.student-answers.history');
  838. /*
  839. |--------------------------------------------------------------------------
  840. | 考试答题分析 API 路由(步骤级分析)
  841. |--------------------------------------------------------------------------
  842. */
  843. use App\Http\Controllers\Api\ExamAnswerAnalysisController;
  844. use App\Http\Controllers\Api\PaperSubmitAnalysisController;
  845. use App\Http\Controllers\Api\HealthCheckController;
  846. // 分析考试答题数据
  847. Route::post('/exam-answer-analysis', [ExamAnswerAnalysisController::class, 'analyze'])
  848. ->withoutMiddleware([
  849. Authenticate::class,
  850. 'auth',
  851. 'auth:sanctum',
  852. 'auth:api',
  853. ])
  854. ->name('api.exam-answer-analysis.analyze');
  855. // 获取分析结果
  856. Route::get('/exam-answer-analysis/{student_id}/{exam_id}', [ExamAnswerAnalysisController::class, 'getAnalysisResult'])
  857. ->withoutMiddleware([
  858. Authenticate::class,
  859. 'auth',
  860. 'auth:sanctum',
  861. 'auth:api',
  862. ])
  863. ->where('student_id', '.*')
  864. ->where('exam_id', '.*')
  865. ->name('api.exam-answer-analysis.result');
  866. // 获取学生历史分析记录
  867. Route::get('/exam-answer-analysis/history/{student_id}', [ExamAnswerAnalysisController::class, 'getHistory'])
  868. ->withoutMiddleware([
  869. Authenticate::class,
  870. 'auth',
  871. 'auth:sanctum',
  872. 'auth:api',
  873. ])
  874. ->where('student_id', '.*')
  875. ->name('api.exam-answer-analysis.history');
  876. // 获取知识点掌握度趋势
  877. Route::get('/exam-answer-analysis/mastery-trend/{student_id}', [ExamAnswerAnalysisController::class, 'getMasteryTrend'])
  878. ->withoutMiddleware([
  879. Authenticate::class,
  880. 'auth',
  881. 'auth:sanctum',
  882. 'auth:api',
  883. ])
  884. ->where('student_id', '.*')
  885. ->name('api.exam-answer-analysis.mastery-trend');
  886. // 获取智能出卷推荐
  887. Route::get('/exam-answer-analysis/smart-quiz/{student_id}', [ExamAnswerAnalysisController::class, 'getSmartQuizRecommendation'])
  888. ->withoutMiddleware([
  889. Authenticate::class,
  890. 'auth',
  891. 'auth:sanctum',
  892. 'auth:api',
  893. ])
  894. ->where('student_id', '.*')
  895. ->name('api.exam-answer-analysis.smart-quiz');
  896. // 导出分析报告
  897. Route::get('/exam-answer-analysis/export/{student_id}/{exam_id}', [ExamAnswerAnalysisController::class, 'export'])
  898. ->withoutMiddleware([
  899. Authenticate::class,
  900. 'auth',
  901. 'auth:sanctum',
  902. 'auth:api',
  903. ])
  904. ->where('student_id', '.*')
  905. ->where('exam_id', '.*')
  906. ->name('api.exam-answer-analysis.export');
  907. // 批量分析多个学生的考试数据
  908. Route::post('/exam-answer-analysis/batch', [ExamAnswerAnalysisController::class, 'batchAnalyze'])
  909. ->withoutMiddleware([
  910. Authenticate::class,
  911. 'auth',
  912. 'auth:sanctum',
  913. 'auth:api',
  914. ])
  915. ->name('api.exam-answer-analysis.batch');
  916. Route::get('/tasks/status/{taskId}', function (string $taskId) {
  917. $task = app(\App\Services\TaskManager::class)->getTaskStatus($taskId);
  918. if (!$task) {
  919. return response()->json([
  920. 'success' => false,
  921. 'message' => '任务不存在',
  922. ], 404);
  923. }
  924. return response()->json([
  925. 'success' => true,
  926. 'data' => $task,
  927. ]);
  928. })->name('api.tasks.status');
  929. /*
  930. |--------------------------------------------------------------------------
  931. | 试卷提交分析 API 路由(前端提交答题数据)
  932. |--------------------------------------------------------------------------
  933. */
  934. // 提交试卷答题数据进行分析
  935. Route::post('/paper-submit-analysis', [PaperSubmitAnalysisController::class, 'analyze'])
  936. ->withoutMiddleware([
  937. Authenticate::class,
  938. 'auth',
  939. 'auth:sanctum',
  940. 'auth:api',
  941. ])
  942. ->name('api.paper-submit-analysis.analyze');
  943. // 获取试卷分析结果
  944. Route::get('/paper-submit-analysis/{paperId}', [PaperSubmitAnalysisController::class, 'getResult'])
  945. ->withoutMiddleware([
  946. Authenticate::class,
  947. 'auth',
  948. 'auth:sanctum',
  949. 'auth:api',
  950. ])
  951. ->name('api.paper-submit-analysis.result');
  952. /*
  953. |--------------------------------------------------------------------------
  954. | 健康检查 API 路由
  955. |--------------------------------------------------------------------------
  956. */
  957. // 检查系统健康状态
  958. Route::get('/health', [HealthCheckController::class, 'index'])
  959. ->withoutMiddleware([
  960. Authenticate::class,
  961. 'auth',
  962. 'auth:sanctum',
  963. 'auth:api',
  964. ])
  965. ->name('api.health.index');