katexRenderer = new KatexRenderer(); } /** * 生成试卷 PDF(不含答案) */ public function generateExamPdf(string $paperId): ?string { Log::info('generateExamPdf 开始:', ['paper_id' => $paperId]); $url = $this->renderAndStoreExamPdf($paperId, includeAnswer: false, suffix: 'exam'); Log::info('generateExamPdf url 生成结果:', ['paper_id' => $paperId, 'url' => $url]); // 如果生成成功,将 URL 写入数据库 if ($url) { $this->savePdfUrlToDatabase($paperId, 'exam_pdf_url', $url); } return $url; } /** * 生成判卷 PDF(含答案与解析) */ public function generateGradingPdf(string $paperId): ?string { Log::info('generateGradingPdf 开始:', ['paper_id' => $paperId]); $url = $this->renderAndStoreExamPdf($paperId, includeAnswer: true, suffix: 'grading', useGradingView: true); Log::info('generateGradingPdf url 生成结果:', ['paper_id' => $paperId, 'url' => $url]); // 如果生成成功,将 URL 写入数据库 if ($url) { $this->savePdfUrlToDatabase($paperId, 'grading_pdf_url', $url); } return $url; } /** * 【优化方案】生成统一PDF(卷子 + 判卷一页完成) * 效率提升40-50%,只需生成一次PDF */ public function generateUnifiedPdf(string $paperId): ?string { Log::info('generateUnifiedPdf 开始(终极优化版本,直接HTML合并生成PDF):', ['paper_id' => $paperId]); try { // 步骤1:同时渲染两个页面的HTML Log::info('generateUnifiedPdf: 开始渲染试卷HTML', ['paper_id' => $paperId]); $examHtml = $this->renderExamHtml($paperId, includeAnswer: false, useGradingView: false); if (!$examHtml) { Log::error('ExamPdfExportService: 渲染卷子HTML失败', ['paper_id' => $paperId]); return null; } Log::info('generateUnifiedPdf: 试卷HTML渲染完成', ['paper_id' => $paperId, 'length' => strlen($examHtml)]); Log::info('generateUnifiedPdf: 开始渲染判卷HTML', ['paper_id' => $paperId]); $gradingHtml = $this->renderExamHtml($paperId, includeAnswer: true, useGradingView: true); if (!$gradingHtml) { Log::error('ExamPdfExportService: 渲染判卷HTML失败', ['paper_id' => $paperId]); return null; } Log::info('generateUnifiedPdf: 判卷HTML渲染完成', ['paper_id' => $paperId, 'length' => strlen($gradingHtml)]); // 步骤2:插入分页符,合并HTML Log::info('generateUnifiedPdf: 开始合并HTML(保留原始样式)', ['paper_id' => $paperId]); $unifiedHtml = $this->mergeHtmlWithPageBreak($examHtml, $gradingHtml); if (!$unifiedHtml) { Log::error('ExamPdfExportService: HTML合并失败', ['paper_id' => $paperId]); return null; } Log::info('generateUnifiedPdf: HTML合并完成(将直接生成PDF,不使用pdfunite)', ['paper_id' => $paperId, 'length' => strlen($unifiedHtml)]); // 步骤3:一次性生成PDF(只需20-25秒,比原来节省10-25秒) Log::info('generateUnifiedPdf: 开始使用buildPdf直接生成PDF(不使用pdfunite)', ['paper_id' => $paperId]); $pdfBinary = $this->buildPdf($unifiedHtml); if (!$pdfBinary) { Log::error('ExamPdfExportService: 生成统一PDF失败', ['paper_id' => $paperId]); return null; } Log::info('generateUnifiedPdf: PDF生成完成', ['paper_id' => $paperId, 'pdf_size' => strlen($pdfBinary)]); // 步骤4:保存PDF $path = "exams/{$paperId}_all.pdf"; Log::info('generateUnifiedPdf: 开始保存PDF到云存储', ['paper_id' => $paperId, 'path' => $path]); $url = $this->pdfStorageService->put($path, $pdfBinary); if (!$url) { Log::error('ExamPdfExportService: 保存统一PDF失败', ['path' => $path]); return null; } Log::info('generateUnifiedPdf: PDF保存完成', ['paper_id' => $paperId, 'url' => $url]); // 步骤5:保存URL到数据库(存储到all_pdf_url字段) Log::info('generateUnifiedPdf: 开始保存URL到数据库', ['paper_id' => $paperId, 'field' => 'all_pdf_url']); $this->savePdfUrlToDatabase($paperId, 'all_pdf_url', $url); Log::info('generateUnifiedPdf: URL保存完成', ['paper_id' => $paperId]); Log::info('generateUnifiedPdf 全部完成(终极优化:直接HTML合并生成一份PDF)', [ 'paper_id' => $paperId, 'url' => $url, 'pdf_size' => strlen($pdfBinary), 'method' => 'direct HTML merge to PDF (no pdfunite)' ]); return $url; } catch (\Throwable $e) { Log::error('generateUnifiedPdf 失败', [ 'paper_id' => $paperId, 'error' => $e->getMessage(), 'trace' => $e->getTraceAsString(), ]); return null; } } /** * 生成合并PDF(试卷 + 判卷) * 先分别生成两个PDF,然后合并 */ public function generateMergedPdf(string $paperId): ?string { Log::info('generateMergedPdf 开始:', ['paper_id' => $paperId]); $tempDir = storage_path("app/temp"); if (!is_dir($tempDir)) { mkdir($tempDir, 0755, true); } $examPdfPath = null; $gradingPdfPath = null; $mergedPdfPath = null; try { // 先生成试卷PDF $examPdfUrl = $this->generateExamPdf($paperId); if (!$examPdfUrl) { Log::error('ExamPdfExportService: 生成试卷PDF失败', ['paper_id' => $paperId]); return null; } // 再生成判卷PDF $gradingPdfUrl = $this->generateGradingPdf($paperId); if (!$gradingPdfUrl) { Log::error('ExamPdfExportService: 生成判卷PDF失败', ['paper_id' => $paperId]); return null; } // 【修复】下载PDF文件到本地临时目录 Log::info('开始下载PDF文件到本地', [ 'exam_url' => $examPdfUrl, 'grading_url' => $gradingPdfUrl ]); $examPdfPath = $tempDir . "/{$paperId}_exam.pdf"; $gradingPdfPath = $tempDir . "/{$paperId}_grading.pdf"; // 下载试卷PDF $examContent = Http::get($examPdfUrl)->body(); if (empty($examContent)) { Log::error('ExamPdfExportService: 下载试卷PDF失败', ['url' => $examPdfUrl]); return null; } file_put_contents($examPdfPath, $examContent); // 下载判卷PDF $gradingContent = Http::get($gradingPdfUrl)->body(); if (empty($gradingContent)) { Log::error('ExamPdfExportService: 下载判卷PDF失败', ['url' => $gradingPdfUrl]); return null; } file_put_contents($gradingPdfPath, $gradingContent); Log::info('PDF文件下载完成', [ 'exam_size' => filesize($examPdfPath), 'grading_size' => filesize($gradingPdfPath) ]); // 合并PDF文件 $mergedPdfPath = $tempDir . "/{$paperId}_merged.pdf"; $merged = $this->pdfMerger->merge([$examPdfPath, $gradingPdfPath], $mergedPdfPath); if (!$merged) { Log::error('ExamPdfExportService: PDF文件合并失败', [ 'tool' => $this->pdfMerger->getMergeTool() ]); return null; } // 读取合并后的PDF内容并上传到云存储 $mergedPdfContent = file_get_contents($mergedPdfPath); $path = "exams/{$paperId}_all.pdf"; $mergedUrl = $this->pdfStorageService->put($path, $mergedPdfContent); if (!$mergedUrl) { Log::error('ExamPdfExportService: 保存合并PDF失败', ['path' => $path]); return null; } // 保存到数据库的all_pdf_url字段 $this->saveAllPdfUrlToDatabase($paperId, $mergedUrl); Log::info('generateMergedPdf 完成:', [ 'paper_id' => $paperId, 'url' => $mergedUrl, 'tool' => $this->pdfMerger->getMergeTool() ]); return $mergedUrl; } catch (\Throwable $e) { Log::error('ExamPdfExportService: 生成合并PDF失败', [ 'paper_id' => $paperId, 'error' => $e->getMessage(), 'trace' => $e->getTraceAsString(), ]); return null; } finally { // 【修复】清理临时文件 $tempFiles = [$examPdfPath, $gradingPdfPath, $mergedPdfPath]; foreach ($tempFiles as $file) { if ($file && file_exists($file)) { @unlink($file); } } Log::debug('清理临时文件完成'); } } /** * 将URL转换为本地文件路径 */ private function convertUrlToPath(string $url): ?string { // 如果是本地存储,URL格式类似:/storage/exams/paper_id_exam.pdf // 需要转换为绝对路径 if (strpos($url, '/storage/') === 0) { return public_path(ltrim($url, '/')); } // 如果是完整路径,直接返回 if (strpos($url, '/') === 0 && file_exists($url)) { return $url; } // 如果是相对路径,转换为绝对路径 $path = public_path($url); if (file_exists($path)) { return $path; } return null; } /** * 保存合并PDF URL到数据库 */ private function saveAllPdfUrlToDatabase(string $paperId, string $url): void { try { \App\Models\Paper::where('paper_id', $paperId)->update([ 'all_pdf_url' => $url ]); Log::debug('保存all_pdf_url成功', ['paper_id' => $paperId, 'url' => $url]); } catch (\Exception $e) { Log::error('保存all_pdf_url失败', [ 'paper_id' => $paperId, 'url' => $url, 'error' => $e->getMessage() ]); throw $e; } } /** * 生成学情分析 PDF */ public function generateAnalysisReportPdf(string $paperId, string $studentId, ?string $recordId = null): ?string { if (function_exists('set_time_limit')) { @set_time_limit(240); } try { // 【调试】打印输入参数 Log::info('ExamPdfExportService: 开始生成学情分析PDF', [ 'paper_id' => $paperId, 'student_id' => $studentId, 'record_id' => $recordId, ]); // 构建分析数据 $analysisData = $this->buildAnalysisData($paperId, $studentId); if (!$analysisData) { Log::warning('ExamPdfExportService: buildAnalysisData返回空数据', [ 'paper_id' => $paperId, 'student_id' => $studentId, ]); return null; } Log::info('ExamPdfExportService: buildAnalysisData返回数据', [ 'paper_id' => $paperId, 'student_id' => $studentId, 'analysisData_keys' => array_keys($analysisData), 'mastery_count' => count($analysisData['mastery']['items'] ?? []), 'questions_count' => count($analysisData['questions'] ?? []), ]); // 创建DTO $dto = ExamAnalysisDataDto::fromArray($analysisData); $payloadDto = ReportPayloadDto::fromExamAnalysisDataDto($dto); // 【调试】打印传给模板的数据 $templateData = $payloadDto->toArray(); Log::info('ExamPdfExportService: 传给模板的数据', [ 'paper' => $templateData['paper'] ?? null, 'student' => $templateData['student'] ?? null, 'mastery' => $templateData['mastery'] ?? null, 'parent_mastery_levels' => $templateData['parent_mastery_levels'] ?? null, // 新增:检查父节点掌握度 'questions_count' => count($templateData['questions'] ?? []), 'insights_count' => count($templateData['question_insights'] ?? []), 'recommendations_count' => count($templateData['recommendations'] ?? []), ]); // 渲染HTML $html = view('exam-analysis.pdf-report', $templateData)->render(); if (!$html) { Log::error('ExamPdfExportService: 渲染HTML为空', ['paper_id' => $paperId]); return null; } // 生成PDF $pdfBinary = $this->buildPdf($html); if (!$pdfBinary) { return null; } // 保存PDF $version = time(); $path = "analysis_reports/{$paperId}_{$studentId}_{$version}.pdf"; $url = $this->pdfStorageService->put($path, $pdfBinary); if (!$url) { Log::error('ExamPdfExportService: 保存学情PDF失败', ['path' => $path]); return null; } // 保存URL到数据库 $this->saveAnalysisPdfUrl($paperId, $studentId, $recordId, $url); return $url; } catch (\Throwable $e) { Log::error('ExamPdfExportService: 生成学情分析PDF失败', [ 'paper_id' => $paperId, 'student_id' => $studentId, 'record_id' => $recordId, 'error' => $e->getMessage(), 'exception' => get_class($e), 'trace' => $e->getTraceAsString(), ]); return null; } } /** * 渲染并存储试卷PDF */ private function renderAndStoreExamPdf( string $paperId, bool $includeAnswer, string $suffix, bool $useGradingView = false ): ?string { // 放宽脚本执行时间 if (function_exists('set_time_limit')) { @set_time_limit(240); } try { $html = $this->renderExamHtml($paperId, $includeAnswer, $useGradingView); if (!$html) { Log::error('ExamPdfExportService: 渲染HTML为空', [ 'paper_id' => $paperId, 'include_answer' => $includeAnswer, 'use_grading_view' => $useGradingView, ]); return null; } $pdfBinary = $this->buildPdf($html); if (!$pdfBinary) { Log::error('ExamPdfExportService: buildPdf为空', [ 'paper_id' => $paperId, 'include_answer' => $includeAnswer, 'use_grading_view' => $useGradingView, ]); return null; } $path = "exams/{$paperId}_{$suffix}.pdf"; $url = $this->pdfStorageService->put($path, $pdfBinary); if (!$url) { Log::error('ExamPdfExportService: 保存PDF失败', ['path' => $path]); return null; } return $url; } catch (\Throwable $e) { Log::error('ExamPdfExportService: 生成PDF失败', [ 'paper_id' => $paperId, 'suffix' => $suffix, 'error' => $e->getMessage(), 'exception' => get_class($e), 'trace' => $e->getTraceAsString(), ]); return null; } } /** * 渲染试卷HTML(重构版) */ private function renderExamHtml(string $paperId, bool $includeAnswer, bool $useGradingView): ?string { // 直接构造请求URL,使用路由生成HTML $routeName = $useGradingView ? 'filament.admin.auth.intelligent-exam.grading' : 'filament.admin.auth.intelligent-exam.pdf'; $url = route($routeName, ['paper_id' => $paperId, 'answer' => $includeAnswer ? 'true' : 'false']); // 使用HTTP客户端获取渲染后的HTML try { $response = Http::get($url); if ($response->successful()) { $html = $response->body(); if (!empty(trim($html))) { return $this->ensureUtf8Html($html); } else { Log::warning('ExamPdfExportService: HTTP返回的HTML为空,使用备用方案', [ 'paper_id' => $paperId, 'url' => $url, ]); } } } catch (\Exception $e) { Log::warning('ExamPdfExportService: 通过HTTP获取HTML失败,使用备用方案', [ 'paper_id' => $paperId, 'error' => $e->getMessage(), ]); } // 备用方案:直接渲染视图(如果路由不可用) try { $paper = Paper::with('questions')->find($paperId); if (!$paper) { Log::error('ExamPdfExportService: 试卷不存在,备用方案无法渲染', [ 'paper_id' => $paperId, 'include_answer' => $includeAnswer, 'use_grading_view' => $useGradingView, ]); return null; } // 检查试卷是否有题目 if ($paper->questions->isEmpty()) { Log::error('ExamPdfExportService: 试卷没有题目数据', [ 'paper_id' => $paperId, 'question_count' => 0, ]); return null; } $viewName = $useGradingView ? 'exam-pdf.grading' : 'exam-pdf.student'; $html = view($viewName, compact('paper'))->render(); if (empty(trim($html))) { Log::error('ExamPdfExportService: 视图渲染结果为空', [ 'paper_id' => $paperId, 'view_name' => $viewName, 'question_count' => $paper->questions->count(), ]); return null; } return $this->ensureUtf8Html($html); } catch (\Exception $e) { Log::error('ExamPdfExportService: 备用方案渲染失败', [ 'paper_id' => $paperId, 'error' => $e->getMessage(), 'trace' => $e->getTraceAsString(), ]); return null; } } /** * 构建分析数据(重构版) * 优先使用本地MySQL数据,减少API依赖 */ private function buildAnalysisData(string $paperId, string $studentId): ?array { // 【关键调试】确认方法被调用 Log::warning('ExamPdfExportService: buildAnalysisData方法被调用了!', [ 'paper_id' => $paperId, 'student_id' => $studentId, 'timestamp' => now()->toISOString() ]); $paper = Paper::with(['questions' => function ($query) { $query->orderBy('question_number')->orderBy('id'); }])->find($paperId); if (!$paper) { Log::warning('ExamPdfExportService: 未找到试卷,将尝试仅基于分析数据生成PDF', [ 'paper_id' => $paperId, 'student_id' => $studentId, ]); // 【修复】即使试卷不存在,也尝试基于分析数据生成PDF $paper = new \stdClass(); $paper->paper_id = $paperId; $paper->paper_name = "学情分析报告_{$studentId}_{$paperId}"; $paper->question_count = 0; $paper->total_score = 0; $paper->created_at = now(); $paper->questions = collect(); } $student = Student::find($studentId); $studentInfo = [ 'id' => $student?->student_id ?? $studentId, 'name' => $student?->name ?? $studentId, 'grade' => $student?->grade ?? '未知年级', 'class' => $student?->class_name ?? '未知班级', ]; // 【修改】直接从本地数据库获取分析数据(不再调用API) $analysisData = []; // 首先尝试从paper->analysis_id获取 if (!empty($paper->analysis_id)) { Log::info('ExamPdfExportService: 从本地数据库获取试卷分析数据', [ 'paper_id' => $paperId, 'student_id' => $studentId, 'analysis_id' => $paper->analysis_id ]); $analysisRecord = \DB::table('exam_analysis_results') ->where('id', $paper->analysis_id) ->where('student_id', $studentId) ->first(); if ($analysisRecord && !empty($analysisRecord->analysis_data)) { $analysisData = json_decode($analysisRecord->analysis_data, true); Log::info('ExamPdfExportService: 成功获取本地分析数据(通过analysis_id)', [ 'data_size' => strlen($analysisRecord->analysis_data) ]); } else { Log::warning('ExamPdfExportService: 未找到本地分析数据,将尝试其他方式', [ 'paper_id' => $paperId, 'student_id' => $studentId, 'analysis_id' => $paper->analysis_id ]); } } // 如果没有analysis_id或未找到数据,直接从exam_analysis_results表查询 if (empty($analysisData)) { Log::info('ExamPdfExportService: 直接从exam_analysis_results表查询分析数据', [ 'paper_id' => $paperId, 'student_id' => $studentId ]); $analysisRecord = \DB::table('exam_analysis_results') ->where('paper_id', $paperId) ->where('student_id', $studentId) ->first(); if ($analysisRecord && !empty($analysisRecord->analysis_data)) { $analysisData = json_decode($analysisRecord->analysis_data, true); Log::info('ExamPdfExportService: 成功获取本地分析数据(直接查询)', [ 'data_size' => strlen($analysisRecord->analysis_data), 'question_count' => count($analysisData['question_analysis'] ?? []) ]); } else { Log::warning('ExamPdfExportService: 未找到任何分析数据,将使用空数据', [ 'paper_id' => $paperId, 'student_id' => $studentId ]); } } // 【修复】优先使用analysisData中的knowledge_point_analysis数据 $masteryData = []; $parentMasteryLevels = []; // 新增:父节点掌握度数据 Log::info('ExamPdfExportService: 开始处理掌握度数据', [ 'student_id' => $studentId, 'analysisData_keys' => array_keys($analysisData), 'has_knowledge_point_analysis' => !empty($analysisData['knowledge_point_analysis']), ]); if (!empty($analysisData['knowledge_point_analysis'])) { // 将knowledge_point_analysis转换为buildMasterySummary期望的格式 foreach ($analysisData['knowledge_point_analysis'] as $kp) { $masteryData[] = [ 'kp_code' => $kp['kp_id'] ?? null, 'kp_name' => $kp['kp_id'] ?? '未知知识点', 'mastery_level' => $kp['mastery_level'] ?? 0, 'mastery_change' => $kp['change'] ?? null, ]; } // 【修复】基于所有兄弟节点历史数据计算父节点掌握度,并获取掌握度变化 try { // 获取本次考试涉及的知识点代码列表 $examKpCodes = array_column($masteryData, 'kp_code'); Log::info('ExamPdfExportService: 本次考试涉及的知识点', [ 'count' => count($examKpCodes), 'kp_codes' => $examKpCodes ]); // 获取上一个快照的数据(用于计算变化) // 如果没有其他试卷的记录,使用同一试卷的上一次快照 $lastSnapshot = DB::connection('mysql') ->table('knowledge_point_mastery_snapshots') ->where('student_id', $studentId) ->where('paper_id', $paper->paper_id) ->where('snapshot_id', '!=', "snap_{$paper->paper_id}_" . date('YmdHis')) ->latest('snapshot_time') ->first(); $previousMasteryData = []; if ($lastSnapshot) { $previousMasteryJson = json_decode($lastSnapshot->mastery_data, true); foreach ($previousMasteryJson as $kpCode => $data) { $previousMasteryData[$kpCode] = [ 'current_mastery' => $data['current_mastery'] ?? 0, 'previous_mastery' => $data['previous_mastery'] ?? null, ]; } Log::info('ExamPdfExportService: 获取到上一次快照数据', [ 'snapshot_time' => $lastSnapshot->snapshot_time, 'kp_count' => count($previousMasteryData) ]); } // 为当前知识点添加变化数据 foreach ($masteryData as &$item) { $kpCode = $item['kp_code']; if (isset($previousMasteryData[$kpCode])) { $previous = floatval($previousMasteryData[$kpCode]['previous_mastery'] ?? 0); $current = floatval($item['mastery_level']); $item['mastery_change'] = $current - $previous; } } unset($item); // 解除引用 // 获取所有父节点掌握度 $masteryOverview = $this->masteryCalculator->getStudentMasteryOverviewWithHierarchy($studentId); $allParentMasteryLevels = $masteryOverview['parent_mastery_levels'] ?? []; // 计算与本次考试相关的父节点掌握度(基于所有兄弟节点) $parentMasteryLevels = []; // 【修复】使用数据库查询正确匹配父子关系,而不是字符串前缀 foreach ($allParentMasteryLevels as $parentKpCode => $parentMastery) { // 查询这个父节点的所有子节点 $childNodes = DB::connection('mysql') ->table('knowledge_points') ->where('parent_kp_code', $parentKpCode) ->pluck('kp_code') ->toArray(); // 检查是否有子节点在本次考试中出现 $relevantChildren = array_intersect($examKpCodes, $childNodes); if (!empty($relevantChildren)) { // 【修复】计算父节点变化:基于所有子节点的平均变化 $childChanges = []; foreach ($relevantChildren as $childKpCode) { $previousChild = $previousMasteryData[$childKpCode]['previous_mastery'] ?? null; $currentChild = null; foreach ($masteryData as $item) { if ($item['kp_code'] === $childKpCode) { $currentChild = $item['mastery_level']; break; } } if ($previousChild !== null && $currentChild !== null) { $childChanges[] = floatval($currentChild) - floatval($previousChild); } } $avgChange = !empty($childChanges) ? array_sum($childChanges) / count($childChanges) : null; // 获取父节点中文名称 $parentKpInfo = DB::connection('mysql') ->table('knowledge_points') ->where('kp_code', $parentKpCode) ->first(); $parentMasteryLevels[$parentKpCode] = [ 'kp_code' => $parentKpCode, 'kp_name' => $parentKpInfo->name ?? $parentKpCode, 'mastery_level' => $parentMastery, 'mastery_percentage' => round($parentMastery * 100, 1), 'mastery_change' => $avgChange, 'children' => $relevantChildren, ]; } } Log::info('ExamPdfExportService: 过滤后的父节点掌握度', [ 'all_parent_count' => count($allParentMasteryLevels), 'filtered_parent_count' => count($parentMasteryLevels), 'filtered_codes' => array_keys($parentMasteryLevels) ]); } catch (\Exception $e) { Log::warning('ExamPdfExportService: 获取父节点掌握度失败', [ 'error' => $e->getMessage() ]); } Log::info('ExamPdfExportService: 使用analysisData中的掌握度数据', [ 'count' => count($masteryData), 'masteryData_sample' => !empty($masteryData) ? array_slice($masteryData, 0, 2) : [] ]); } else { // 如果没有knowledge_point_analysis,使用MasteryCalculator获取多层级掌握度概览 try { Log::info('ExamPdfExportService: 获取学生多层级掌握度概览', [ 'student_id' => $studentId ]); $masteryOverview = $this->masteryCalculator->getStudentMasteryOverviewWithHierarchy($studentId); $masteryData = $masteryOverview['details'] ?? []; $parentMasteryLevels = $masteryOverview['parent_mastery_levels'] ?? []; // 获取父节点掌握度 // 【修复】将对象数组转换为关联数组(避免 stdClass 对象访问错误) if (!empty($masteryData) && is_array($masteryData)) { $masteryData = array_map(function($item) { if (is_object($item)) { return [ 'kp_code' => $item->kp_code ?? null, 'kp_name' => $item->kp_name ?? null, 'mastery_level' => floatval($item->mastery_level ?? 0), 'mastery_change' => $item->mastery_change !== null ? floatval($item->mastery_change) : null, ]; } return $item; }, $masteryData); } // 【修复】获取快照数据以计算掌握度变化 $lastSnapshot = DB::connection('mysql') ->table('knowledge_point_mastery_snapshots') ->where('student_id', $studentId) ->latest('snapshot_time') ->first(); if ($lastSnapshot) { $previousMasteryJson = json_decode($lastSnapshot->mastery_data, true); foreach ($masteryData as &$item) { $kpCode = $item['kp_code']; if (isset($previousMasteryJson[$kpCode])) { $previous = floatval($previousMasteryJson[$kpCode]['previous_mastery'] ?? 0); $current = floatval($item['mastery_level']); $item['mastery_change'] = $current - $previous; } } unset($item); } Log::info('ExamPdfExportService: 成功获取多层级掌握度数据', [ 'count' => count($masteryData), 'parent_count' => count($parentMasteryLevels) ]); } catch (\Exception $e) { Log::error('ExamPdfExportService: 获取掌握度数据失败', [ 'student_id' => $studentId, 'error' => $e->getMessage() ]); } } // 【修改】使用本地方法获取学习路径推荐(替代API调用) $recommendations = []; try { Log::info('ExamPdfExportService: 获取学习路径推荐', [ 'student_id' => $studentId ]); $learningPaths = $this->learningAnalyticsService->recommendLearningPaths($studentId, 3); $recommendations = $learningPaths['recommendations'] ?? []; Log::info('ExamPdfExportService: 成功获取学习路径推荐', [ 'count' => count($recommendations) ]); } catch (\Exception $e) { Log::error('ExamPdfExportService: 获取学习路径推荐失败', [ 'student_id' => $studentId, 'error' => $e->getMessage() ]); } // 获取知识点名称映射 $kpNameMap = $this->buildKnowledgePointNameMap(); Log::info('ExamPdfExportService: 获取知识点名称映射', [ 'kpNameMap_count' => count($kpNameMap), 'kpNameMap_keys_sample' => !empty($kpNameMap) ? array_slice(array_keys($kpNameMap), 0, 5) : [] ]); // 【修复】直接从MySQL数据库获取题目详情(不通过API) $questionDetails = $this->getQuestionDetailsFromMySQL($paper); // 处理题目数据 $questions = $this->processQuestionsForReport($paper, $questionDetails, $kpNameMap); // 【关键调试】查看buildMasterySummary的返回结果 $masterySummary = $this->buildMasterySummary($masteryData, $kpNameMap); Log::info('ExamPdfExportService: buildMasterySummary返回结果', [ 'masteryData_count' => count($masteryData), 'kpNameMap_count' => count($kpNameMap), 'masterySummary_keys' => array_keys($masterySummary), 'masterySummary_items_count' => count($masterySummary['items'] ?? []), 'masterySummary_items_sample' => !empty($masterySummary['items']) ? array_slice($masterySummary['items'], 0, 2) : [] ]); // 【修复】处理父节点掌握度数据:过滤零值、转换名称、构建层级关系 $examKpCodes = array_column($masteryData, 'kp_code'); // 本次考试涉及的知识点 $processedParentMastery = $this->processParentMasteryLevels($parentMasteryLevels, $kpNameMap, $examKpCodes); Log::info('ExamPdfExportService: 处理后的父节点掌握度', [ 'raw_count' => count($parentMasteryLevels), 'processed_count' => count($processedParentMastery), 'processed_sample' => !empty($processedParentMastery) ? array_slice($processedParentMastery, 0, 3) : [] ]); return [ 'paper' => [ 'id' => $paper->paper_id, 'name' => $paper->paper_name, 'total_questions' => $paper->question_count, 'total_score' => $paper->total_score, 'created_at' => $paper->created_at, ], 'student' => $studentInfo, 'questions' => $questions, 'mastery' => $masterySummary, 'parent_mastery_levels' => $processedParentMastery, // 【修复】使用处理后的父节点数据 'insights' => $analysisData['question_analysis'] ?? [], // 使用question_analysis替代question_results 'recommendations' => $recommendations, 'analysis_data' => $analysisData, ]; } /** * 【修复】直接从PaperQuestion表获取题目详情(不通过API) */ private function getQuestionDetailsFromMySQL(Paper $paper): array { $details = []; Log::info('ExamPdfExportService: 从PaperQuestion表查询题目详情', [ 'paper_id' => $paper->paper_id, 'question_count' => $paper->questions->count() ]); foreach ($paper->questions as $pq) { try { // 【关键修复】直接从PaperQuestion对象获取solution和correct_answer $detail = [ 'id' => $pq->question_id, 'content' => $pq->question_text, 'question_type' => $pq->question_type, 'answer' => $pq->correct_answer ?? null, // 【修复】从PaperQuestion获取正确答案 'solution' => $pq->solution ?? null, // 【修复】从PaperQuestion获取解题思路 ]; $details[(string) ($pq->question_id ?? $pq->id)] = $detail; Log::debug('ExamPdfExportService: 成功获取题目详情', [ 'paper_question_id' => $pq->id, 'question_id' => $pq->question_id, 'has_answer' => !empty($pq->correct_answer), 'has_solution' => !empty($pq->solution), 'answer_preview' => $pq->correct_answer ? substr($pq->correct_answer, 0, 50) : null ]); } catch (\Throwable $e) { Log::error('ExamPdfExportService: 获取题目详情失败', [ 'paper_question_id' => $pq->id, 'error' => $e->getMessage(), ]); } } return $details; } /** * 处理题目数据(用于报告) */ private function processQuestionsForReport($paper, array $questionDetails, array $kpNameMap): array { $grouped = [ 'choice' => [], 'fill' => [], 'answer' => [], ]; // 【修复】处理空的试卷(questions可能不存在) $questions = $paper->questions ?? collect(); if ($questions->isEmpty()) { Log::info('ExamPdfExportService: 试卷没有题目,返回空数组'); return $grouped; } $sortedQuestions = $questions ->sortBy(function ($q, int $idx) { $number = $q->question_number ?? $idx + 1; return is_numeric($number) ? (float) $number : ($q->id ?? $idx); }); foreach ($sortedQuestions as $idx => $question) { $kpCode = $question->knowledge_point ?? ''; $kpName = $kpNameMap[$kpCode] ?? $kpCode ?: '未标注'; // 【修复】直接从PaperQuestion对象获取solution和correct_answer $answer = $question->correct_answer ?? null; // 直接从PaperQuestion获取 $solution = $question->solution ?? null; // 直接从PaperQuestion获取 $detail = $questionDetails[(string) ($question->question_id ?? $question->id)] ?? []; $typeRaw = $question->question_type ?? ($detail['question_type'] ?? $detail['type'] ?? ''); $normalizedType = $this->normalizeQuestionType($typeRaw); $number = $question->question_number ?? ($idx + 1); $payload = [ 'question_number' => $number, 'question_text' => is_array($question->question_text) ? json_encode($question->question_text, JSON_UNESCAPED_UNICODE) : ($question->question_text ?? ''), 'question_type' => $normalizedType, 'knowledge_point' => $kpCode, 'knowledge_point_name' => $kpName, 'score' => $question->score, 'answer' => $answer, // 正确答案 'solution' => $solution, // 解题思路 'student_answer' => $question->student_answer ?? null, // 【新增】学生答案 'correct_answer' => $answer, // 【新增】正确答案 'is_correct' => $question->is_correct ?? null, // 【新增】判分结果 'score_obtained' => $question->score_obtained ?? null, // 【新增】得分 ]; $grouped[$normalizedType][] = $payload; // 【调试】记录题目数据 Log::debug('ExamPdfExportService: 处理题目数据', [ 'paper_question_id' => $question->id, 'question_id' => $question->question_id, 'has_answer' => !empty($answer), 'has_solution' => !empty($solution), 'answer_preview' => $answer ? substr($answer, 0, 50) : null ]); } $ordered = array_merge($grouped['choice'], $grouped['fill'], $grouped['answer']); // 按卷面顺序重新编号 foreach ($ordered as $i => &$q) { $q['display_number'] = $i + 1; } unset($q); return $ordered; } /** * 构建PDF */ private function buildPdf(string $html): ?string { $tmpHtml = tempnam(sys_get_temp_dir(), 'exam_html_') . '.html'; $utf8Html = $this->ensureUtf8Html($html); $written = file_put_contents($tmpHtml, $utf8Html); Log::warning('ExamPdfExportService: [调试] HTML文件已创建', [ 'path' => $tmpHtml, 'html_length' => strlen($utf8Html), 'written_bytes' => $written, 'file_exists' => file_exists($tmpHtml), 'file_size' => file_exists($tmpHtml) ? filesize($tmpHtml) : 0, ]); // 仅使用Chrome渲染 $chromePdf = $this->renderWithChrome($tmpHtml); @unlink($tmpHtml); return $chromePdf; } /** * 使用Chrome渲染PDF */ private function renderWithChrome(string $htmlPath): ?string { $tmpPdf = tempnam(sys_get_temp_dir(), 'exam_pdf_') . '.pdf'; $userDataDir = sys_get_temp_dir() . '/chrome-profile-' . uniqid(); $chromeBinary = $this->findChromeBinary(); if (!$chromeBinary) { Log::error('ExamPdfExportService: 未找到可用的Chrome/Chromium'); return null; } // 设置运行时目录 $runtimeHome = sys_get_temp_dir() . '/chrome-home'; $runtimeXdg = sys_get_temp_dir() . '/chrome-xdg'; if (!File::exists($runtimeHome)) { @File::makeDirectory($runtimeHome, 0755, true); } if (!File::exists($runtimeXdg)) { @File::makeDirectory($runtimeXdg, 0755, true); } $process = new Process([ $chromeBinary, '--headless=new', // 【优化】使用新渲染引擎 '--disable-gpu', '--no-sandbox', '--disable-setuid-sandbox', '--disable-dev-shm-usage', '--disable-web-security', '--disable-features=VizDisplayCompositor', '--disable-extensions', // '--disable-background-networking', // 注释掉,可能阻止必要的网络请求 '--disable-component-update', '--disable-client-side-phishing-detection', '--disable-default-apps', '--disable-domain-reliability', '--disable-sync', '--no-first-run', '--no-default-browser-check', '--disable-crash-reporter', '--disable-print-preview', '--disable-features=TranslateUI', '--disable-features=OptimizationHints', '--disable-ipc-flooding-protection', '--disable-background-timer-throttling', '--disable-backgrounding-occluded-windows', '--disable-renderer-backgrounding', '--disable-features=AudioServiceOutOfProcess', '--disable-gpu-sandbox', '--disable-software-rasterizer', '--disable-background-mode', '--disable-extensions-http-throttling', '--disable-ipc-flooding-protection', '--disable-features=Dbus', // 禁用 dbus // 【关键修复】添加虚拟时间预算,让Chrome有足够时间加载CDN资源和执行JS '--virtual-time-budget=30000', // 30秒虚拟时间用于加载外部资源 '--run-all-compositor-stages-before-draw', // 确保所有渲染完成后再生成PDF '--user-data-dir=' . $userDataDir, '--print-to-pdf=' . $tmpPdf, '--print-to-pdf-no-header', '--allow-file-access-from-files', '--font-render-hinting=none', // 【优化】禁用字体渲染提示 '--disable-font-antialiasing', 'file://' . $htmlPath, ], null, [ 'HOME' => $runtimeHome, 'XDG_RUNTIME_DIR' => $runtimeXdg, ]); $process->setTimeout(90); // 【修复】增加超时时间到90秒 $killSignal = \defined('SIGKILL') ? \SIGKILL : 9; Log::warning('ExamPdfExportService: [调试] Chrome命令准备执行', [ 'chrome_binary' => $chromeBinary, 'html_path' => $htmlPath, 'html_exists' => file_exists($htmlPath), 'html_size' => file_exists($htmlPath) ? filesize($htmlPath) : 0, 'pdf_path' => $tmpPdf, 'user_data_dir' => $userDataDir, ]); try { $startedAt = microtime(true); $process->start(); $pdfGenerated = false; // 轮询检测PDF是否生成 $pollStart = microtime(true); $maxPollSeconds = 80; // 【修复】增加轮询超时到80秒 while ($process->isRunning() && (microtime(true) - $pollStart) < $maxPollSeconds) { if (file_exists($tmpPdf) && filesize($tmpPdf) > 0) { $pdfGenerated = true; Log::info('ExamPdfExportService: PDF生成成功,提前终止Chrome进程', [ 'elapsed' => round(microtime(true) - $startedAt, 2), 'pdf_size' => filesize($tmpPdf), ]); $process->stop(3, $killSignal); // 【优化】减少停止等待时间 break; } usleep(100_000); // 【优化】从200ms减少到100ms } if ($process->isRunning()) { $process->stop(3, $killSignal); // 【优化】减少停止等待时间 } // 【优化】删除不必要的wait()调用,避免重复等待 // $process->wait(); } catch (ProcessTimedOutException|ProcessSignaledException $e) { if ($process->isRunning()) { $process->stop(3, $killSignal); // 【优化】减少停止等待时间 } Log::warning('ExamPdfExportService: Chrome进程超时或被信号中断', [ 'elapsed' => round((microtime(true) - $startedAt), 2), 'exception' => get_class($e), ]); return $this->handleChromeProcessResult($tmpPdf, $userDataDir, $process, $startedAt); } catch (\Throwable $e) { if ($process->isRunning()) { $process->stop(3, $killSignal); // 【优化】减少停止等待时间 } Log::error('ExamPdfExportService: Chrome进程异常', [ 'elapsed' => round((microtime(true) - $startedAt), 2), 'error' => $e->getMessage(), ]); return $this->handleChromeProcessResult($tmpPdf, $userDataDir, $process, null); } return $this->handleChromeProcessResult($tmpPdf, $userDataDir, $process, null); } /** * 处理Chrome进程结果 */ private function handleChromeProcessResult(string $tmpPdf, string $userDataDir, Process $process, ?float $startedAt): ?string { $pdfExists = file_exists($tmpPdf); $pdfSize = $pdfExists ? filesize($tmpPdf) : null; $elapsed = $startedAt ? round((microtime(true) - $startedAt), 2) : null; // 【优化】即使进程未成功,只要PDF存在且大小合理就返回 if ($pdfExists && $pdfSize > 1000) { // 至少1KB Log::info('ExamPdfExportService: PDF生成成功', [ 'elapsed' => $elapsed, 'pdf_size' => $pdfSize, 'exit_code' => $process->getExitCode(), 'is_successful' => $process->isSuccessful(), ]); $pdfBinary = file_get_contents($tmpPdf); @unlink($tmpPdf); File::deleteDirectory($userDataDir); return $pdfBinary; } // 如果PDF不存在或太小,记录错误 Log::error('ExamPdfExportService: Chrome渲染失败', [ 'elapsed' => $elapsed, 'pdf_exists' => $pdfExists, 'pdf_size' => $pdfSize, 'exit_code' => $process->getExitCode(), 'error' => $process->getErrorOutput(), 'output' => $process->getOutput(), ]); @unlink($tmpPdf); File::deleteDirectory($userDataDir); return null; } /** * 查找Chrome二进制文件 */ private function findChromeBinary(): ?string { $candidates = [ env('PDF_CHROME_BINARY'), env('CHROME_BIN'), // Docker Alpine 环境变量 '/usr/bin/chromium-browser', // Alpine Linux '/usr/bin/chromium', '/usr/bin/google-chrome-stable', '/usr/bin/google-chrome', '/Applications/Google Chrome.app/Contents/MacOS/Google Chrome', // macOS ]; foreach ($candidates as $path) { if ($path && is_file($path) && is_executable($path)) { return $path; } } return null; } /** * 确保HTML为UTF-8编码,并内联外部资源 */ private function ensureUtf8Html(string $html): string { $meta = ''; if (stripos($html, '
') !== false) { $html = preg_replace('//i', "{$meta}", $html, 1); } else { $html = $meta . $html; } // 【关键修复】内联KaTeX CSS,避免Chrome在容器中加载CDN资源超时 $html = $this->inlineExternalResources($html); return $html; } /** * 将CDN资源替换为内联资源 * 【关键修复】避免Chrome在容器中加载CDN资源超时 */ private function inlineExternalResources(string $html): string { // 【调试】记录HTML内容信息 Log::warning('ExamPdfExportService: inlineExternalResources', [ 'html_length' => strlen($html), 'has_katex_cdn' => strpos($html, 'cdn.jsdelivr.net/npm/katex') !== false, 'html_head_preview' => substr($html, 0, 1000), ]); // 检查是否包含KaTeX CDN链接 if (strpos($html, 'cdn.jsdelivr.net/npm/katex') === false) { Log::warning('ExamPdfExportService: HTML中没有KaTeX CDN链接,跳过内联'); return $html; } try { // 读取本地KaTeX CSS文件并内联 $katexCssPath = public_path('css/katex/katex.min.css'); if (file_exists($katexCssPath)) { $katexCss = file_get_contents($katexCssPath); // 修复字体路径:将相对路径改为绝对路径(使用data URI或绝对路径) // KaTeX CSS中的字体引用格式: url(fonts/KaTeX_xxx.woff2) $fontsDir = public_path('css/katex/fonts'); $katexCss = preg_replace_callback( '/url\(["\']?fonts\/([^"\')\s]+)["\']?\)/i', function ($matches) use ($fontsDir) { $fontFile = $fontsDir . '/' . $matches[1]; if (file_exists($fontFile)) { // 将字体转换为data URI(适用于PDF生成) $fontData = base64_encode(file_get_contents($fontFile)); $mimeType = str_ends_with($matches[1], '.woff2') ? 'font/woff2' : 'font/woff'; return 'url(data:' . $mimeType . ';base64,' . $fontData . ')'; } return $matches[0]; }, $katexCss ); // 替换CDN CSS链接为内联样式 $html = preg_replace( '/]*href=["\']https:\/\/cdn\.jsdelivr\.net\/npm\/katex[^"\']*katex\.min\.css["\'][^>]*>/i', '', $html ); Log::info('ExamPdfExportService: KaTeX CSS已内联(含字体data URI)'); } // 读取本地KaTeX JS并内联 $katexJsPath = public_path('js/katex.min.js'); $autoRenderJsPath = public_path('js/auto-render.min.js'); if (file_exists($katexJsPath)) { $katexJs = file_get_contents($katexJsPath); $html = preg_replace( '/', $html ); } if (file_exists($autoRenderJsPath)) { $autoRenderJs = file_get_contents($autoRenderJsPath); $html = preg_replace( '/', $html ); } // 【关键修复】使用服务端预渲染,而不是依赖客户端 JavaScript // Chrome headless 的 --print-to-pdf 不会等待 JS 执行完成 // 所以我们使用 Node.js KaTeX 在服务端预渲染所有公式 // 1. 移除所有 KaTeX JavaScript(不再需要,因为使用服务端渲染) // 移除内联的 katex.min.js $html = preg_replace( '/