tessera-learn 0.0.9 → 0.0.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (45) hide show
  1. package/dist/plugin/cli.js +5 -3
  2. package/dist/plugin/cli.js.map +1 -1
  3. package/dist/plugin/index.d.ts.map +1 -1
  4. package/dist/plugin/index.js +215 -124
  5. package/dist/plugin/index.js.map +1 -1
  6. package/dist/{validation-BxWAMMnJ.js → validation-D9DXlqNP.js} +77 -65
  7. package/dist/validation-D9DXlqNP.js.map +1 -0
  8. package/package.json +9 -6
  9. package/src/components/Audio.svelte +5 -2
  10. package/src/components/DefaultLayout.svelte +2 -0
  11. package/src/components/FillInTheBlank.svelte +60 -98
  12. package/src/components/Image.svelte +3 -8
  13. package/src/components/LockedBanner.svelte +3 -4
  14. package/src/components/MultipleChoice.svelte +53 -94
  15. package/src/components/Quiz.svelte +2 -1
  16. package/src/components/Video.svelte +4 -2
  17. package/src/components/util.ts +1 -0
  18. package/src/plugin/cli.ts +2 -7
  19. package/src/plugin/export.ts +23 -41
  20. package/src/plugin/index.ts +197 -56
  21. package/src/plugin/layout.ts +6 -51
  22. package/src/plugin/manifest.ts +31 -5
  23. package/src/plugin/override-plugin.ts +68 -0
  24. package/src/plugin/quiz.ts +9 -54
  25. package/src/plugin/validation.ts +38 -67
  26. package/src/runtime/App.svelte +48 -36
  27. package/src/runtime/LoadingBar.svelte +47 -0
  28. package/src/runtime/Sidebar.svelte +2 -0
  29. package/src/runtime/adapters/cmi5.ts +13 -83
  30. package/src/runtime/adapters/format.ts +67 -0
  31. package/src/runtime/adapters/index.ts +28 -29
  32. package/src/runtime/adapters/retry.ts +0 -64
  33. package/src/runtime/adapters/scorm12.ts +1 -1
  34. package/src/runtime/adapters/scorm2004.ts +11 -16
  35. package/src/runtime/hooks.svelte.ts +14 -16
  36. package/src/runtime/navigation.svelte.ts +51 -45
  37. package/src/runtime/progress.svelte.ts +25 -10
  38. package/src/runtime/quiz-policy.ts +21 -178
  39. package/src/runtime/xapi/agent-rules.ts +7 -2
  40. package/src/runtime/xapi/publisher.ts +1 -11
  41. package/src/runtime/xapi/validation.ts +1 -1
  42. package/src/virtual.d.ts +13 -0
  43. package/styles/layout.css +34 -24
  44. package/dist/validation-BxWAMMnJ.js.map +0 -1
  45. package/src/runtime/LoadingSkeleton.svelte +0 -26
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":["__dirname"],"sources":["../../src/runtime/slugify.ts","../../src/plugin/export.ts","../../src/plugin/layout.ts","../../src/plugin/quiz.ts","../../src/plugin/index.ts"],"sourcesContent":["/**\n * Slugify a string for use as a URL-safe / filename-safe identifier.\n * \"My Course Title\" → \"my-course-title\"\n *\n * Shared by the runtime (`WebAdapter` localStorage key) and the build-time\n * exporter (`runExport` zip filename). Both want identical, deterministic\n * output so a course's storage key matches its package name.\n */\nexport function slugify(text: string): string {\n return text\n .toLowerCase()\n .trim()\n .replace(/[^\\w\\s-]/g, '')\n .replace(/[\\s_]+/g, '-')\n .replace(/-+/g, '-')\n .replace(/^-|-$/g, '');\n}\n","import { existsSync, readdirSync, statSync, writeFileSync, unlinkSync } from 'node:fs';\nimport { resolve } from 'node:path';\nimport { createWriteStream } from 'node:fs';\nimport { createHash } from 'node:crypto';\nimport { ZipArchive } from 'archiver';\nimport { slugify } from '../runtime/slugify.js';\n\n// ---------- Types ----------\n\ninterface ExportConfig {\n title: string;\n description?: string;\n version?: string;\n scoring?: { passingScore?: number };\n completion?: { mode?: 'quiz' | 'percentage' };\n export?: { standard?: string };\n}\n\n// ---------- Helpers ----------\n\nfunction escapeXml(str: string): string {\n return str\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;')\n .replace(/'/g, '&apos;');\n}\n\n/**\n * Recursively collect all file paths relative to a directory.\n */\nfunction collectFiles(dir: string, base: string = ''): string[] {\n const files: string[] = [];\n if (!existsSync(dir)) return files;\n\n for (const entry of readdirSync(dir)) {\n const fullPath = resolve(dir, entry);\n const relPath = base ? `${base}/${entry}` : entry;\n if (statSync(fullPath).isDirectory()) {\n files.push(...collectFiles(fullPath, relPath));\n } else {\n files.push(relPath);\n }\n }\n return files;\n}\n\n/**\n * Derive a stable URN IRI from a seed string. cmi5 §13.1 / xs:anyURI\n * require course / AU ids to be IRIs — bare hex or UUID-shaped strings\n * (without correct version/variant bits) aren't conformant URNs and may\n * be rejected by strict LMS importers.\n *\n * Hash the seed so the id survives rebuilds, then format as\n * `urn:tessera:<kind>:<hex>`. The same seed always produces the same\n * IRI, so existing LRS records are not orphaned by re-export.\n */\nfunction stableUrn(kind: 'course' | 'au', seed: string): string {\n const h = createHash('sha256').update(seed).digest('hex');\n // 32 hex chars (128 bits of entropy) is plenty; trim to keep ids short.\n return `urn:tessera:${kind}:${h.slice(0, 32)}`;\n}\n\nfunction formatSize(bytes: number): string {\n if (bytes < 1024) return `${bytes} B`;\n if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;\n return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;\n}\n\n// ---------- Manifest Generators ----------\n\n/** Per-version XML differences in imsmanifest.xml between SCORM 1.2 and 2004. */\ninterface ScormManifestDialect {\n rootNs: string;\n adlcpNs: string;\n schemaversion: string;\n /** Attribute name on <resource>: SCORM 1.2 uses lowercase, 2004 uses camelCase. */\n scormTypeAttr: 'scormtype' | 'scormType';\n /** Whitespace-separated namespace+XSD pairs for xsi:schemaLocation. */\n schemaLocation: string;\n}\n\nconst SCORM_DIALECTS: Record<'1.2' | '2004', ScormManifestDialect> = {\n '1.2': {\n rootNs: 'http://www.imsproject.org/xsd/imscp_rootv1p1p2',\n adlcpNs: 'http://www.adlnet.org/xsd/adlcp_rootv1p2',\n schemaversion: '1.2',\n scormTypeAttr: 'scormtype',\n schemaLocation:\n 'http://www.imsproject.org/xsd/imscp_rootv1p1p2 imscp_rootv1p1p2.xsd ' +\n 'http://www.imsglobal.org/xsd/imsmd_rootv1p2p1 imsmd_rootv1p2p1.xsd ' +\n 'http://www.adlnet.org/xsd/adlcp_rootv1p2 adlcp_rootv1p2.xsd',\n },\n '2004': {\n rootNs: 'http://www.imsglobal.org/xsd/imscp_v1p1',\n adlcpNs: 'http://www.adlnet.org/xsd/adlcp_v1p3',\n schemaversion: '2004 4th Edition',\n scormTypeAttr: 'scormType',\n schemaLocation:\n 'http://www.imsglobal.org/xsd/imscp_v1p1 imscp_v1p1.xsd ' +\n 'http://www.adlnet.org/xsd/adlcp_v1p3 adlcp_v1p3.xsd',\n },\n};\n\nexport function generateScormManifest(\n version: '1.2' | '2004',\n config: ExportConfig,\n distDir: string\n): string {\n const dialect = SCORM_DIALECTS[version];\n const title = escapeXml(config.title || 'Tessera Course');\n const files = collectFiles(distDir);\n const fileElements = files\n .map((f) => ` <file href=\"${escapeXml(f)}\" />`)\n .join('\\n');\n\n return `<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<manifest identifier=\"tessera-course\" version=\"1.0\"\n xmlns=\"${dialect.rootNs}\"\n xmlns:adlcp=\"${dialect.adlcpNs}\"\n xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\n xsi:schemaLocation=\"${dialect.schemaLocation}\">\n <metadata>\n <schema>ADL SCORM</schema>\n <schemaversion>${dialect.schemaversion}</schemaversion>\n </metadata>\n <organizations default=\"org-1\">\n <organization identifier=\"org-1\">\n <title>${title}</title>\n <item identifier=\"item-1\" identifierref=\"res-1\">\n <title>${title}</title>\n </item>\n </organization>\n </organizations>\n <resources>\n <resource identifier=\"res-1\" type=\"webcontent\" adlcp:${dialect.scormTypeAttr}=\"sco\" href=\"index.html\">\n${fileElements}\n </resource>\n </resources>\n</manifest>`;\n}\n\nexport function generateSCORM12Manifest(\n config: ExportConfig,\n distDir: string\n): string {\n return generateScormManifest('1.2', config, distDir);\n}\n\nexport function generateSCORM2004Manifest(\n config: ExportConfig,\n distDir: string\n): string {\n return generateScormManifest('2004', config, distDir);\n}\n\nexport function generateCMI5Xml(config: ExportConfig): string {\n const title = escapeXml(config.title || 'Tessera Course');\n const description = escapeXml(config.description || '');\n // Derive stable IDs from the course title so they survive rebuilds without\n // orphaning existing learner records in the LRS.\n const courseId = stableUrn('course', `tessera-course:${config.title || ''}`);\n const auId = stableUrn('au', `tessera-au:${config.title || ''}`);\n // cmi5 §10.2.4 caps masteryScore at 4 decimals; avoid float drift like 0.7000000000000001.\n const masteryScore = Number(\n ((config.scoring?.passingScore ?? 70) / 100).toFixed(4)\n );\n // cmi5 §13.1.4 — `moveOn` decides which verb(s) the LMS treats as\n // satisfying the AU. For graded courses (completion gated on a quiz)\n // a learner who completes without passing should NOT receive credit, so\n // the LMS needs both a Completed AND a Passed before satisfaction.\n // Percentage-mode courses don't surface pass/fail, so completion alone\n // is the right signal.\n const moveOn =\n config.completion?.mode === 'quiz' ? 'CompletedAndPassed' : 'Completed';\n\n return `<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<courseStructure xmlns=\"https://w3id.org/xapi/profiles/cmi5/v1/CourseStructure.xsd\">\n <course id=\"${courseId}\">\n <title><langstring lang=\"en-US\">${title}</langstring></title>\n <description><langstring lang=\"en-US\">${description}</langstring></description>\n </course>\n <au id=\"${auId}\" launchMethod=\"AnyWindow\" moveOn=\"${moveOn}\" masteryScore=\"${masteryScore}\">\n <title><langstring lang=\"en-US\">${title}</langstring></title>\n <description><langstring lang=\"en-US\">${description}</langstring></description>\n <url>index.html</url>\n </au>\n</courseStructure>`;\n}\n\n// ---------- ZIP Packaging ----------\n\nexport async function createZip(\n distDir: string,\n outputPath: string\n): Promise<number> {\n return new Promise((res, reject) => {\n const output = createWriteStream(outputPath);\n const archive = new ZipArchive({ zlib: { level: 9 } });\n\n output.on('close', () => {\n res(archive.pointer());\n });\n output.on('error', reject);\n archive.on('error', reject);\n\n archive.pipe(output);\n archive.directory(distDir, false);\n archive.finalize();\n });\n}\n\n// ---------- Main Export ----------\n\n/**\n * Run the export process after Vite build completes.\n * Writes manifest XML into dist/, then packages into ZIP if needed.\n */\n/** Remove any previously built zips for this package to prevent accumulation. */\nfunction cleanOldZips(projectRoot: string, slug: string): void {\n try {\n for (const f of readdirSync(projectRoot)) {\n if (f.startsWith(`${slug}-`) && f.endsWith('.zip')) {\n try { unlinkSync(resolve(projectRoot, f)); } catch {}\n }\n }\n } catch {}\n}\n\nexport async function runExport(\n projectRoot: string,\n config: ExportConfig\n): Promise<void> {\n const distDir = resolve(projectRoot, 'dist');\n const standard = config.export?.standard || 'web';\n const slug = slugify(config.title || 'tessera-course') || 'tessera-course';\n const version = config.version || '1.0.0';\n const zipName = `${slug}-${version}.zip`;\n const zipPath = resolve(projectRoot, zipName);\n\n switch (standard) {\n case 'web': {\n // Compute dist size\n const files = collectFiles(distDir);\n let totalSize = 0;\n for (const f of files) {\n totalSize += statSync(resolve(distDir, f)).size;\n }\n console.log(`✓ Web export: dist/ (${formatSize(totalSize)})`);\n break;\n }\n\n case 'scorm12': {\n const manifest = generateSCORM12Manifest(config, distDir);\n writeFileSync(resolve(distDir, 'imsmanifest.xml'), manifest, 'utf-8');\n cleanOldZips(projectRoot, slug);\n const zipSize = await createZip(distDir, zipPath);\n console.log(\n `✓ SCORM 1.2 export: ${zipName} (${formatSize(zipSize)})`\n );\n break;\n }\n\n case 'scorm2004': {\n const manifest = generateSCORM2004Manifest(config, distDir);\n writeFileSync(resolve(distDir, 'imsmanifest.xml'), manifest, 'utf-8');\n cleanOldZips(projectRoot, slug);\n const zipSize = await createZip(distDir, zipPath);\n console.log(\n `✓ SCORM 2004 export: ${zipName} (${formatSize(zipSize)})`\n );\n break;\n }\n\n case 'cmi5': {\n const xml = generateCMI5Xml(config);\n writeFileSync(resolve(distDir, 'cmi5.xml'), xml, 'utf-8');\n cleanOldZips(projectRoot, slug);\n const zipSize = await createZip(distDir, zipPath);\n console.log(`✓ CMI5 export: ${zipName} (${formatSize(zipSize)})`);\n break;\n }\n }\n}\n","import type { Plugin, ResolvedConfig, ViteDevServer } from 'vite';\nimport { existsSync } from 'node:fs';\nimport { resolve } from 'node:path';\n\nconst VIRTUAL_LAYOUT_ID = 'virtual:tessera-layout';\nconst RESOLVED_LAYOUT_ID = '\\0' + VIRTUAL_LAYOUT_ID;\n\nexport function tesseraLayoutPlugin(): Plugin {\n let projectRoot: string;\n\n return {\n name: 'tessera:layout',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n },\n\n resolveId(id) {\n if (id === VIRTUAL_LAYOUT_ID) return RESOLVED_LAYOUT_ID;\n return null;\n },\n\n load(id) {\n if (id !== RESOLVED_LAYOUT_ID) return null;\n const layoutPath = resolve(projectRoot, 'layout.svelte');\n if (existsSync(layoutPath)) {\n // Register the file with Vite so edits trigger HMR / build --watch\n // re-runs. Only add when the file actually exists — calling\n // addWatchFile on a non-existent path makes Vite's importAnalysis\n // try to resolve it as a real import.\n this.addWatchFile(layoutPath);\n const normalized = layoutPath.replace(/\\\\/g, '/');\n return `export { default } from '${normalized}';`;\n }\n return `export default null;`;\n },\n\n configureServer(server: ViteDevServer) {\n const layoutPath = resolve(projectRoot, 'layout.svelte');\n // Only react to add/unlink: those flip the virtual module's load() output\n // between `export default null` and `export { default } from '...'`. A\n // `change` event leaves that output identical and is handled by Svelte's\n // own HMR for the underlying file — full-reloading on every edit would\n // wipe in-page state for no reason.\n server.watcher.on('all', (event, filePath) => {\n if (filePath !== layoutPath) return;\n if (event !== 'add' && event !== 'unlink') return;\n const mod = server.moduleGraph.getModuleById(RESOLVED_LAYOUT_ID);\n if (mod) server.moduleGraph.invalidateModule(mod);\n server.ws.send({ type: 'full-reload' });\n });\n },\n };\n}\n","import type { Plugin, ResolvedConfig, ViteDevServer } from 'vite';\nimport { existsSync } from 'node:fs';\nimport { resolve, dirname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\n\nconst VIRTUAL_QUIZ_ID = 'virtual:tessera-quiz';\nconst RESOLVED_QUIZ_ID = '\\0' + VIRTUAL_QUIZ_ID;\n\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = dirname(__filename);\n\n/**\n * Resolve the project's quiz shell.\n * `projectRoot/quiz.svelte` overrides the built-in `<Quiz>` if it exists,\n * otherwise the built-in is used. Mirrors `tesseraLayoutPlugin` (Phase 3A).\n */\nexport function tesseraQuizPlugin(): Plugin {\n let projectRoot: string;\n // Resolve the built-in Quiz.svelte once. The plugin lives in\n // `dist/plugin/quiz.js` after build and `src/plugin/quiz.ts` in source —\n // both layouts put `Quiz.svelte` two levels up under `src/components/`.\n const packageRoot = resolve(__dirname, '..', '..');\n const builtinQuiz = resolve(packageRoot, 'src', 'components', 'Quiz.svelte');\n\n return {\n name: 'tessera:quiz',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n },\n\n resolveId(id) {\n if (id === VIRTUAL_QUIZ_ID) return RESOLVED_QUIZ_ID;\n return null;\n },\n\n load(id) {\n if (id !== RESOLVED_QUIZ_ID) return null;\n const userQuizPath = resolve(projectRoot, 'quiz.svelte');\n if (existsSync(userQuizPath)) {\n // Watch the user file so add/remove flips through HMR (see below).\n this.addWatchFile(userQuizPath);\n const normalized = userQuizPath.replace(/\\\\/g, '/');\n return `export { default } from '${normalized}';`;\n }\n const normalized = builtinQuiz.replace(/\\\\/g, '/');\n return `export { default } from '${normalized}';`;\n },\n\n configureServer(server: ViteDevServer) {\n const userQuizPath = resolve(projectRoot, 'quiz.svelte');\n // Only react to add/unlink — those flip the load() output between the\n // user quiz and the built-in. A `change` event leaves the resolved\n // module identical and is handled by Svelte's own HMR.\n server.watcher.on('all', (event, filePath) => {\n if (filePath !== userQuizPath) return;\n if (event !== 'add' && event !== 'unlink') return;\n const mod = server.moduleGraph.getModuleById(RESOLVED_QUIZ_ID);\n if (mod) server.moduleGraph.invalidateModule(mod);\n server.ws.send({ type: 'full-reload' });\n });\n },\n };\n}\n","import type { Plugin, ResolvedConfig, ViteDevServer } from 'vite';\nimport { svelte } from '@sveltejs/vite-plugin-svelte';\nimport { fileURLToPath } from 'node:url';\nimport { dirname, resolve } from 'node:path';\nimport { existsSync, readFileSync, readdirSync, statSync, writeFileSync, unlinkSync, cpSync, mkdirSync } from 'node:fs';\nimport { generateManifest, extractDefaultExportObjectLiteral } from './manifest.js';\nimport JSON5 from 'json5';\nimport type { Manifest } from './manifest.js';\nimport { validateProject } from './validation.js';\nimport { runExport } from './export.js';\nimport { tesseraLayoutPlugin } from './layout.js';\nimport { tesseraQuizPlugin } from './quiz.js';\n\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = dirname(__filename);\n\n// Resolve the runtime directory where App.svelte lives\nfunction resolveRuntimeDir(): string {\n const packageRoot = resolve(__dirname, '..', '..');\n return resolve(packageRoot, 'src', 'runtime');\n}\n\n// Resolve the framework styles directory\nfunction resolveStylesDir(): string {\n const packageRoot = resolve(__dirname, '..', '..');\n return resolve(packageRoot, 'styles');\n}\n\nexport function tesseraPlugin() {\n return [\n svelte({\n compilerOptions: { css: 'injected' },\n }),\n tesseraValidationPlugin(),\n tesseraEntryPlugin(),\n tesseraConfigPlugin(),\n tesseraPagesPlugin(),\n tesseraManifestPlugin(),\n tesseraLayoutPlugin(),\n tesseraQuizPlugin(),\n tesseraExportPlugin(),\n ];\n}\n\n// ---------- Entry Plugin ----------\n\nconst VIRTUAL_ENTRY_ID = 'virtual:tessera-entry';\nconst RESOLVED_ENTRY_ID = '\\0' + VIRTUAL_ENTRY_ID;\nconst VIRTUAL_MAIN_ID = '/virtual:tessera-main';\nconst RESOLVED_MAIN_ID = '\\0virtual:tessera-main';\n\nfunction tesseraEntryPlugin(): Plugin {\n const runtimeDir = resolveRuntimeDir();\n const stylesDir = resolveStylesDir();\n const appSveltePath = resolve(runtimeDir, 'App.svelte');\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:entry',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n },\n\n // For build mode: write index.html so Rollup can find it\n buildStart() {\n if (isBuild) {\n writeFileSync(resolve(projectRoot, 'index.html'), generateIndexHtml(), 'utf-8');\n }\n },\n\n // For build mode: clean up temporary index.html and copy assets\n closeBundle() {\n if (isBuild) {\n const htmlPath = resolve(projectRoot, 'index.html');\n if (existsSync(htmlPath)) {\n try { unlinkSync(htmlPath); } catch {}\n }\n\n // Copy assets/ directory to dist/assets/ so $assets/ references resolve\n const assetsDir = resolve(projectRoot, 'assets');\n const distAssetsDir = resolve(projectRoot, 'dist', 'assets');\n if (existsSync(assetsDir)) {\n mkdirSync(distAssetsDir, { recursive: true });\n cpSync(assetsDir, distAssetsDir, { recursive: true });\n }\n }\n },\n\n // Serve index.html for the dev server\n configureServer(server: ViteDevServer) {\n return () => {\n server.middlewares.use(async (req, res, next) => {\n if (req.url === '/' || req.url === '/index.html') {\n const html = generateIndexHtml();\n const transformed = await server.transformIndexHtml(req.url, html);\n res.setHeader('Content-Type', 'text/html');\n res.statusCode = 200;\n res.end(transformed);\n return;\n }\n next();\n });\n };\n },\n\n resolveId(id) {\n if (id === VIRTUAL_ENTRY_ID) return RESOLVED_ENTRY_ID;\n if (id === VIRTUAL_MAIN_ID || id === 'virtual:tessera-main') return RESOLVED_MAIN_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_ENTRY_ID || id === RESOLVED_MAIN_ID) {\n return generateEntryScript(appSveltePath, stylesDir, projectRoot);\n }\n return null;\n },\n };\n}\n\nfunction generateIndexHtml(): string {\n return `<!DOCTYPE html>\n<html lang=\"en\">\n<head>\n <meta charset=\"UTF-8\" />\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\" />\n <title>Tessera Course</title>\n</head>\n<body>\n <div id=\"tessera-root\"></div>\n <script type=\"module\" src=\"/virtual:tessera-main\"></script>\n</body>\n</html>`;\n}\n\nfunction generateEntryScript(appSveltePath: string, frameworkStylesDir: string, projectRoot: string): string {\n const normalizedPath = appSveltePath.replace(/\\\\/g, '/');\n\n // Framework CSS imports (theme → base → layout)\n const frameworkCssOrder = ['theme.css', 'base.css', 'layout.css'];\n const frameworkImports = frameworkCssOrder\n .map(file => resolve(frameworkStylesDir, file).replace(/\\\\/g, '/'))\n .filter(path => existsSync(path))\n .map(path => `import '${path}';`)\n .join('\\n');\n\n // User CSS imports from project's styles/ directory\n const userStylesDir = resolve(projectRoot, 'styles');\n let userImports = '';\n if (existsSync(userStylesDir)) {\n const userCssFiles = readdirSync(userStylesDir)\n .filter(f => f.endsWith('.css'))\n .sort();\n userImports = userCssFiles\n .map(f => resolve(userStylesDir, f).replace(/\\\\/g, '/'))\n .map(path => `import '${path}';`)\n .join('\\n');\n }\n\n return `// Framework styles\n${frameworkImports}\n// User styles\n${userImports}\n\nimport { mount } from 'svelte';\nimport App from '${normalizedPath}';\n\nmount(App, {\n target: document.getElementById('tessera-root'),\n});\n`;\n}\n\n// ---------- Config Plugin ----------\n\nconst VIRTUAL_CONFIG_ID = 'virtual:tessera-config';\nconst RESOLVED_CONFIG_ID = '\\0' + VIRTUAL_CONFIG_ID;\n\nfunction completionDefaults(mode: string | undefined): {\n completion: Record<string, unknown>;\n passingScore: number;\n} {\n if (mode === 'manual') {\n return { completion: { mode: 'manual' }, passingScore: 0 };\n }\n return { completion: { mode: 'percentage', percentageThreshold: 100 }, passingScore: 70 };\n}\n\nfunction tesseraConfigPlugin(): Plugin {\n let projectRoot: string;\n\n return {\n name: 'tessera:config',\n enforce: 'pre',\n\n config(config) {\n const root = config.root || process.cwd();\n\n return {\n base: './',\n resolve: {\n alias: {\n '$assets': resolve(root, 'assets'),\n },\n },\n // tessera-learn ships .ts/.svelte.ts source; Vite's dep optimizer\n // doesn't run vite-plugin-svelte's preprocessor, so skip pre-bundling.\n optimizeDeps: {\n exclude: ['tessera-learn'],\n },\n };\n },\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n },\n\n resolveId(id) {\n if (id === VIRTUAL_CONFIG_ID) return RESOLVED_CONFIG_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_CONFIG_ID) {\n const configPath = resolve(projectRoot, 'course.config.js');\n let userConfig: Record<string, any> = {};\n\n if (existsSync(configPath)) {\n this.addWatchFile(configPath);\n const objectStr = extractDefaultExportObjectLiteral(readFileSync(configPath, 'utf-8'));\n if (objectStr) {\n try { userConfig = JSON5.parse(objectStr); } catch {}\n }\n }\n\n const { completion, passingScore } = completionDefaults(userConfig.completion?.mode);\n const merged = {\n title: userConfig.title || 'Untitled Course',\n ...userConfig,\n navigation: { mode: 'free', ...userConfig.navigation },\n completion: { ...completion, ...userConfig.completion },\n scoring: { passingScore, ...userConfig.scoring },\n export: { standard: 'web', ...userConfig.export },\n };\n\n return `export default ${JSON.stringify(merged)};`;\n }\n return null;\n },\n };\n}\n\n// ---------- Manifest Watch Helpers ----------\n\n/** Register all _meta.js and .svelte files under pagesDir as watch files for build mode. */\nfunction addWatchFiles(ctx: { addWatchFile(id: string): void }, dir: string): void {\n if (!existsSync(dir)) return;\n for (const entry of readdirSync(dir)) {\n const full = resolve(dir, entry);\n if (statSync(full).isDirectory()) {\n addWatchFiles(ctx, full);\n } else if (entry.endsWith('.svelte') || entry === '_meta.js') {\n ctx.addWatchFile(full);\n }\n }\n}\n\n// ---------- Pages Plugin ----------\n\nconst VIRTUAL_PAGES_ID = 'virtual:tessera-pages';\nconst RESOLVED_PAGES_ID = '\\0' + VIRTUAL_PAGES_ID;\n\n/**\n * Provides a virtual module that exports an import.meta.glob map for all .svelte\n * pages. This runs in the user's project context so the glob resolves against their\n * pages/ directory, and Vite can statically analyze it for code splitting.\n */\nfunction tesseraPagesPlugin(): Plugin {\n return {\n name: 'tessera:pages',\n enforce: 'pre',\n\n resolveId(id) {\n if (id === VIRTUAL_PAGES_ID) return RESOLVED_PAGES_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_PAGES_ID) {\n return `export default import.meta.glob('/pages/**/*.svelte');`;\n }\n return null;\n },\n };\n}\n\n// ---------- Validation Plugin ----------\n\nfunction tesseraValidationPlugin(): Plugin {\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:validation',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n // Run validation during dev (configResolved fires before server starts)\n if (!isBuild) {\n runValidation(projectRoot);\n }\n },\n\n buildStart() {\n // Run validation during build (buildStart fires once before bundling)\n if (isBuild) {\n runValidation(projectRoot);\n }\n },\n };\n}\n\nfunction runValidation(projectRoot: string): void {\n const { errors, warnings } = validateProject(projectRoot);\n\n for (const warning of warnings) {\n console.warn(`\\x1b[33m[tessera warning]\\x1b[0m ${warning}`);\n }\n\n if (errors.length > 0) {\n for (const error of errors) {\n console.error(`\\x1b[31m[tessera error]\\x1b[0m ${error}`);\n }\n throw new Error(\n `Tessera validation failed with ${errors.length} error(s). Fix the errors above to continue.`\n );\n }\n}\n\n// ---------- Export Plugin ----------\n\nfunction tesseraExportPlugin(): Plugin {\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:export',\n enforce: 'post',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n },\n\n async closeBundle() {\n if (!isBuild) return;\n\n const configPath = resolve(projectRoot, 'course.config.js');\n if (!existsSync(configPath)) {\n // Validation already required course.config.js — getting here means\n // the file vanished mid-build. Surface that loudly rather than\n // shipping a bundle with no LMS export silently.\n throw new Error(\n '[tessera:export] course.config.js not found at closeBundle. The file must exist for the export step to run.'\n );\n }\n\n const objectStr = extractDefaultExportObjectLiteral(readFileSync(configPath, 'utf-8'));\n if (!objectStr) {\n throw new Error(\n '[tessera:export] course.config.js: could not locate `export default { ... }`. Cannot determine export.standard.'\n );\n }\n\n let config: any;\n try {\n config = JSON5.parse(objectStr);\n } catch (err) {\n throw new Error(\n `[tessera:export] course.config.js: failed to parse export-default object literal — ${(err as Error).message}`\n );\n }\n\n await runExport(projectRoot, config);\n },\n };\n}\n\n// ---------- Manifest Plugin ----------\n\nconst VIRTUAL_MANIFEST_ID = 'virtual:tessera-manifest';\nconst RESOLVED_MANIFEST_ID = '\\0' + VIRTUAL_MANIFEST_ID;\n\nfunction tesseraManifestPlugin(): Plugin {\n let projectRoot: string;\n let pagesDir: string;\n let currentManifest: Manifest | null = null;\n let server: ViteDevServer | null = null;\n\n function buildManifest(): Manifest {\n currentManifest = generateManifest(pagesDir);\n return currentManifest;\n }\n\n return {\n name: 'tessera:manifest',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n pagesDir = resolve(projectRoot, 'pages');\n },\n\n configureServer(devServer: ViteDevServer) {\n server = devServer;\n\n // Watch the pages directory for changes\n devServer.watcher.on('all', (event, filePath) => {\n if (!filePath.startsWith(pagesDir)) return;\n\n // Rebuild manifest on relevant file changes\n const isRelevant =\n filePath.endsWith('.svelte') ||\n filePath.endsWith('_meta.js') ||\n event === 'addDir' ||\n event === 'unlinkDir';\n\n if (isRelevant) {\n currentManifest = null; // invalidate cache\n\n // Invalidate the virtual module to trigger HMR\n const mod = devServer.moduleGraph.getModuleById(RESOLVED_MANIFEST_ID);\n if (mod) {\n devServer.moduleGraph.invalidateModule(mod);\n devServer.ws.send({ type: 'full-reload' });\n }\n\n console.log(`[tessera] Manifest rebuilt (${event}: ${filePath.replace(projectRoot, '')})`);\n }\n });\n },\n\n buildStart() {\n buildManifest();\n },\n\n resolveId(id) {\n if (id === VIRTUAL_MANIFEST_ID) return RESOLVED_MANIFEST_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_MANIFEST_ID) {\n if (!currentManifest) {\n buildManifest();\n }\n\n // Register watch files so Vite's built-in watcher (used in build --watch)\n // knows to re-trigger when pages/ content changes.\n addWatchFiles(this, pagesDir);\n\n // Encode as base64 to prevent Vite's import analysis from\n // scanning .svelte importPath strings as module imports.\n // Replace Infinity with 1e9 since JSON.stringify drops it.\n const json = JSON.stringify(currentManifest, (_key, value) =>\n value === Infinity ? 1e9 : value\n );\n const b64 = Buffer.from(json).toString('base64');\n return `export default JSON.parse(atob(\"${b64}\"));`;\n }\n return null;\n },\n };\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":["__dirname"],"sources":["../../src/runtime/slugify.ts","../../src/plugin/export.ts","../../src/plugin/override-plugin.ts","../../src/plugin/layout.ts","../../src/plugin/quiz.ts","../../src/plugin/index.ts"],"sourcesContent":["/**\n * Slugify a string for use as a URL-safe / filename-safe identifier.\n * \"My Course Title\" → \"my-course-title\"\n *\n * Shared by the runtime (`WebAdapter` localStorage key) and the build-time\n * exporter (`runExport` zip filename). Both want identical, deterministic\n * output so a course's storage key matches its package name.\n */\nexport function slugify(text: string): string {\n return text\n .toLowerCase()\n .trim()\n .replace(/[^\\w\\s-]/g, '')\n .replace(/[\\s_]+/g, '-')\n .replace(/-+/g, '-')\n .replace(/^-|-$/g, '');\n}\n","import { existsSync, readdirSync, statSync, writeFileSync, unlinkSync } from 'node:fs';\nimport { resolve } from 'node:path';\nimport { createWriteStream } from 'node:fs';\nimport { createHash } from 'node:crypto';\nimport { ZipArchive } from 'archiver';\nimport { slugify } from '../runtime/slugify.js';\n\n// ---------- Types ----------\n\ninterface ExportConfig {\n title: string;\n description?: string;\n version?: string;\n scoring?: { passingScore?: number };\n completion?: { mode?: 'quiz' | 'percentage' };\n export?: { standard?: string };\n}\n\n// ---------- Helpers ----------\n\nfunction escapeXml(str: string): string {\n return str\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;')\n .replace(/'/g, '&apos;');\n}\n\n/**\n * Recursively collect all file paths relative to a directory.\n */\nfunction collectFiles(dir: string, base: string = ''): string[] {\n const files: string[] = [];\n if (!existsSync(dir)) return files;\n\n for (const entry of readdirSync(dir)) {\n const fullPath = resolve(dir, entry);\n const relPath = base ? `${base}/${entry}` : entry;\n if (statSync(fullPath).isDirectory()) {\n files.push(...collectFiles(fullPath, relPath));\n } else {\n files.push(relPath);\n }\n }\n return files;\n}\n\n/**\n * Derive a stable URN IRI from a seed string. cmi5 §13.1 / xs:anyURI\n * require course / AU ids to be IRIs — bare hex or UUID-shaped strings\n * (without correct version/variant bits) aren't conformant URNs and may\n * be rejected by strict LMS importers.\n *\n * Hash the seed so the id survives rebuilds, then format as\n * `urn:tessera:<kind>:<hex>`. The same seed always produces the same\n * IRI, so existing LRS records are not orphaned by re-export.\n */\nfunction stableUrn(kind: 'course' | 'au', seed: string): string {\n const h = createHash('sha256').update(seed).digest('hex');\n // 32 hex chars (128 bits of entropy) is plenty; trim to keep ids short.\n return `urn:tessera:${kind}:${h.slice(0, 32)}`;\n}\n\nfunction formatSize(bytes: number): string {\n if (bytes < 1024) return `${bytes} B`;\n if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;\n return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;\n}\n\n// ---------- Manifest Generators ----------\n\n/** Per-version XML differences in imsmanifest.xml between SCORM 1.2 and 2004. */\ninterface ScormManifestDialect {\n rootNs: string;\n adlcpNs: string;\n schemaversion: string;\n /** Attribute name on <resource>: SCORM 1.2 uses lowercase, 2004 uses camelCase. */\n scormTypeAttr: 'scormtype' | 'scormType';\n /** Whitespace-separated namespace+XSD pairs for xsi:schemaLocation. */\n schemaLocation: string;\n}\n\nconst SCORM_DIALECTS: Record<'1.2' | '2004', ScormManifestDialect> = {\n '1.2': {\n rootNs: 'http://www.imsproject.org/xsd/imscp_rootv1p1p2',\n adlcpNs: 'http://www.adlnet.org/xsd/adlcp_rootv1p2',\n schemaversion: '1.2',\n scormTypeAttr: 'scormtype',\n schemaLocation:\n 'http://www.imsproject.org/xsd/imscp_rootv1p1p2 imscp_rootv1p1p2.xsd ' +\n 'http://www.imsglobal.org/xsd/imsmd_rootv1p2p1 imsmd_rootv1p2p1.xsd ' +\n 'http://www.adlnet.org/xsd/adlcp_rootv1p2 adlcp_rootv1p2.xsd',\n },\n '2004': {\n rootNs: 'http://www.imsglobal.org/xsd/imscp_v1p1',\n adlcpNs: 'http://www.adlnet.org/xsd/adlcp_v1p3',\n schemaversion: '2004 4th Edition',\n scormTypeAttr: 'scormType',\n schemaLocation:\n 'http://www.imsglobal.org/xsd/imscp_v1p1 imscp_v1p1.xsd ' +\n 'http://www.adlnet.org/xsd/adlcp_v1p3 adlcp_v1p3.xsd',\n },\n};\n\nexport function generateScormManifest(\n version: '1.2' | '2004',\n config: ExportConfig,\n distDir: string\n): string {\n const dialect = SCORM_DIALECTS[version];\n const title = escapeXml(config.title || 'Tessera Course');\n const files = collectFiles(distDir);\n const fileElements = files\n .map((f) => ` <file href=\"${escapeXml(f)}\" />`)\n .join('\\n');\n\n return `<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<manifest identifier=\"tessera-course\" version=\"1.0\"\n xmlns=\"${dialect.rootNs}\"\n xmlns:adlcp=\"${dialect.adlcpNs}\"\n xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\n xsi:schemaLocation=\"${dialect.schemaLocation}\">\n <metadata>\n <schema>ADL SCORM</schema>\n <schemaversion>${dialect.schemaversion}</schemaversion>\n </metadata>\n <organizations default=\"org-1\">\n <organization identifier=\"org-1\">\n <title>${title}</title>\n <item identifier=\"item-1\" identifierref=\"res-1\">\n <title>${title}</title>\n </item>\n </organization>\n </organizations>\n <resources>\n <resource identifier=\"res-1\" type=\"webcontent\" adlcp:${dialect.scormTypeAttr}=\"sco\" href=\"index.html\">\n${fileElements}\n </resource>\n </resources>\n</manifest>`;\n}\n\nexport function generateSCORM12Manifest(\n config: ExportConfig,\n distDir: string\n): string {\n return generateScormManifest('1.2', config, distDir);\n}\n\nexport function generateSCORM2004Manifest(\n config: ExportConfig,\n distDir: string\n): string {\n return generateScormManifest('2004', config, distDir);\n}\n\nexport function generateCMI5Xml(config: ExportConfig): string {\n const title = escapeXml(config.title || 'Tessera Course');\n const description = escapeXml(config.description || '');\n // Derive stable IDs from the course title so they survive rebuilds without\n // orphaning existing learner records in the LRS.\n const courseId = stableUrn('course', `tessera-course:${config.title || ''}`);\n const auId = stableUrn('au', `tessera-au:${config.title || ''}`);\n // cmi5 §10.2.4 caps masteryScore at 4 decimals; avoid float drift like 0.7000000000000001.\n const masteryScore = Number(\n ((config.scoring?.passingScore ?? 70) / 100).toFixed(4)\n );\n // cmi5 §13.1.4 — `moveOn` decides which verb(s) the LMS treats as\n // satisfying the AU. For graded courses (completion gated on a quiz)\n // a learner who completes without passing should NOT receive credit, so\n // the LMS needs both a Completed AND a Passed before satisfaction.\n // Percentage-mode courses don't surface pass/fail, so completion alone\n // is the right signal.\n const moveOn =\n config.completion?.mode === 'quiz' ? 'CompletedAndPassed' : 'Completed';\n\n return `<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<courseStructure xmlns=\"https://w3id.org/xapi/profiles/cmi5/v1/CourseStructure.xsd\">\n <course id=\"${courseId}\">\n <title><langstring lang=\"en-US\">${title}</langstring></title>\n <description><langstring lang=\"en-US\">${description}</langstring></description>\n </course>\n <au id=\"${auId}\" launchMethod=\"AnyWindow\" moveOn=\"${moveOn}\" masteryScore=\"${masteryScore}\">\n <title><langstring lang=\"en-US\">${title}</langstring></title>\n <description><langstring lang=\"en-US\">${description}</langstring></description>\n <url>index.html</url>\n </au>\n</courseStructure>`;\n}\n\n// ---------- ZIP Packaging ----------\n\nexport async function createZip(\n distDir: string,\n outputPath: string\n): Promise<number> {\n return new Promise((res, reject) => {\n const output = createWriteStream(outputPath);\n const archive = new ZipArchive({ zlib: { level: 9 } });\n\n output.on('close', () => {\n res(archive.pointer());\n });\n output.on('error', reject);\n archive.on('error', reject);\n\n archive.pipe(output);\n archive.directory(distDir, false);\n archive.finalize();\n });\n}\n\n// ---------- Main Export ----------\n\n/**\n * Run the export process after Vite build completes.\n * Writes manifest XML into dist/, then packages into ZIP if needed.\n */\n/** Remove any previously built zips for this package to prevent accumulation. */\nfunction cleanOldZips(projectRoot: string, slug: string): void {\n try {\n for (const f of readdirSync(projectRoot)) {\n if (f.startsWith(`${slug}-`) && f.endsWith('.zip')) {\n try { unlinkSync(resolve(projectRoot, f)); } catch {}\n }\n }\n } catch {}\n}\n\n/** Packaged (zipped) export targets: which manifest file to write and how. */\nconst PACKAGED_EXPORTS: Record<\n 'scorm12' | 'scorm2004' | 'cmi5',\n { manifestFile: string; label: string; generate: (config: ExportConfig, distDir: string) => string }\n> = {\n scorm12: { manifestFile: 'imsmanifest.xml', label: 'SCORM 1.2', generate: generateSCORM12Manifest },\n scorm2004: { manifestFile: 'imsmanifest.xml', label: 'SCORM 2004', generate: generateSCORM2004Manifest },\n cmi5: { manifestFile: 'cmi5.xml', label: 'CMI5', generate: (config) => generateCMI5Xml(config) },\n};\n\nexport async function runExport(\n projectRoot: string,\n config: ExportConfig\n): Promise<void> {\n const distDir = resolve(projectRoot, 'dist');\n const standard = config.export?.standard || 'web';\n const slug = slugify(config.title || 'tessera-course') || 'tessera-course';\n const version = config.version || '1.0.0';\n const zipName = `${slug}-${version}.zip`;\n const zipPath = resolve(projectRoot, zipName);\n\n if (standard === 'web') {\n const files = collectFiles(distDir);\n let totalSize = 0;\n for (const f of files) totalSize += statSync(resolve(distDir, f)).size;\n console.log(`✓ Web export: dist/ (${formatSize(totalSize)})`);\n return;\n }\n\n const spec = PACKAGED_EXPORTS[standard as keyof typeof PACKAGED_EXPORTS];\n if (!spec) return; // unknown standard — the validator rejects these upstream\n\n writeFileSync(resolve(distDir, spec.manifestFile), spec.generate(config, distDir), 'utf-8');\n cleanOldZips(projectRoot, slug);\n const zipSize = await createZip(distDir, zipPath);\n console.log(`✓ ${spec.label} export: ${zipName} (${formatSize(zipSize)})`);\n}\n","import type { Plugin, ResolvedConfig, ViteDevServer } from 'vite';\nimport { normalizePath } from 'vite';\nimport { existsSync } from 'node:fs';\nimport { resolve } from 'node:path';\n\nexport interface OverridePluginOptions {\n name: string;\n virtualId: string;\n projectFile: string;\n /** Built-in re-exported when the project file is absent; null export otherwise. */\n builtinFile?: string;\n}\n\n/**\n * A virtual module that resolves to a project-root override file when present,\n * and to the built-in (or a null export) otherwise. Shared by the layout and\n * quiz plugins — they differ only in the virtual id, file name, and built-in.\n */\nexport function createOverridePlugin({\n name,\n virtualId,\n projectFile,\n builtinFile,\n}: OverridePluginOptions): Plugin {\n const resolvedId = '\\0' + virtualId;\n const fallback = builtinFile\n ? `export { default } from '${normalizePath(builtinFile)}';`\n : 'export default null;';\n let filePath: string;\n\n return {\n name,\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n filePath = resolve(config.root, projectFile);\n },\n\n resolveId(id) {\n if (id === virtualId) return resolvedId;\n return null;\n },\n\n load(id) {\n if (id !== resolvedId) return null;\n if (existsSync(filePath)) {\n // Only watch when it exists — addWatchFile on a missing path makes\n // Vite's importAnalysis try to resolve it as a real import.\n this.addWatchFile(filePath);\n return `export { default } from '${normalizePath(filePath)}';`;\n }\n return fallback;\n },\n\n configureServer(server: ViteDevServer) {\n // Only add/unlink flips load()'s output between the override and the\n // fallback; a `change` leaves it identical and Svelte's own HMR handles\n // the underlying file.\n server.watcher.on('all', (event, changed) => {\n if (changed !== filePath) return;\n if (event !== 'add' && event !== 'unlink') return;\n const mod = server.moduleGraph.getModuleById(resolvedId);\n if (mod) server.moduleGraph.invalidateModule(mod);\n server.ws.send({ type: 'full-reload' });\n });\n },\n };\n}\n","import type { Plugin } from 'vite';\nimport { createOverridePlugin } from './override-plugin.js';\n\nexport function tesseraLayoutPlugin(): Plugin {\n return createOverridePlugin({\n name: 'tessera:layout',\n virtualId: 'virtual:tessera-layout',\n projectFile: 'layout.svelte',\n });\n}\n","import type { Plugin } from 'vite';\nimport { resolve, dirname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { createOverridePlugin } from './override-plugin.js';\n\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = dirname(__filename);\n\nexport function tesseraQuizPlugin(): Plugin {\n // The plugin lives in `dist/plugin/quiz.js` after build and `src/plugin/quiz.ts`\n // in source — both put `Quiz.svelte` two levels up under `src/components/`.\n const packageRoot = resolve(__dirname, '..', '..');\n const builtinQuiz = resolve(packageRoot, 'src', 'components', 'Quiz.svelte');\n return createOverridePlugin({\n name: 'tessera:quiz',\n virtualId: 'virtual:tessera-quiz',\n projectFile: 'quiz.svelte',\n builtinFile: builtinQuiz,\n });\n}\n","import type { Plugin, ResolvedConfig, ViteDevServer } from 'vite';\nimport { svelte } from '@sveltejs/vite-plugin-svelte';\nimport { fileURLToPath } from 'node:url';\nimport { dirname, resolve } from 'node:path';\nimport { existsSync, readdirSync, statSync, writeFileSync, unlinkSync, cpSync, mkdirSync } from 'node:fs';\nimport { generateManifest, readCourseConfig } from './manifest.js';\nimport type { Manifest } from './manifest.js';\nimport { validateProject, reportValidationIssues } from './validation.js';\nimport { runExport } from './export.js';\nimport { tesseraLayoutPlugin } from './layout.js';\nimport { tesseraQuizPlugin } from './quiz.js';\n\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = dirname(__filename);\n\n// Resolve the runtime directory where App.svelte lives\nfunction resolveRuntimeDir(): string {\n const packageRoot = resolve(__dirname, '..', '..');\n return resolve(packageRoot, 'src', 'runtime');\n}\n\n// Resolve the framework styles directory\nfunction resolveStylesDir(): string {\n const packageRoot = resolve(__dirname, '..', '..');\n return resolve(packageRoot, 'styles');\n}\n\nexport function tesseraPlugin() {\n const manifestRef: { current: Manifest | null; root: string } = { current: null, root: '' };\n return [\n svelte({\n compilerOptions: { css: 'external' },\n }),\n tesseraValidationPlugin(),\n tesseraEntryPlugin(),\n tesseraConfigPlugin(),\n tesseraPagesPlugin(),\n tesseraManifestPlugin(manifestRef),\n tesseraLayoutPlugin(),\n tesseraQuizPlugin(),\n tesseraAdapterPlugin(),\n tesseraXAPISetupPlugin(),\n tesseraFirstPagePreloadPlugin(manifestRef),\n tesseraExportPlugin(),\n ];\n}\n\n// ---------- Entry Plugin ----------\n\nconst VIRTUAL_ENTRY_ID = 'virtual:tessera-entry';\nconst RESOLVED_ENTRY_ID = '\\0' + VIRTUAL_ENTRY_ID;\nconst VIRTUAL_MAIN_ID = '/virtual:tessera-main';\nconst RESOLVED_MAIN_ID = '\\0virtual:tessera-main';\n\nfunction tesseraEntryPlugin(): Plugin {\n const runtimeDir = resolveRuntimeDir();\n const stylesDir = resolveStylesDir();\n const appSveltePath = resolve(runtimeDir, 'App.svelte');\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:entry',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n },\n\n // For build mode: write index.html so Rollup can find it\n buildStart() {\n if (isBuild) {\n writeFileSync(resolve(projectRoot, 'index.html'), generateIndexHtml(), 'utf-8');\n }\n },\n\n // For build mode: clean up temporary index.html and copy assets\n closeBundle() {\n if (isBuild) {\n const htmlPath = resolve(projectRoot, 'index.html');\n if (existsSync(htmlPath)) {\n try { unlinkSync(htmlPath); } catch {}\n }\n\n // Copy assets/ directory to dist/assets/ so $assets/ references resolve\n const assetsDir = resolve(projectRoot, 'assets');\n const distAssetsDir = resolve(projectRoot, 'dist', 'assets');\n if (existsSync(assetsDir)) {\n mkdirSync(distAssetsDir, { recursive: true });\n cpSync(assetsDir, distAssetsDir, { recursive: true });\n }\n }\n },\n\n // Serve index.html for the dev server\n configureServer(server: ViteDevServer) {\n return () => {\n server.middlewares.use(async (req, res, next) => {\n if (req.url === '/' || req.url === '/index.html') {\n const html = generateIndexHtml();\n const transformed = await server.transformIndexHtml(req.url, html);\n res.setHeader('Content-Type', 'text/html');\n res.statusCode = 200;\n res.end(transformed);\n return;\n }\n next();\n });\n };\n },\n\n resolveId(id) {\n if (id === VIRTUAL_ENTRY_ID) return RESOLVED_ENTRY_ID;\n if (id === VIRTUAL_MAIN_ID || id === 'virtual:tessera-main') return RESOLVED_MAIN_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_ENTRY_ID || id === RESOLVED_MAIN_ID) {\n return generateEntryScript(appSveltePath, stylesDir, projectRoot);\n }\n return null;\n },\n };\n}\n\nfunction generateIndexHtml(): string {\n return `<!DOCTYPE html>\n<html lang=\"en\">\n<head>\n <meta charset=\"UTF-8\" />\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\" />\n <title>Tessera Course</title>\n</head>\n<body>\n <div id=\"tessera-root\"></div>\n <script type=\"module\" src=\"/virtual:tessera-main\"></script>\n</body>\n</html>`;\n}\n\nfunction generateEntryScript(appSveltePath: string, frameworkStylesDir: string, projectRoot: string): string {\n const normalizedPath = appSveltePath.replace(/\\\\/g, '/');\n\n // Framework CSS imports (theme → base → layout)\n const frameworkCssOrder = ['theme.css', 'base.css', 'layout.css'];\n const frameworkImports = frameworkCssOrder\n .map(file => resolve(frameworkStylesDir, file).replace(/\\\\/g, '/'))\n .filter(path => existsSync(path))\n .map(path => `import '${path}';`)\n .join('\\n');\n\n // User CSS imports from project's styles/ directory\n const userStylesDir = resolve(projectRoot, 'styles');\n let userImports = '';\n if (existsSync(userStylesDir)) {\n const userCssFiles = readdirSync(userStylesDir)\n .filter(f => f.endsWith('.css'))\n .sort();\n userImports = userCssFiles\n .map(f => resolve(userStylesDir, f).replace(/\\\\/g, '/'))\n .map(path => `import '${path}';`)\n .join('\\n');\n }\n\n return `// Framework styles\n${frameworkImports}\n// User styles\n${userImports}\n\nimport { mount } from 'svelte';\nimport App from '${normalizedPath}';\n\nmount(App, {\n target: document.getElementById('tessera-root'),\n});\n`;\n}\n\n// ---------- Config Plugin ----------\n\nconst VIRTUAL_CONFIG_ID = 'virtual:tessera-config';\nconst RESOLVED_CONFIG_ID = '\\0' + VIRTUAL_CONFIG_ID;\n\nfunction completionDefaults(mode: string | undefined): {\n completion: Record<string, unknown>;\n passingScore: number;\n} {\n if (mode === 'manual') {\n return { completion: { mode: 'manual' }, passingScore: 0 };\n }\n return { completion: { mode: 'percentage', percentageThreshold: 100 }, passingScore: 70 };\n}\n\nfunction tesseraConfigPlugin(): Plugin {\n let projectRoot: string;\n\n return {\n name: 'tessera:config',\n enforce: 'pre',\n\n config(config) {\n const root = config.root || process.cwd();\n\n return {\n base: './',\n build: {\n assetsDir: 'tessera',\n },\n resolve: {\n alias: {\n '$assets': resolve(root, 'assets'),\n },\n },\n // tessera-learn ships .ts/.svelte.ts source; Vite's dep optimizer\n // doesn't run vite-plugin-svelte's preprocessor, so skip pre-bundling.\n optimizeDeps: {\n exclude: ['tessera-learn'],\n },\n };\n },\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n },\n\n resolveId(id) {\n if (id === VIRTUAL_CONFIG_ID) return RESOLVED_CONFIG_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_CONFIG_ID) {\n const configPath = resolve(projectRoot, 'course.config.js');\n if (existsSync(configPath)) this.addWatchFile(configPath);\n const read = readCourseConfig(projectRoot);\n const userConfig: Record<string, any> = read.ok ? read.config : {};\n\n const { completion, passingScore } = completionDefaults(userConfig.completion?.mode);\n const merged = {\n title: userConfig.title || 'Untitled Course',\n ...userConfig,\n navigation: { mode: 'free', ...userConfig.navigation },\n completion: { ...completion, ...userConfig.completion },\n scoring: { passingScore, ...userConfig.scoring },\n export: { standard: 'web', ...userConfig.export },\n };\n\n return `export default ${JSON.stringify(merged)};`;\n }\n return null;\n },\n };\n}\n\n// ---------- Manifest Watch Helpers ----------\n\n/** Register all _meta.js and .svelte files under pagesDir as watch files for build mode. */\nfunction addWatchFiles(ctx: { addWatchFile(id: string): void }, dir: string): void {\n if (!existsSync(dir)) return;\n for (const entry of readdirSync(dir)) {\n const full = resolve(dir, entry);\n if (statSync(full).isDirectory()) {\n addWatchFiles(ctx, full);\n } else if (entry.endsWith('.svelte') || entry === '_meta.js') {\n ctx.addWatchFile(full);\n }\n }\n}\n\n// ---------- Pages Plugin ----------\n\nconst VIRTUAL_PAGES_ID = 'virtual:tessera-pages';\nconst RESOLVED_PAGES_ID = '\\0' + VIRTUAL_PAGES_ID;\n\n/**\n * Provides a virtual module that exports an import.meta.glob map for all .svelte\n * pages. This runs in the user's project context so the glob resolves against their\n * pages/ directory, and Vite can statically analyze it for code splitting.\n */\nfunction tesseraPagesPlugin(): Plugin {\n return {\n name: 'tessera:pages',\n enforce: 'pre',\n\n resolveId(id) {\n if (id === VIRTUAL_PAGES_ID) return RESOLVED_PAGES_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_PAGES_ID) {\n return `export default import.meta.glob('/pages/**/*.svelte');`;\n }\n return null;\n },\n };\n}\n\n// ---------- Validation Plugin ----------\n\nfunction tesseraValidationPlugin(): Plugin {\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:validation',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n // Run validation during dev (configResolved fires before server starts)\n if (!isBuild) {\n runValidation(projectRoot);\n }\n },\n\n buildStart() {\n // Run validation during build (buildStart fires once before bundling)\n if (isBuild) {\n runValidation(projectRoot);\n }\n },\n };\n}\n\nfunction runValidation(projectRoot: string): void {\n const result = validateProject(projectRoot);\n reportValidationIssues(result);\n if (result.errors.length > 0) {\n throw new Error(\n `Tessera validation failed with ${result.errors.length} error(s). Fix the errors above to continue.`\n );\n }\n}\n\n// ---------- Export Plugin ----------\n\nfunction tesseraExportPlugin(): Plugin {\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:export',\n enforce: 'post',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n },\n\n async closeBundle() {\n if (!isBuild) return;\n\n const read = readCourseConfig(projectRoot);\n if (!read.ok) {\n // Validation already required a parseable course.config.js — getting\n // here means it vanished or broke mid-build. Surface that loudly\n // rather than shipping a bundle with no LMS export silently.\n if (read.reason === 'missing') {\n throw new Error(\n '[tessera:export] course.config.js not found at closeBundle. The file must exist for the export step to run.'\n );\n }\n if (read.reason === 'no-export') {\n throw new Error(\n '[tessera:export] course.config.js: could not locate `export default { ... }`. Cannot determine export.standard.'\n );\n }\n throw new Error(\n `[tessera:export] course.config.js: failed to parse export-default object literal — ${(read.error as Error).message}`\n );\n }\n\n await runExport(projectRoot, read.config as Parameters<typeof runExport>[1]);\n },\n };\n}\n\n// ---------- Manifest Plugin ----------\n\nconst VIRTUAL_MANIFEST_ID = 'virtual:tessera-manifest';\nconst RESOLVED_MANIFEST_ID = '\\0' + VIRTUAL_MANIFEST_ID;\n\nfunction tesseraManifestPlugin(manifestRef: { current: Manifest | null; root: string }): Plugin {\n let projectRoot: string;\n let pagesDir: string;\n let server: ViteDevServer | null = null;\n\n function buildManifest(): Manifest {\n const m = generateManifest(pagesDir);\n manifestRef.current = m;\n return m;\n }\n\n return {\n name: 'tessera:manifest',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n pagesDir = resolve(projectRoot, 'pages');\n manifestRef.root = projectRoot;\n },\n\n configureServer(devServer: ViteDevServer) {\n server = devServer;\n\n // Watch the pages directory for changes\n devServer.watcher.on('all', (event, filePath) => {\n if (!filePath.startsWith(pagesDir)) return;\n\n // Rebuild manifest on relevant file changes\n const isRelevant =\n filePath.endsWith('.svelte') ||\n filePath.endsWith('_meta.js') ||\n event === 'addDir' ||\n event === 'unlinkDir';\n\n if (isRelevant) {\n manifestRef.current = null; // invalidate cache\n\n // Invalidate the virtual module to trigger HMR\n const mod = devServer.moduleGraph.getModuleById(RESOLVED_MANIFEST_ID);\n if (mod) {\n devServer.moduleGraph.invalidateModule(mod);\n devServer.ws.send({ type: 'full-reload' });\n }\n\n console.log(`[tessera] Manifest rebuilt (${event}: ${filePath.replace(projectRoot, '')})`);\n }\n });\n },\n\n buildStart() {\n buildManifest();\n },\n\n resolveId(id) {\n if (id === VIRTUAL_MANIFEST_ID) return RESOLVED_MANIFEST_ID;\n return null;\n },\n\n load(id) {\n if (id === RESOLVED_MANIFEST_ID) {\n if (!manifestRef.current) {\n buildManifest();\n }\n\n // Register watch files so Vite's built-in watcher (used in build --watch)\n // knows to re-trigger when pages/ content changes.\n addWatchFiles(this, pagesDir);\n\n // Encode as base64 to prevent Vite's import analysis from\n // scanning .svelte importPath strings as module imports.\n // Replace Infinity with 1e9 since JSON.stringify drops it.\n const json = JSON.stringify(manifestRef.current, (_key, value) =>\n value === Infinity ? 1e9 : value\n );\n const b64 = Buffer.from(json).toString('base64');\n return `export default JSON.parse(atob(\"${b64}\"));`;\n }\n return null;\n },\n };\n}\n\nconst VIRTUAL_ADAPTER_ID = 'virtual:tessera-adapter';\nconst RESOLVED_ADAPTER_ID = '\\0' + VIRTUAL_ADAPTER_ID;\n\nfunction tesseraAdapterPlugin(): Plugin {\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:adapter',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n },\n\n resolveId(id) {\n if (id === VIRTUAL_ADAPTER_ID) return RESOLVED_ADAPTER_ID;\n return null;\n },\n\n load(id) {\n if (id !== RESOLVED_ADAPTER_ID) return null;\n\n // In dev, defer to the runtime selector so its WebAdapter fallback\n // for unreachable LMS APIs keeps working.\n if (!isBuild) {\n return `export { createAdapter } from 'tessera-learn/runtime/adapters/index.js';`;\n }\n\n let standard = 'web';\n const read = readCourseConfig(projectRoot);\n if (read.ok && typeof read.config.export?.standard === 'string') {\n standard = read.config.export.standard;\n }\n\n switch (standard) {\n case 'scorm12':\n return `\nimport { SCORM12Adapter } from 'tessera-learn/runtime/adapters/scorm12.js';\nimport { findSCORM12API } from 'tessera-learn/runtime/adapters/discovery.js';\nimport { LMSAdapterError } from 'tessera-learn/runtime/adapters/index.js';\nexport function createAdapter() {\n const api = findSCORM12API();\n if (!api) throw new LMSAdapterError('scorm12', 'Tessera: SCORM 1.2 API not found in window.parent/opener chain. Course must be launched from a SCORM 1.2 LMS.');\n return new SCORM12Adapter(api);\n}\n`;\n case 'scorm2004':\n return `\nimport { SCORM2004Adapter } from 'tessera-learn/runtime/adapters/scorm2004.js';\nimport { findSCORM2004API } from 'tessera-learn/runtime/adapters/discovery.js';\nimport { LMSAdapterError } from 'tessera-learn/runtime/adapters/index.js';\nexport function createAdapter() {\n const api = findSCORM2004API();\n if (!api) throw new LMSAdapterError('scorm2004', 'Tessera: SCORM 2004 API not found in window.parent/opener chain. Course must be launched from a SCORM 2004 LMS.');\n return new SCORM2004Adapter(api);\n}\n`;\n case 'cmi5':\n return `\nimport { CMI5Adapter } from 'tessera-learn/runtime/adapters/cmi5.js';\nimport { hasCMI5LaunchParams } from 'tessera-learn/runtime/adapters/discovery.js';\nimport { LMSAdapterError } from 'tessera-learn/runtime/adapters/index.js';\nexport function createAdapter() {\n if (!hasCMI5LaunchParams()) throw new LMSAdapterError('cmi5', 'Tessera: cmi5 launch parameters not present on URL. Course must be launched from a cmi5-compliant LMS.');\n return new CMI5Adapter();\n}\n`;\n default:\n return `\nimport { WebAdapter } from 'tessera-learn/runtime/adapters/web.js';\nexport function createAdapter(config) {\n return new WebAdapter(config);\n}\n`;\n }\n },\n };\n}\n\nconst VIRTUAL_XAPI_SETUP_ID = 'virtual:tessera-xapi-setup';\nconst RESOLVED_XAPI_SETUP_ID = '\\0' + VIRTUAL_XAPI_SETUP_ID;\n\nfunction tesseraXAPISetupPlugin(): Plugin {\n let projectRoot: string;\n let isBuild = false;\n\n return {\n name: 'tessera:xapi-setup',\n enforce: 'pre',\n\n configResolved(config: ResolvedConfig) {\n projectRoot = config.root;\n isBuild = config.command === 'build';\n },\n\n resolveId(id) {\n if (id === VIRTUAL_XAPI_SETUP_ID) return RESOLVED_XAPI_SETUP_ID;\n return null;\n },\n\n load(id) {\n if (id !== RESOLVED_XAPI_SETUP_ID) return null;\n\n if (!isBuild) {\n return `export { buildXAPIClient } from 'tessera-learn/runtime/xapi/setup.js';`;\n }\n\n let standard = 'web';\n let hasXapi = false;\n const read = readCourseConfig(projectRoot);\n if (read.ok) {\n if (typeof read.config.export?.standard === 'string') standard = read.config.export.standard;\n hasXapi = read.config.xapi != null;\n }\n\n // cmi5 needs the publisher regardless of explicit xapi config (cmi5\n // adapter shares the publisher queue for its own LMS-required statements).\n if (hasXapi || standard === 'cmi5') {\n return `export { buildXAPIClient } from 'tessera-learn/runtime/xapi/setup.js';`;\n }\n\n return `export async function buildXAPIClient() { return null; }`;\n },\n };\n}\n\nfunction tesseraFirstPagePreloadPlugin(manifestRef: { current: Manifest | null; root: string }): Plugin {\n return {\n name: 'tessera:first-page-preload',\n apply: 'build',\n transformIndexHtml: {\n order: 'post',\n handler(_html, ctx) {\n const firstPagePath = manifestRef.current?.pages[0]?.importPath;\n if (!firstPagePath || !ctx.bundle) return;\n const normalized = resolve(manifestRef.root, firstPagePath.replace(/^\\//, '')).replace(/\\\\/g, '/');\n const chunk = Object.values(ctx.bundle).find(\n (c): c is import('vite').Rollup.OutputChunk =>\n c.type === 'chunk' && !!c.facadeModuleId && c.facadeModuleId.replace(/\\\\/g, '/') === normalized\n );\n if (!chunk) return;\n return [{\n tag: 'link',\n attrs: { rel: 'modulepreload', href: `./${chunk.fileName}` },\n injectTo: 'head',\n }];\n },\n },\n };\n}\n"],"mappings":"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"}
@@ -57,6 +57,37 @@ function extractDefaultExportObjectLiteral(source) {
57
57
  return extractObjectLiteral(source, startIndex);
58
58
  }
59
59
  /**
60
+ * Read and JSON5-parse the `export default { ... }` literal from a project's
61
+ * course.config.js. Shared by the build plugin and the validator so the read,
62
+ * cache, and parse rules live in one place. The discriminated `reason` lets
63
+ * callers that care (export, validation) emit precise errors while callers
64
+ * that just need a value can fall back on `!ok`.
65
+ */
66
+ function readCourseConfig(projectRoot) {
67
+ const configPath = resolve(projectRoot, "course.config.js");
68
+ if (!existsSync(configPath)) return {
69
+ ok: false,
70
+ reason: "missing"
71
+ };
72
+ const objectStr = extractDefaultExportObjectLiteral(readSourceFileCached(configPath));
73
+ if (!objectStr) return {
74
+ ok: false,
75
+ reason: "no-export"
76
+ };
77
+ try {
78
+ return {
79
+ ok: true,
80
+ config: JSON5.parse(objectStr)
81
+ };
82
+ } catch (error) {
83
+ return {
84
+ ok: false,
85
+ reason: "parse-error",
86
+ error
87
+ };
88
+ }
89
+ }
90
+ /**
60
91
  * Read a _meta.js file and extract its default export object.
61
92
  * Uses the same JSON5 approach as pageConfig extraction — find the object literal
62
93
  * after `export default` and parse it.
@@ -100,11 +131,14 @@ function extractPageConfig(filePath) {
100
131
  return {};
101
132
  }
102
133
  /**
103
- * Extract an object literal from source starting at the opening brace.
104
- * Tracks brace depth to find the matching closing brace.
134
+ * Extract a balanced `{...}` or `[...]` span starting at the opening bracket,
135
+ * skipping strings and comments. Returns the substring (inclusive) or null if
136
+ * the open char is wrong or no matching close is found. Shared by manifest
137
+ * extraction, _meta/pageConfig parsing, and the validator's tag-prop parser.
105
138
  */
106
139
  function extractObjectLiteral(source, startIndex) {
107
- if (source[startIndex] !== "{") return null;
140
+ const open = source[startIndex];
141
+ if (open !== "{" && open !== "[") return null;
108
142
  let depth = 0;
109
143
  let inString = null;
110
144
  let escaped = false;
@@ -136,8 +170,8 @@ function extractObjectLiteral(source, startIndex) {
136
170
  i = end === -1 ? source.length : end + 1;
137
171
  continue;
138
172
  }
139
- if (char === "{") depth++;
140
- if (char === "}") {
173
+ if (char === "{" || char === "[") depth++;
174
+ if (char === "}" || char === "]") {
141
175
  depth--;
142
176
  if (depth === 0) return source.slice(startIndex, i + 1);
143
177
  }
@@ -240,6 +274,10 @@ function orderPageFiles(allFiles, pagesArray) {
240
274
  * (build-time validation of static `course.config.js` actor / auth).
241
275
  * Keeping the rules in one place prevents the two callsites from drifting.
242
276
  */
277
+ /** Join a field label with a validator suffix: `.foo` chains, others get `: `. */
278
+ function joinFieldError(label, suffix) {
279
+ return suffix.startsWith(".") ? `${label}${suffix}` : `${label}: ${suffix}`;
280
+ }
243
281
  /**
244
282
  * Validate that a candidate is an Identified Agent per xAPI 1.0.3.
245
283
  * Returns null on success or a human-readable error suffix on failure.
@@ -286,8 +324,24 @@ function validateAgent(actor) {
286
324
  }
287
325
  return null;
288
326
  }
327
+ /**
328
+ * Validate a Basic-auth credential string (the value after "Basic ").
329
+ * v1 supports Basic only. Bearer is a hard error so OAuth users see the
330
+ * non-goal explicitly.
331
+ */
332
+ function validateAuthCredential(auth) {
333
+ if (typeof auth !== "string" || !auth) return "must be a non-empty string";
334
+ if (/^basic\s/i.test(auth)) return "must be the Basic credential value only, not the full header. Drop the 'Basic ' prefix.";
335
+ if (/^bearer\s/i.test(auth)) return "Bearer/OAuth credentials are not supported in v1. Use Basic auth, or wrap your token-exchange in an auth function that returns a Basic credential.";
336
+ return null;
337
+ }
289
338
  //#endregion
290
339
  //#region src/plugin/validation.ts
340
+ /** Print validation warnings (yellow) then errors (red). Shared by the dev/build plugin and the CLI. */
341
+ function reportValidationIssues({ errors, warnings }) {
342
+ for (const warning of warnings) console.warn(`\x1b[33m[tessera warning]\x1b[0m ${warning}`);
343
+ for (const error of errors) console.error(`\x1b[31m[tessera error]\x1b[0m ${error}`);
344
+ }
291
345
  const KNOWN_CONFIG_FIELDS = new Set([
292
346
  "title",
293
347
  "description",
@@ -322,15 +376,14 @@ const VALID_REQUIRE_SUCCESS_STATUS = ["passed", "failed"];
322
376
  function validateProject(projectRoot) {
323
377
  const errors = [];
324
378
  const warnings = [];
325
- const configPath = resolve(projectRoot, "course.config.js");
326
- if (!existsSync(configPath)) {
379
+ if (!existsSync(resolve(projectRoot, "course.config.js"))) {
327
380
  errors.push("course.config.js not found in project root");
328
381
  return {
329
382
  errors,
330
383
  warnings
331
384
  };
332
385
  }
333
- const config = parseConfig(configPath, errors, warnings);
386
+ const config = parseConfig(projectRoot, errors, warnings);
334
387
  const pageResults = validatePages(resolve(projectRoot, "pages"), resolve(projectRoot, "assets"), projectRoot);
335
388
  errors.push(...pageResults.errors);
336
389
  warnings.push(...pageResults.warnings);
@@ -344,19 +397,14 @@ function validateProject(projectRoot) {
344
397
  warnings
345
398
  };
346
399
  }
347
- function parseConfig(configPath, errors, warnings) {
348
- const objectStr = extractDefaultExportObjectLiteral(readSourceFileCached(configPath));
349
- if (!objectStr) {
350
- errors.push("course.config.js: could not parse — must use `export default { ... }` syntax");
351
- return null;
352
- }
353
- let config;
354
- try {
355
- config = JSON5.parse(objectStr);
356
- } catch {
357
- errors.push("course.config.js: syntax error — must export a static object literal");
400
+ function parseConfig(projectRoot, errors, warnings) {
401
+ const read = readCourseConfig(projectRoot);
402
+ if (!read.ok) {
403
+ if (read.reason === "no-export") errors.push("course.config.js: could not parse — must use `export default { ... }` syntax");
404
+ else if (read.reason === "parse-error") errors.push("course.config.js: syntax error — must export a static object literal");
358
405
  return null;
359
406
  }
407
+ const config = read.config;
360
408
  for (const key of Object.keys(config)) if (!KNOWN_CONFIG_FIELDS.has(key)) warnings.push(`course.config.js: unknown field "${key}" — will be ignored`);
361
409
  if (config.navigation?.mode !== void 0) {
362
410
  if (!VALID_NAV_MODES.includes(config.navigation.mode)) errors.push(`course.config.js: "navigation.mode" must be "free" or "sequential", got "${config.navigation.mode}"`);
@@ -452,11 +500,11 @@ function validateSingleXAPIEntry(entry, label, standard, errors, warnings) {
452
500
  if (!endpoint.endsWith("/")) warnings.push(`course.config.js: ${label}.endpoint should end with a slash to avoid concatenation surprises (e.g. 'https://lrs.example.com/xapi/' not 'https://lrs.example.com/xapi'). Runtime normalizes regardless.`);
453
501
  const auth = entry.auth;
454
502
  if (auth === void 0) errors.push(`course.config.js: ${label}.auth is required`);
455
- else if (typeof auth === "string") if (!auth) errors.push(`course.config.js: ${label}.auth must be a non-empty string`);
456
- else if (/^basic\s/i.test(auth)) errors.push(`course.config.js: ${label}.auth must be the Basic credential value only, not the full header. Drop the 'Basic ' prefix.`);
457
- else if (/^bearer\s/i.test(auth)) errors.push(`course.config.js: ${label}.auth: Bearer/OAuth credentials are not supported in v1. Use Basic auth, or wrap your token-exchange in an auth function that returns a Basic credential.`);
458
- else warnings.push(`course.config.js: ${label}.auth is a static string and will be embedded in the bundle. For production, pass a function that fetches a short-lived token from a server endpoint.`);
459
- else if (typeof auth !== "function") errors.push(`course.config.js: ${label}.auth must be a string or a function, got ${typeof auth}`);
503
+ else if (typeof auth === "string") {
504
+ const authErr = validateAuthCredential(auth);
505
+ if (authErr) errors.push(`course.config.js: ${joinFieldError(`${label}.auth`, authErr)}`);
506
+ else warnings.push(`course.config.js: ${label}.auth is a static string and will be embedded in the bundle. For production, pass a function that fetches a short-lived token from a server endpoint.`);
507
+ } else if (typeof auth !== "function") errors.push(`course.config.js: ${label}.auth must be a string or a function, got ${typeof auth}`);
460
508
  const activityId = entry.activityId;
461
509
  if (activityId === void 0 || activityId === "") errors.push(`course.config.js: ${label}.activityId is required`);
462
510
  else if (typeof activityId !== "string") errors.push(`course.config.js: ${label}.activityId must be a string`);
@@ -470,10 +518,7 @@ function validateSingleXAPIEntry(entry, label, standard, errors, warnings) {
470
518
  if (standard === "web") errors.push(`course.config.js: ${label}.actor is required for web export — there is no LMS to derive a learner identity from. Provide either a static actor object or a function that resolves one (e.g. from your auth system).`);
471
519
  } else if (typeof actor === "object" && actor !== null) {
472
520
  const err = validateAgent(actor);
473
- if (err) {
474
- const joined = err.startsWith(".") ? `${label}.actor${err}` : `${label}.actor ${err}`;
475
- errors.push(`course.config.js: ${joined}`);
476
- }
521
+ if (err) errors.push(`course.config.js: ${joinFieldError(`${label}.actor`, err)}`);
477
522
  } else if (typeof actor !== "function") errors.push(`course.config.js: ${label}.actor must be an object or function, got ${typeof actor}`);
478
523
  const aahp = entry.actorAccountHomePage;
479
524
  if (aahp !== void 0) {
@@ -693,39 +738,6 @@ const QUESTION_COMPONENT_REQUIRED = {
693
738
  "correct"
694
739
  ]
695
740
  };
696
- /** Extract a balanced {...} or [...] span starting at startIndex, or null. */
697
- function extractBalanced(source, startIndex) {
698
- const open = source[startIndex];
699
- if (open !== "{" && open !== "[") return null;
700
- let depth = 0;
701
- let inString = null;
702
- let escaped = false;
703
- for (let i = startIndex; i < source.length; i++) {
704
- const char = source[i];
705
- if (escaped) {
706
- escaped = false;
707
- continue;
708
- }
709
- if (char === "\\" && inString) {
710
- escaped = true;
711
- continue;
712
- }
713
- if (inString) {
714
- if (char === inString) inString = null;
715
- continue;
716
- }
717
- if (char === "\"" || char === "'" || char === "`") {
718
- inString = char;
719
- continue;
720
- }
721
- if (char === "{" || char === "[") depth++;
722
- if (char === "}" || char === "]") {
723
- depth--;
724
- if (depth === 0) return source.slice(startIndex, i + 1);
725
- }
726
- }
727
- return null;
728
- }
729
741
  /**
730
742
  * Parse the props of an opening tag starting just after the component name.
731
743
  * Returns null if the tag can't be parsed cleanly — callers then skip it
@@ -741,7 +753,7 @@ function parseTagProps(content, start) {
741
753
  if (c === ">") return props;
742
754
  if (c === "/" && content[i + 1] === ">") return props;
743
755
  if (c === "{") {
744
- const block = extractBalanced(content, i);
756
+ const block = extractObjectLiteral(content, i);
745
757
  if (!block) return null;
746
758
  i += block.length;
747
759
  continue;
@@ -767,7 +779,7 @@ function parseTagProps(content, start) {
767
779
  });
768
780
  i = end + 1;
769
781
  } else if (v === "{") {
770
- const block = extractBalanced(content, i);
782
+ const block = extractObjectLiteral(content, i);
771
783
  if (!block) return null;
772
784
  props.set(propName, {
773
785
  kind: "expr",
@@ -903,6 +915,6 @@ function crossValidate(config, pageResults, errors, warnings) {
903
915
  }
904
916
  }
905
917
  //#endregion
906
- export { extractDefaultExportObjectLiteral as n, generateManifest as r, validateProject as t };
918
+ export { readCourseConfig as i, validateProject as n, generateManifest as r, reportValidationIssues as t };
907
919
 
908
- //# sourceMappingURL=validation-BxWAMMnJ.js.map
920
+ //# sourceMappingURL=validation-D9DXlqNP.js.map