SystemHealthPanel.tsx 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. import type { ElementType } from 'react';
  2. import { useTranslation } from 'react-i18next';
  3. import { XCircle, AlertTriangle, CheckCircle2, ExternalLink, Wrench, ServerCog, Bug } from 'lucide-react';
  4. import type { LogFinding, LogFindingCategory, SystemHealthResult } from '../api/client';
  5. const WIKI_TROUBLESHOOTING = 'https://wiki.bambuddy.cool/reference/troubleshooting/';
  6. const CATEGORY_META: Record<LogFindingCategory, { icon: ElementType; badgeClass: string }> = {
  7. layer8: { icon: Wrench, badgeClass: 'bg-bambu-green/15 text-bambu-green border-bambu-green/30' },
  8. environment: { icon: ServerCog, badgeClass: 'bg-amber-500/15 text-amber-300 border-amber-500/30' },
  9. bug: { icon: Bug, badgeClass: 'bg-red-500/15 text-red-300 border-red-500/30' },
  10. };
  11. /**
  12. * One detected log-health finding. Cause/fix/name text is rendered from i18n
  13. * keyed by signature_id; an unknown signature (frontend older than backend)
  14. * still renders gracefully via the defaultValue fallbacks.
  15. */
  16. function FindingCard({ finding }: { finding: LogFinding }) {
  17. const { t } = useTranslation();
  18. const id = finding.signature_id;
  19. const name = t(`systemHealth.signature.${id}.name`, { defaultValue: id });
  20. const cause = t(`systemHealth.signature.${id}.cause`, { defaultValue: '' });
  21. const fix = t(`systemHealth.signature.${id}.fix`, { defaultValue: '' });
  22. const meta = CATEGORY_META[finding.category] ?? CATEGORY_META.bug;
  23. const CategoryIcon = meta.icon;
  24. const SeverityIcon = finding.severity === 'error' ? XCircle : AlertTriangle;
  25. const severityColor = finding.severity === 'error' ? 'text-red-400' : 'text-amber-400';
  26. return (
  27. <div className="bg-bambu-dark rounded-lg border border-bambu-dark-tertiary p-4 space-y-2">
  28. <div className="flex items-start gap-3">
  29. <SeverityIcon className={`w-5 h-5 flex-shrink-0 mt-0.5 ${severityColor}`} />
  30. <div className="flex-1 min-w-0">
  31. <div className="flex items-center gap-2 flex-wrap">
  32. <span className="text-sm font-medium text-white">{name}</span>
  33. <span
  34. className={`inline-flex items-center gap-1 text-xs px-2 py-0.5 rounded-full border ${meta.badgeClass}`}
  35. >
  36. <CategoryIcon className="w-3 h-3" />
  37. {t(`systemHealth.category.${finding.category}`)}
  38. </span>
  39. </div>
  40. {cause && <p className="text-xs text-bambu-gray mt-1">{cause}</p>}
  41. </div>
  42. </div>
  43. {fix && (
  44. <div className="text-xs text-white/90 bg-bambu-dark-secondary rounded px-3 py-2">
  45. <span className="text-bambu-green font-medium">{t('systemHealth.fixLabel')}</span> {fix}
  46. </div>
  47. )}
  48. <div className="text-xs font-mono text-bambu-gray/70 bg-bambu-dark-secondary rounded px-3 py-2 break-all">
  49. {finding.sample}
  50. </div>
  51. <div className="flex items-center justify-between gap-2 flex-wrap">
  52. <span className="text-xs text-bambu-gray">
  53. {t('systemHealth.occurrences', { times: finding.count, lastSeen: finding.last_seen })}
  54. </span>
  55. <a
  56. href={`${WIKI_TROUBLESHOOTING}#${finding.wiki_anchor}`}
  57. target="_blank"
  58. rel="noopener noreferrer"
  59. className="inline-flex items-center gap-1 text-xs text-bambu-green hover:underline"
  60. >
  61. {t('systemHealth.learnMore')}
  62. <ExternalLink className="w-3 h-3" />
  63. </a>
  64. </div>
  65. </div>
  66. );
  67. }
  68. /**
  69. * Presentational panel for a log-health scan result. Shared by the System page
  70. * section and the bug-report bubble so both surfaces look identical.
  71. */
  72. export function SystemHealthPanel({ result }: { result: SystemHealthResult }) {
  73. const { t } = useTranslation();
  74. if (!result.log_available) {
  75. return (
  76. <div className="rounded-lg bg-amber-500/10 border border-amber-500/30 px-4 py-3 text-sm text-amber-300">
  77. {t('systemHealth.logUnavailable')}
  78. </div>
  79. );
  80. }
  81. if (result.findings.length === 0) {
  82. return (
  83. <div className="rounded-lg bg-bambu-green/10 border border-bambu-green/30 px-4 py-3 text-sm text-bambu-green flex items-center gap-2">
  84. <CheckCircle2 className="w-5 h-5 flex-shrink-0" />
  85. <span>{t('systemHealth.clean', { times: result.scanned_entries })}</span>
  86. </div>
  87. );
  88. }
  89. return (
  90. <div className="space-y-3">
  91. {result.findings.map((finding) => (
  92. <FindingCard key={finding.signature_id} finding={finding} />
  93. ))}
  94. </div>
  95. );
  96. }