@redaksjon/protokoll-engine 0.1.1-dev.0

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 (240) hide show
  1. package/README.md +47 -0
  2. package/dist/agentic/executor.d.ts +21 -0
  3. package/dist/agentic/executor.d.ts.map +1 -0
  4. package/dist/agentic/index.d.ts +27 -0
  5. package/dist/agentic/index.d.ts.map +1 -0
  6. package/dist/agentic/registry.d.ts +11 -0
  7. package/dist/agentic/registry.d.ts.map +1 -0
  8. package/dist/agentic/tools/lookup-person.d.ts +3 -0
  9. package/dist/agentic/tools/lookup-person.d.ts.map +1 -0
  10. package/dist/agentic/tools/lookup-project.d.ts +3 -0
  11. package/dist/agentic/tools/lookup-project.d.ts.map +1 -0
  12. package/dist/agentic/tools/route-note.d.ts +3 -0
  13. package/dist/agentic/tools/route-note.d.ts.map +1 -0
  14. package/dist/agentic/tools/store-context.d.ts +3 -0
  15. package/dist/agentic/tools/store-context.d.ts.map +1 -0
  16. package/dist/agentic/tools/verify-spelling.d.ts +3 -0
  17. package/dist/agentic/tools/verify-spelling.d.ts.map +1 -0
  18. package/dist/agentic/types.d.ts +110 -0
  19. package/dist/agentic/types.d.ts.map +1 -0
  20. package/dist/constants.d.ts +98 -0
  21. package/dist/constants.d.ts.map +1 -0
  22. package/dist/feedback/analyzer.d.ts +13 -0
  23. package/dist/feedback/analyzer.d.ts.map +1 -0
  24. package/dist/feedback/decision-tracker.d.ts +14 -0
  25. package/dist/feedback/decision-tracker.d.ts.map +1 -0
  26. package/dist/feedback/handler.d.ts +14 -0
  27. package/dist/feedback/handler.d.ts.map +1 -0
  28. package/dist/feedback/index.d.ts +12 -0
  29. package/dist/feedback/index.d.ts.map +1 -0
  30. package/dist/feedback/types.d.ts +72 -0
  31. package/dist/feedback/types.d.ts.map +1 -0
  32. package/dist/index.d.ts +24 -0
  33. package/dist/index.d.ts.map +1 -0
  34. package/dist/index.js +32 -0
  35. package/dist/index.js.map +1 -0
  36. package/dist/index10.js +4 -0
  37. package/dist/index10.js.map +1 -0
  38. package/dist/index11.js +22 -0
  39. package/dist/index11.js.map +1 -0
  40. package/dist/index12.js +125 -0
  41. package/dist/index12.js.map +1 -0
  42. package/dist/index13.js +124 -0
  43. package/dist/index13.js.map +1 -0
  44. package/dist/index14.js +296 -0
  45. package/dist/index14.js.map +1 -0
  46. package/dist/index15.js +100 -0
  47. package/dist/index15.js.map +1 -0
  48. package/dist/index16.js +107 -0
  49. package/dist/index16.js.map +1 -0
  50. package/dist/index17.js +185 -0
  51. package/dist/index17.js.map +1 -0
  52. package/dist/index18.js +53 -0
  53. package/dist/index18.js.map +1 -0
  54. package/dist/index19.js +19 -0
  55. package/dist/index19.js.map +1 -0
  56. package/dist/index2.js +33 -0
  57. package/dist/index2.js.map +1 -0
  58. package/dist/index20.js +105 -0
  59. package/dist/index20.js.map +1 -0
  60. package/dist/index21.js +26 -0
  61. package/dist/index21.js.map +1 -0
  62. package/dist/index22.js +49 -0
  63. package/dist/index22.js.map +1 -0
  64. package/dist/index23.js +119 -0
  65. package/dist/index23.js.map +1 -0
  66. package/dist/index24.js +330 -0
  67. package/dist/index24.js.map +1 -0
  68. package/dist/index25.js +57 -0
  69. package/dist/index25.js.map +1 -0
  70. package/dist/index26.js +38 -0
  71. package/dist/index26.js.map +1 -0
  72. package/dist/index27.js +127 -0
  73. package/dist/index27.js.map +1 -0
  74. package/dist/index28.js +157 -0
  75. package/dist/index28.js.map +1 -0
  76. package/dist/index29.js +163 -0
  77. package/dist/index29.js.map +1 -0
  78. package/dist/index3.js +36 -0
  79. package/dist/index3.js.map +1 -0
  80. package/dist/index30.js +173 -0
  81. package/dist/index30.js.map +1 -0
  82. package/dist/index31.js +423 -0
  83. package/dist/index31.js.map +1 -0
  84. package/dist/index32.js +161 -0
  85. package/dist/index32.js.map +1 -0
  86. package/dist/index33.js +152 -0
  87. package/dist/index33.js.map +1 -0
  88. package/dist/index34.js +56 -0
  89. package/dist/index34.js.map +1 -0
  90. package/dist/index35.js +103 -0
  91. package/dist/index35.js.map +1 -0
  92. package/dist/index36.js +451 -0
  93. package/dist/index36.js.map +1 -0
  94. package/dist/index37.js +431 -0
  95. package/dist/index37.js.map +1 -0
  96. package/dist/index38.js +87 -0
  97. package/dist/index38.js.map +1 -0
  98. package/dist/index39.js +122 -0
  99. package/dist/index39.js.map +1 -0
  100. package/dist/index4.js +3 -0
  101. package/dist/index4.js.map +1 -0
  102. package/dist/index40.js +299 -0
  103. package/dist/index40.js.map +1 -0
  104. package/dist/index41.js +49 -0
  105. package/dist/index41.js.map +1 -0
  106. package/dist/index42.js +151 -0
  107. package/dist/index42.js.map +1 -0
  108. package/dist/index43.js +226 -0
  109. package/dist/index43.js.map +1 -0
  110. package/dist/index44.js +49 -0
  111. package/dist/index44.js.map +1 -0
  112. package/dist/index45.js +45 -0
  113. package/dist/index45.js.map +1 -0
  114. package/dist/index46.js +37 -0
  115. package/dist/index46.js.map +1 -0
  116. package/dist/index47.js +51 -0
  117. package/dist/index47.js.map +1 -0
  118. package/dist/index48.js +39 -0
  119. package/dist/index48.js.map +1 -0
  120. package/dist/index49.js +239 -0
  121. package/dist/index49.js.map +1 -0
  122. package/dist/index5.js +17 -0
  123. package/dist/index5.js.map +1 -0
  124. package/dist/index50.js +163 -0
  125. package/dist/index50.js.map +1 -0
  126. package/dist/index51.js +81 -0
  127. package/dist/index51.js.map +1 -0
  128. package/dist/index52.js +78 -0
  129. package/dist/index52.js.map +1 -0
  130. package/dist/index53.js +22 -0
  131. package/dist/index53.js.map +1 -0
  132. package/dist/index54.js +8 -0
  133. package/dist/index54.js.map +1 -0
  134. package/dist/index55.js +8 -0
  135. package/dist/index55.js.map +1 -0
  136. package/dist/index56.js +17 -0
  137. package/dist/index56.js.map +1 -0
  138. package/dist/index57.js +4 -0
  139. package/dist/index57.js.map +1 -0
  140. package/dist/index58.js +17 -0
  141. package/dist/index58.js.map +1 -0
  142. package/dist/index59.js +4 -0
  143. package/dist/index59.js.map +1 -0
  144. package/dist/index6.js +22 -0
  145. package/dist/index6.js.map +1 -0
  146. package/dist/index60.js +6 -0
  147. package/dist/index60.js.map +1 -0
  148. package/dist/index7.js +27 -0
  149. package/dist/index7.js.map +1 -0
  150. package/dist/index8.js +22 -0
  151. package/dist/index8.js.map +1 -0
  152. package/dist/index9.js +5 -0
  153. package/dist/index9.js.map +1 -0
  154. package/dist/logging.d.ts +7 -0
  155. package/dist/logging.d.ts.map +1 -0
  156. package/dist/output/index.d.ts +15 -0
  157. package/dist/output/index.d.ts.map +1 -0
  158. package/dist/phases/complete.d.ts +17 -0
  159. package/dist/phases/complete.d.ts.map +1 -0
  160. package/dist/phases/index.d.ts +5 -0
  161. package/dist/phases/index.d.ts.map +1 -0
  162. package/dist/phases/locate.d.ts +15 -0
  163. package/dist/phases/locate.d.ts.map +1 -0
  164. package/dist/phases/simple-replace.d.ts +72 -0
  165. package/dist/phases/simple-replace.d.ts.map +1 -0
  166. package/dist/phases/transcribe.d.ts +19 -0
  167. package/dist/phases/transcribe.d.ts.map +1 -0
  168. package/dist/pipeline/index.d.ts +10 -0
  169. package/dist/pipeline/index.d.ts.map +1 -0
  170. package/dist/pipeline/orchestrator.d.ts +13 -0
  171. package/dist/pipeline/orchestrator.d.ts.map +1 -0
  172. package/dist/pipeline/types.d.ts +58 -0
  173. package/dist/pipeline/types.d.ts.map +1 -0
  174. package/dist/prompt/index.d.ts +3 -0
  175. package/dist/prompt/index.d.ts.map +1 -0
  176. package/dist/prompt/templates.d.ts +40 -0
  177. package/dist/prompt/templates.d.ts.map +1 -0
  178. package/dist/prompt/transcribe.d.ts +42 -0
  179. package/dist/prompt/transcribe.d.ts.map +1 -0
  180. package/dist/reasoning/client.d.ts +42 -0
  181. package/dist/reasoning/client.d.ts.map +1 -0
  182. package/dist/reasoning/index.d.ts +17 -0
  183. package/dist/reasoning/index.d.ts.map +1 -0
  184. package/dist/reasoning/strategy.d.ts +12 -0
  185. package/dist/reasoning/strategy.d.ts.map +1 -0
  186. package/dist/reasoning/types.d.ts +58 -0
  187. package/dist/reasoning/types.d.ts.map +1 -0
  188. package/dist/reflection/collector.d.ts +18 -0
  189. package/dist/reflection/collector.d.ts.map +1 -0
  190. package/dist/reflection/index.d.ts +13 -0
  191. package/dist/reflection/index.d.ts.map +1 -0
  192. package/dist/reflection/reporter.d.ts +10 -0
  193. package/dist/reflection/reporter.d.ts.map +1 -0
  194. package/dist/reflection/types.d.ts +99 -0
  195. package/dist/reflection/types.d.ts.map +1 -0
  196. package/dist/routing/classifier.d.ts +8 -0
  197. package/dist/routing/classifier.d.ts.map +1 -0
  198. package/dist/routing/index.d.ts +12 -0
  199. package/dist/routing/index.d.ts.map +1 -0
  200. package/dist/routing/router.d.ts +8 -0
  201. package/dist/routing/router.d.ts.map +1 -0
  202. package/dist/routing/types.d.ts +68 -0
  203. package/dist/routing/types.d.ts.map +1 -0
  204. package/dist/transcript/feedback.d.ts +70 -0
  205. package/dist/transcript/feedback.d.ts.map +1 -0
  206. package/dist/transcript/index.d.ts +10 -0
  207. package/dist/transcript/index.d.ts.map +1 -0
  208. package/dist/transcript/operations.d.ts +152 -0
  209. package/dist/transcript/operations.d.ts.map +1 -0
  210. package/dist/transcript/pkl-utils.d.ts +66 -0
  211. package/dist/transcript/pkl-utils.d.ts.map +1 -0
  212. package/dist/transcription/index.d.ts +17 -0
  213. package/dist/transcription/index.d.ts.map +1 -0
  214. package/dist/transcription/service.d.ts +10 -0
  215. package/dist/transcription/service.d.ts.map +1 -0
  216. package/dist/transcription/types.d.ts +41 -0
  217. package/dist/transcription/types.d.ts.map +1 -0
  218. package/dist/types.d.ts +28 -0
  219. package/dist/types.d.ts.map +1 -0
  220. package/dist/util/collision-detector.d.ts +77 -0
  221. package/dist/util/collision-detector.d.ts.map +1 -0
  222. package/dist/util/dates.d.ts +57 -0
  223. package/dist/util/dates.d.ts.map +1 -0
  224. package/dist/util/general.d.ts +3 -0
  225. package/dist/util/general.d.ts.map +1 -0
  226. package/dist/util/media.d.ts +9 -0
  227. package/dist/util/media.d.ts.map +1 -0
  228. package/dist/util/metadata.d.ts +138 -0
  229. package/dist/util/metadata.d.ts.map +1 -0
  230. package/dist/util/openai.d.ts +22 -0
  231. package/dist/util/openai.d.ts.map +1 -0
  232. package/dist/util/sounds-like-database.d.ts +98 -0
  233. package/dist/util/sounds-like-database.d.ts.map +1 -0
  234. package/dist/util/storage.d.ts +35 -0
  235. package/dist/util/storage.d.ts.map +1 -0
  236. package/dist/util/text-replacer.d.ts +56 -0
  237. package/dist/util/text-replacer.d.ts.map +1 -0
  238. package/dist/utils/entityFinder.d.ts +29 -0
  239. package/dist/utils/entityFinder.d.ts.map +1 -0
  240. package/package.json +84 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index36.js","sources":["../src/transcript/operations.ts"],"sourcesContent":["/**\n * Transcript Operations\n * \n * Core business logic for transcript parsing, listing, editing, and combining.\n * PKL-only implementation - all transcripts are stored in PKL format.\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'node:path';\nimport { glob } from 'glob';\nimport * as Context from '@redaksjon/context';\nimport * as Routing from '../routing';\nimport { Project } from '@redaksjon/context';\nimport { findProjectResilient } from '../utils/entityFinder';\nimport { \n PklTranscript, \n listTranscripts as listTranscriptsFromStorage,\n type TranscriptMetadata as PklMetadata,\n type ListTranscriptsOptions as StorageListOptions,\n} from '@redaksjon/protokoll-format';\nimport { ensurePklExtension } from './pkl-utils';\n\n/**\n * Parsed transcript structure\n */\nexport interface ParsedTranscript {\n filePath: string;\n title?: string;\n metadata: TranscriptMetadata;\n content: string;\n rawText: string;\n}\n\nexport interface TranscriptMetadata {\n date?: string;\n time?: string;\n project?: string;\n projectId?: string;\n destination?: string;\n confidence?: string;\n signals?: string[];\n reasoning?: string;\n tags?: string[];\n duration?: string;\n}\n\n/**\n * Check if input looks like a UUID (8+ hex chars)\n */\nexport function isUuidInput(input: string): boolean {\n return /^[a-f0-9]{8}/.test(input);\n}\n\n/**\n * Find transcript by UUID using glob scan\n * TODO: Replace with index-based lookup for better performance with large collections\n * \n * @param uuid - Full UUID or 8-character prefix\n * @param searchDirectories - Directories to search in\n * @returns Absolute path to transcript file, or null if not found\n */\nexport async function findTranscriptByUuid(\n uuid: string,\n searchDirectories: string[]\n): Promise<string | null> {\n const prefix = uuid.substring(0, 8); // Support both full UUID and prefix\n const pattern = `${prefix}-*.pkl`;\n \n for (const dir of searchDirectories) {\n const matches = await glob(pattern, { cwd: dir, absolute: true });\n if (matches.length > 0) {\n // Return first match - UUIDs should be unique\n return matches[0];\n }\n }\n \n return null;\n}\n\n/**\n * Parse a transcript file into its components\n * PKL-only implementation\n * \n * @param filePathOrUuid - File path or UUID to parse\n * @param searchDirectories - Optional directories to search if UUID is provided\n */\nexport const parseTranscript = async (\n filePathOrUuid: string,\n searchDirectories?: string[]\n): Promise<ParsedTranscript> => {\n let resolvedPath: string;\n \n // Check if input is a UUID\n if (isUuidInput(filePathOrUuid)) {\n if (!searchDirectories || searchDirectories.length === 0) {\n throw new Error('Search directories required for UUID lookup');\n }\n const foundPath = await findTranscriptByUuid(filePathOrUuid, searchDirectories);\n if (!foundPath) {\n throw new Error(`Transcript not found for UUID: ${filePathOrUuid}`);\n }\n resolvedPath = foundPath;\n } else {\n // Existing path-based logic\n resolvedPath = ensurePklExtension(filePathOrUuid);\n }\n \n const transcript = PklTranscript.open(resolvedPath, { readOnly: true });\n \n try {\n const pklMetadata = transcript.metadata;\n const content = transcript.content;\n \n const result: ParsedTranscript = {\n filePath: resolvedPath,\n title: pklMetadata.title,\n metadata: {\n date: pklMetadata.date instanceof Date \n ? pklMetadata.date.toISOString().split('T')[0] \n : undefined,\n time: pklMetadata.recordingTime,\n project: pklMetadata.project,\n projectId: pklMetadata.projectId,\n destination: pklMetadata.routing?.destination,\n confidence: pklMetadata.routing?.confidence?.toString(),\n signals: pklMetadata.routing?.signals,\n reasoning: pklMetadata.routing?.reasoning,\n tags: pklMetadata.tags,\n duration: pklMetadata.duration,\n },\n content,\n rawText: content, // For PKL files, content is the enhanced text\n };\n \n return result;\n } finally {\n transcript.close();\n }\n};\n\n/**\n * Extract the timestamp from a transcript filename\n */\nexport const extractTimestampFromFilename = (filePath: string): { day: number; hour: number; minute: number } | null => {\n const ext = path.extname(filePath);\n const basename = path.basename(filePath, ext);\n const match = basename.match(/^(\\d{1,2})-(\\d{2})(\\d{2})/);\n \n if (match) {\n return {\n day: parseInt(match[1], 10),\n hour: parseInt(match[2], 10),\n minute: parseInt(match[3], 10),\n };\n }\n \n return null;\n};\n\n/**\n * Slugify a title for use in filenames\n */\nexport const slugifyTitle = (title: string): string => {\n return title\n .toLowerCase()\n .replace(/[^a-z0-9]+/g, '-')\n .replace(/--+/g, '-')\n .replace(/^-|-$/g, '')\n .slice(0, 50);\n};\n\n/**\n * Parse duration string to seconds\n */\nconst parseDuration = (duration: string): number => {\n const match = duration.match(/(\\d+):(\\d+)/);\n if (match) {\n const [, minutes, seconds] = match;\n return parseInt(minutes, 10) * 60 + parseInt(seconds, 10);\n }\n return 0;\n};\n\n/**\n * Format seconds as duration string\n */\nconst formatDuration = (seconds: number): string => {\n const minutes = Math.floor(seconds / 60);\n const secs = seconds % 60;\n return `${minutes}:${secs.toString().padStart(2, '0')}`;\n};\n\n/**\n * Expand ~ in paths\n */\nconst expandPath = (p: string): string => {\n if (p.startsWith('~')) {\n return path.join(process.env.HOME || '', p.slice(1));\n }\n return p;\n};\n\n/**\n * Extract date from metadata\n */\nconst extractDateFromMetadata = (metadata: TranscriptMetadata, filePath: string): Date => {\n if (metadata.date) {\n return new Date(metadata.date);\n }\n const timestamp = extractTimestampFromFilename(filePath);\n if (timestamp) {\n const now = new Date();\n return new Date(now.getFullYear(), now.getMonth(), timestamp.day, timestamp.hour, timestamp.minute);\n }\n return new Date();\n};\n\n/**\n * Build routing config from context and project\n */\nconst buildRoutingConfig = (\n context: Context.ContextInstance,\n _targetProject: Project\n): Routing.RoutingConfig => {\n const config = context.getConfig();\n const defaultPath = expandPath((config.outputDirectory as string) || '~/notes');\n \n const resolveRoutingPath = (routingPath: string | undefined): string => {\n if (!routingPath) {\n return defaultPath;\n }\n const expanded = expandPath(routingPath);\n if (!expanded.startsWith('/') && !expanded.match(/^[A-Za-z]:/)) {\n return path.resolve(defaultPath, expanded);\n }\n return expanded;\n };\n\n return {\n default: {\n path: resolveRoutingPath(undefined),\n structure: 'month',\n filename_options: ['date', 'time', 'subject'],\n },\n projects: context.getAllProjects()\n .filter(p => p.active !== false)\n .map(p => ({\n projectId: p.id,\n destination: {\n path: resolveRoutingPath(p.routing?.destination),\n structure: p.routing?.structure || 'month',\n filename_options: p.routing?.filename_options || ['date', 'time', 'subject'],\n },\n classification: p.classification,\n active: p.active,\n })),\n conflict_resolution: 'primary' as const,\n };\n};\n\n/**\n * Combine multiple transcripts into a single document\n * PKL-only implementation\n */\nexport const combineTranscripts = async (\n filePaths: string[],\n options: {\n projectId?: string;\n title?: string;\n dryRun?: boolean;\n verbose?: boolean;\n contextDirectory?: string;\n /** Explicit context directories (from protokoll-config.yaml) */\n contextDirectories?: string[];\n } = {}\n): Promise<{ outputPath: string; content: string }> => {\n if (filePaths.length === 0) {\n throw new Error('No transcript files provided');\n }\n \n const transcripts: ParsedTranscript[] = [];\n for (const filePath of filePaths) {\n try {\n const parsed = await parseTranscript(filePath);\n transcripts.push(parsed);\n } catch (error) {\n throw new Error(`Failed to parse transcript: ${filePath} - ${error}`);\n }\n }\n \n transcripts.sort((a, b) => {\n const aName = path.basename(a.filePath);\n const bName = path.basename(b.filePath);\n return aName.localeCompare(bName);\n });\n \n const firstTranscript = transcripts[0];\n const baseMetadata = { ...firstTranscript.metadata };\n \n // Use explicit contextDirectories from options if provided (from protokoll-config.yaml)\n const context = await Context.create({\n startingDir: options.contextDirectory || path.dirname(firstTranscript.filePath),\n contextDirectories: options.contextDirectories,\n });\n let targetProject: Project | undefined;\n \n if (options.projectId) {\n targetProject = findProjectResilient(context, options.projectId);\n baseMetadata.project = targetProject.name;\n baseMetadata.projectId = targetProject.id;\n \n if (targetProject.routing?.destination) {\n const config = context.getConfig();\n const defaultPath = expandPath((config.outputDirectory as string) || '~/notes');\n const routingPath = expandPath(targetProject.routing.destination);\n const resolvedPath = !routingPath.startsWith('/') && !routingPath.match(/^[A-Za-z]:/)\n ? path.resolve(defaultPath, routingPath)\n : routingPath;\n baseMetadata.destination = resolvedPath;\n }\n }\n \n let totalSeconds = 0;\n let hasDuration = false;\n for (const t of transcripts) {\n if (t.metadata.duration) {\n hasDuration = true;\n totalSeconds += parseDuration(t.metadata.duration);\n }\n }\n if (hasDuration && totalSeconds > 0) {\n baseMetadata.duration = formatDuration(totalSeconds);\n }\n \n const allTags = new Set<string>();\n for (const t of transcripts) {\n if (t.metadata.tags) {\n for (const tag of t.metadata.tags) {\n allTags.add(tag);\n }\n }\n }\n if (allTags.size > 0) {\n baseMetadata.tags = Array.from(allTags).sort();\n }\n \n const combinedTitle = options.title \n ? options.title\n : (firstTranscript.title \n ? `${firstTranscript.title} (Combined)`\n : 'Combined Transcript');\n \n // Build combined content\n const contentParts: string[] = [];\n for (let i = 0; i < transcripts.length; i++) {\n const t = transcripts[i];\n const sectionTitle = t.title || `Part ${i + 1}`;\n const sourceFile = path.basename(t.filePath);\n \n contentParts.push(`## ${sectionTitle}`);\n contentParts.push(`*Source: ${sourceFile}*`);\n contentParts.push('');\n contentParts.push(t.content);\n contentParts.push('');\n }\n \n const combinedContent = contentParts.join('\\n');\n \n // Determine output path\n let outputPath: string;\n \n if (targetProject?.routing?.destination) {\n const routingConfig = buildRoutingConfig(context, targetProject);\n const routing = Routing.create(routingConfig, context);\n \n const audioDate = extractDateFromMetadata(baseMetadata, firstTranscript.filePath);\n \n const routingContext: Routing.RoutingContext = {\n transcriptText: combinedContent,\n audioDate,\n sourceFile: firstTranscript.filePath,\n };\n \n const decision = routing.route(routingContext);\n outputPath = routing.buildOutputPath(decision, routingContext);\n // Ensure .pkl extension\n outputPath = outputPath.replace(/\\.md$/, '.pkl');\n } else {\n const firstDir = path.dirname(firstTranscript.filePath);\n const timestamp = extractTimestampFromFilename(firstTranscript.filePath);\n \n const filenameSuffix = options.title \n ? slugifyTitle(options.title)\n : 'combined';\n \n if (timestamp) {\n const day = timestamp.day.toString().padStart(2, '0');\n const hour = timestamp.hour.toString().padStart(2, '0');\n const minute = timestamp.minute.toString().padStart(2, '0');\n outputPath = path.join(firstDir, `${day}-${hour}${minute}-${filenameSuffix}.pkl`);\n } else {\n outputPath = path.join(firstDir, `${filenameSuffix}.pkl`);\n }\n }\n \n // Create the combined PKL transcript\n if (!options.dryRun) {\n const initialMetadata: PklMetadata = {\n id: '', // Will be auto-generated by PklTranscript.create()\n title: combinedTitle,\n date: baseMetadata.date ? new Date(baseMetadata.date) : undefined,\n recordingTime: baseMetadata.time,\n project: targetProject?.name || baseMetadata.project,\n projectId: targetProject?.id || baseMetadata.projectId,\n tags: baseMetadata.tags || [],\n duration: baseMetadata.duration,\n status: 'reviewed',\n };\n \n if (targetProject) {\n initialMetadata.entities = {\n people: [],\n projects: [{\n id: targetProject.id,\n name: targetProject.name,\n type: 'project',\n }],\n terms: [],\n companies: [],\n };\n }\n \n const newTranscript = PklTranscript.create(outputPath, initialMetadata);\n try {\n newTranscript.updateContent(combinedContent);\n } finally {\n newTranscript.close();\n }\n }\n \n return { outputPath, content: combinedContent };\n};\n\n/**\n * Edit transcript metadata and content\n * PKL-only implementation\n */\nexport const editTranscript = async (\n filePath: string,\n options: {\n title?: string;\n projectId?: string;\n tagsToAdd?: string[];\n tagsToRemove?: string[];\n dryRun?: boolean;\n verbose?: boolean;\n contextDirectory?: string;\n /** Explicit context directories (from protokoll-config.yaml) */\n contextDirectories?: string[];\n }\n): Promise<{ outputPath: string; content: string }> => {\n const pklPath = ensurePklExtension(filePath);\n const transcript = PklTranscript.open(pklPath, { readOnly: false });\n \n try {\n const pklMetadata = transcript.metadata;\n const content = transcript.content;\n \n // Use explicit contextDirectories from options if provided (from protokoll-config.yaml)\n const context = await Context.create({\n startingDir: options.contextDirectory || path.dirname(pklPath),\n contextDirectories: options.contextDirectories,\n });\n let targetProject: Project | undefined;\n \n if (options.projectId) {\n targetProject = findProjectResilient(context, options.projectId);\n }\n \n const newTitle = options.title || pklMetadata.title || 'Untitled';\n \n // Build updated metadata\n const updatedMetadata: Partial<PklMetadata> = {};\n \n if (options.title) {\n updatedMetadata.title = newTitle;\n }\n \n if (targetProject) {\n updatedMetadata.project = targetProject.name;\n updatedMetadata.projectId = targetProject.id;\n \n // Update entities with the project\n const existingEntities = pklMetadata.entities || { people: [], projects: [], terms: [], companies: [] };\n updatedMetadata.entities = {\n people: existingEntities.people || [],\n projects: [{\n id: targetProject.id,\n name: targetProject.name,\n type: 'project',\n }],\n terms: existingEntities.terms || [],\n companies: existingEntities.companies || [],\n };\n }\n \n // Handle tag updates\n if (options.tagsToAdd || options.tagsToRemove) {\n const currentTags = new Set(pklMetadata.tags || []);\n \n if (options.tagsToRemove) {\n for (const tag of options.tagsToRemove) {\n currentTags.delete(tag);\n }\n }\n \n if (options.tagsToAdd) {\n for (const tag of options.tagsToAdd) {\n currentTags.add(tag);\n }\n }\n \n updatedMetadata.tags = Array.from(currentTags).sort();\n }\n \n // Determine output path\n let outputPath = pklPath;\n \n if (targetProject?.routing?.destination || options.title) {\n const config = context.getConfig();\n const defaultPath = expandPath((config.outputDirectory as string) || '~/notes');\n \n if (targetProject?.routing?.destination) {\n const routingConfig = buildRoutingConfig(context, targetProject);\n const routing = Routing.create(routingConfig, context);\n \n const audioDate = pklMetadata.date instanceof Date ? pklMetadata.date : new Date();\n \n const routingContext: Routing.RoutingContext = {\n transcriptText: content,\n audioDate,\n sourceFile: pklPath,\n };\n \n const decision = routing.route(routingContext);\n \n if (options.title) {\n const basePath = path.dirname(routing.buildOutputPath(decision, routingContext));\n const timestamp = extractTimestampFromFilename(pklPath);\n const sluggedTitle = slugifyTitle(options.title);\n \n if (timestamp) {\n const day = timestamp.day.toString().padStart(2, '0');\n const hour = timestamp.hour.toString().padStart(2, '0');\n const minute = timestamp.minute.toString().padStart(2, '0');\n outputPath = path.join(basePath, `${day}-${hour}${minute}-${sluggedTitle}.pkl`);\n } else {\n outputPath = path.join(basePath, `${sluggedTitle}.pkl`);\n }\n } else {\n outputPath = routing.buildOutputPath(decision, routingContext);\n outputPath = outputPath.replace(/\\.md$/, '.pkl');\n }\n } else if (options.title) {\n const dir = path.dirname(pklPath);\n const timestamp = extractTimestampFromFilename(pklPath);\n const sluggedTitle = slugifyTitle(options.title);\n \n if (timestamp) {\n const day = timestamp.day.toString().padStart(2, '0');\n const hour = timestamp.hour.toString().padStart(2, '0');\n const minute = timestamp.minute.toString().padStart(2, '0');\n outputPath = path.join(dir, `${day}-${hour}${minute}-${sluggedTitle}.pkl`);\n } else {\n outputPath = path.join(dir, `${sluggedTitle}.pkl`);\n }\n }\n }\n \n // Apply updates\n if (!options.dryRun) {\n if (Object.keys(updatedMetadata).length > 0) {\n transcript.updateMetadata(updatedMetadata);\n }\n \n // If output path changed, we need to move the file\n if (outputPath !== pklPath) {\n // Close current transcript\n transcript.close();\n \n // Create directory if needed\n await fs.mkdir(path.dirname(outputPath), { recursive: true });\n \n // Copy to new location\n await fs.copyFile(pklPath, outputPath);\n \n // Delete old file\n await fs.unlink(pklPath);\n }\n }\n \n return { outputPath, content };\n } finally {\n // Only close if not already closed (due to move operation)\n try {\n transcript.close();\n } catch {\n // Already closed\n }\n }\n};\n\n/**\n * Transcript list item\n */\nexport interface TranscriptListItem {\n path: string;\n filename: string;\n uuid: string; // UUID identifier for this transcript\n date: string;\n time?: string;\n title: string;\n hasRawTranscript: boolean;\n createdAt: Date;\n status?: 'initial' | 'enhanced' | 'reviewed' | 'in_progress' | 'closed' | 'archived';\n openTasksCount?: number;\n contentSize?: number;\n entities?: {\n people?: Array<{ id: string; name: string }>;\n projects?: Array<{ id: string; name: string }>;\n terms?: Array<{ id: string; name: string }>;\n companies?: Array<{ id: string; name: string }>;\n };\n}\n\nexport interface ListTranscriptsOptions {\n directory: string;\n limit?: number;\n offset?: number;\n sortBy?: 'date' | 'filename' | 'title';\n startDate?: string;\n endDate?: string;\n search?: string;\n projectId?: string;\n}\n\nexport interface ListTranscriptsResult {\n transcripts: TranscriptListItem[];\n total: number;\n hasMore: boolean;\n limit: number;\n offset: number;\n}\n\n/**\n * List transcripts with filtering and pagination\n * Uses the protokoll-format storage API\n */\nexport const listTranscripts = async (options: ListTranscriptsOptions): Promise<ListTranscriptsResult> => {\n const {\n directory,\n limit = 50,\n offset = 0,\n sortBy = 'date',\n startDate,\n endDate,\n search,\n projectId,\n } = options;\n \n // Use the storage API from protokoll-format\n const storageOptions: StorageListOptions = {\n directory,\n limit,\n offset,\n sortBy,\n search,\n project: projectId,\n startDate,\n endDate,\n };\n \n const result = await listTranscriptsFromStorage(storageOptions);\n \n // Convert storage result to operations result format\n const transcripts: TranscriptListItem[] = result.transcripts.map(item => {\n // Extract UUID from the transcript file\n let uuid = '';\n try {\n const transcript = PklTranscript.open(item.filePath, { readOnly: true });\n uuid = transcript.metadata.id;\n transcript.close();\n } catch {\n // If we can't open the file, leave uuid empty\n uuid = '';\n }\n \n return {\n path: item.filePath,\n filename: path.basename(item.filePath),\n uuid,\n date: item.date instanceof Date ? item.date.toISOString().split('T')[0] : '',\n time: undefined, // Not in storage result\n title: item.title,\n hasRawTranscript: false, // Not in storage result, would need to open file to check\n createdAt: item.date || new Date(),\n status: item.status,\n openTasksCount: undefined, // Not in storage result\n contentSize: item.contentPreview?.length,\n entities: item.project ? {\n projects: [{\n id: item.project,\n name: item.project,\n }],\n } : undefined,\n };\n });\n \n return {\n transcripts,\n total: result.total,\n hasMore: result.hasMore,\n limit,\n offset,\n };\n};\n"],"names":["Routing.create","listTranscriptsFromStorage"],"mappings":";;;;;;;;;AAiDO,SAAS,YAAY,KAAA,EAAwB;AAChD,EAAA,OAAO,cAAA,CAAe,KAAK,KAAK,CAAA;AACpC;AAUA,eAAsB,oBAAA,CAClB,MACA,iBAAA,EACsB;AACtB,EAAA,MAAM,MAAA,GAAS,IAAA,CAAK,SAAA,CAAU,CAAA,EAAG,CAAC,CAAA;AAClC,EAAA,MAAM,OAAA,GAAU,GAAG,MAAM,CAAA,MAAA,CAAA;AAEzB,EAAA,KAAA,MAAW,OAAO,iBAAA,EAAmB;AACjC,IAAA,MAAM,OAAA,GAAU,MAAM,IAAA,CAAK,OAAA,EAAS,EAAE,GAAA,EAAK,GAAA,EAAK,QAAA,EAAU,IAAA,EAAM,CAAA;AAChE,IAAA,IAAI,OAAA,CAAQ,SAAS,CAAA,EAAG;AAEpB,MAAA,OAAO,QAAQ,CAAC,CAAA;AAAA,IACpB;AAAA,EACJ;AAEA,EAAA,OAAO,IAAA;AACX;AASO,MAAM,eAAA,GAAkB,OAC3B,cAAA,EACA,iBAAA,KAC4B;AAC5B,EAAA,IAAI,YAAA;AAGJ,EAAA,IAAI,WAAA,CAAY,cAAc,CAAA,EAAG;AAC7B,IAAA,IAAI,CAAC,iBAAA,IAAqB,iBAAA,CAAkB,MAAA,KAAW,CAAA,EAAG;AACtD,MAAA,MAAM,IAAI,MAAM,6CAA6C,CAAA;AAAA,IACjE;AACA,IAAA,MAAM,SAAA,GAAY,MAAM,oBAAA,CAAqB,cAAA,EAAgB,iBAAiB,CAAA;AAC9E,IAAA,IAAI,CAAC,SAAA,EAAW;AACZ,MAAA,MAAM,IAAI,KAAA,CAAM,CAAA,+BAAA,EAAkC,cAAc,CAAA,CAAE,CAAA;AAAA,IACtE;AACA,IAAA,YAAA,GAAe,SAAA;AAAA,EACnB,CAAA,MAAO;AAEH,IAAA,YAAA,GAAe,mBAAmB,cAAc,CAAA;AAAA,EACpD;AAEA,EAAA,MAAM,aAAa,aAAA,CAAc,IAAA,CAAK,cAAc,EAAE,QAAA,EAAU,MAAM,CAAA;AAEtE,EAAA,IAAI;AACA,IAAA,MAAM,cAAc,UAAA,CAAW,QAAA;AAC/B,IAAA,MAAM,UAAU,UAAA,CAAW,OAAA;AAE3B,IAAA,MAAM,MAAA,GAA2B;AAAA,MAC7B,QAAA,EAAU,YAAA;AAAA,MACV,OAAO,WAAA,CAAY,KAAA;AAAA,MACnB,QAAA,EAAU;AAAA,QACN,IAAA,EAAM,WAAA,CAAY,IAAA,YAAgB,IAAA,GAC5B,WAAA,CAAY,IAAA,CAAK,WAAA,EAAY,CAAE,KAAA,CAAM,GAAG,CAAA,CAAE,CAAC,CAAA,GAC3C,KAAA,CAAA;AAAA,QACN,MAAM,WAAA,CAAY,aAAA;AAAA,QAClB,SAAS,WAAA,CAAY,OAAA;AAAA,QACrB,WAAW,WAAA,CAAY,SAAA;AAAA,QACvB,WAAA,EAAa,YAAY,OAAA,EAAS,WAAA;AAAA,QAClC,UAAA,EAAY,WAAA,CAAY,OAAA,EAAS,UAAA,EAAY,QAAA,EAAS;AAAA,QACtD,OAAA,EAAS,YAAY,OAAA,EAAS,OAAA;AAAA,QAC9B,SAAA,EAAW,YAAY,OAAA,EAAS,SAAA;AAAA,QAChC,MAAM,WAAA,CAAY,IAAA;AAAA,QAClB,UAAU,WAAA,CAAY;AAAA,OAC1B;AAAA,MACA,OAAA;AAAA,MACA,OAAA,EAAS;AAAA;AAAA,KACb;AAEA,IAAA,OAAO,MAAA;AAAA,EACX,CAAA,SAAE;AACE,IAAA,UAAA,CAAW,KAAA,EAAM;AAAA,EACrB;AACJ;AAKO,MAAM,4BAAA,GAA+B,CAAC,QAAA,KAA2E;AACpH,EAAA,MAAM,GAAA,GAAM,IAAA,CAAK,OAAA,CAAQ,QAAQ,CAAA;AACjC,EAAA,MAAM,QAAA,GAAW,IAAA,CAAK,QAAA,CAAS,QAAA,EAAU,GAAG,CAAA;AAC5C,EAAA,MAAM,KAAA,GAAQ,QAAA,CAAS,KAAA,CAAM,2BAA2B,CAAA;AAExD,EAAA,IAAI,KAAA,EAAO;AACP,IAAA,OAAO;AAAA,MACH,GAAA,EAAK,QAAA,CAAS,KAAA,CAAM,CAAC,GAAG,EAAE,CAAA;AAAA,MAC1B,IAAA,EAAM,QAAA,CAAS,KAAA,CAAM,CAAC,GAAG,EAAE,CAAA;AAAA,MAC3B,MAAA,EAAQ,QAAA,CAAS,KAAA,CAAM,CAAC,GAAG,EAAE;AAAA,KACjC;AAAA,EACJ;AAEA,EAAA,OAAO,IAAA;AACX;AAKO,MAAM,YAAA,GAAe,CAAC,KAAA,KAA0B;AACnD,EAAA,OAAO,MACF,WAAA,EAAY,CACZ,OAAA,CAAQ,aAAA,EAAe,GAAG,CAAA,CAC1B,OAAA,CAAQ,MAAA,EAAQ,GAAG,EACnB,OAAA,CAAQ,QAAA,EAAU,EAAE,CAAA,CACpB,KAAA,CAAM,GAAG,EAAE,CAAA;AACpB;AAKA,MAAM,aAAA,GAAgB,CAAC,QAAA,KAA6B;AAChD,EAAA,MAAM,KAAA,GAAQ,QAAA,CAAS,KAAA,CAAM,aAAa,CAAA;AAC1C,EAAA,IAAI,KAAA,EAAO;AACP,IAAA,MAAM,GAAG,OAAA,EAAS,OAAO,CAAA,GAAI,KAAA;AAC7B,IAAA,OAAO,SAAS,OAAA,EAAS,EAAE,IAAI,EAAA,GAAK,QAAA,CAAS,SAAS,EAAE,CAAA;AAAA,EAC5D;AACA,EAAA,OAAO,CAAA;AACX,CAAA;AAKA,MAAM,cAAA,GAAiB,CAAC,OAAA,KAA4B;AAChD,EAAA,MAAM,OAAA,GAAU,IAAA,CAAK,KAAA,CAAM,OAAA,GAAU,EAAE,CAAA;AACvC,EAAA,MAAM,OAAO,OAAA,GAAU,EAAA;AACvB,EAAA,OAAO,CAAA,EAAG,OAAO,CAAA,CAAA,EAAI,IAAA,CAAK,UAAS,CAAE,QAAA,CAAS,CAAA,EAAG,GAAG,CAAC,CAAA,CAAA;AACzD,CAAA;AAKA,MAAM,UAAA,GAAa,CAAC,CAAA,KAAsB;AACtC,EAAA,IAAI,CAAA,CAAE,UAAA,CAAW,GAAG,CAAA,EAAG;AACnB,IAAA,OAAO,IAAA,CAAK,KAAK,OAAA,CAAQ,GAAA,CAAI,QAAQ,EAAA,EAAI,CAAA,CAAE,KAAA,CAAM,CAAC,CAAC,CAAA;AAAA,EACvD;AACA,EAAA,OAAO,CAAA;AACX,CAAA;AAKA,MAAM,uBAAA,GAA0B,CAAC,QAAA,EAA8B,QAAA,KAA2B;AACtF,EAAA,IAAI,SAAS,IAAA,EAAM;AACf,IAAA,OAAO,IAAI,IAAA,CAAK,QAAA,CAAS,IAAI,CAAA;AAAA,EACjC;AACA,EAAA,MAAM,SAAA,GAAY,6BAA6B,QAAQ,CAAA;AACvD,EAAA,IAAI,SAAA,EAAW;AACX,IAAA,MAAM,GAAA,uBAAU,IAAA,EAAK;AACrB,IAAA,OAAO,IAAI,IAAA,CAAK,GAAA,CAAI,WAAA,EAAY,EAAG,GAAA,CAAI,QAAA,EAAS,EAAG,SAAA,CAAU,GAAA,EAAK,SAAA,CAAU,IAAA,EAAM,UAAU,MAAM,CAAA;AAAA,EACtG;AACA,EAAA,2BAAW,IAAA,EAAK;AACpB,CAAA;AAKA,MAAM,kBAAA,GAAqB,CACvB,OAAA,EACA,cAAA,KACwB;AACxB,EAAA,MAAM,MAAA,GAAS,QAAQ,SAAA,EAAU;AACjC,EAAA,MAAM,WAAA,GAAc,UAAA,CAAY,MAAA,CAAO,eAAA,IAA8B,SAAS,CAAA;AAE9E,EAAA,MAAM,kBAAA,GAAqB,CAAC,WAAA,KAA4C;AACpE,IAAA,IAAI,CAAC,WAAA,EAAa;AACd,MAAA,OAAO,WAAA;AAAA,IACX;AACA,IAAA,MAAM,QAAA,GAAW,WAAW,WAAW,CAAA;AACvC,IAAA,IAAI,CAAC,SAAS,UAAA,CAAW,GAAG,KAAK,CAAC,QAAA,CAAS,KAAA,CAAM,YAAY,CAAA,EAAG;AAC5D,MAAA,OAAO,IAAA,CAAK,OAAA,CAAQ,WAAA,EAAa,QAAQ,CAAA;AAAA,IAC7C;AACA,IAAA,OAAO,QAAA;AAAA,EACX,CAAA;AAEA,EAAA,OAAO;AAAA,IACH,OAAA,EAAS;AAAA,MACL,IAAA,EAAM,mBAAmB,MAAS,CAAA;AAAA,MAClC,SAAA,EAAW,OAAA;AAAA,MACX,gBAAA,EAAkB,CAAC,MAAA,EAAQ,MAAA,EAAQ,SAAS;AAAA,KAChD;AAAA,IACA,QAAA,EAAU,OAAA,CAAQ,cAAA,EAAe,CAC5B,MAAA,CAAO,CAAA,CAAA,KAAK,CAAA,CAAE,MAAA,KAAW,KAAK,CAAA,CAC9B,GAAA,CAAI,CAAA,CAAA,MAAM;AAAA,MACP,WAAW,CAAA,CAAE,EAAA;AAAA,MACb,WAAA,EAAa;AAAA,QACT,IAAA,EAAM,kBAAA,CAAmB,CAAA,CAAE,OAAA,EAAS,WAAW,CAAA;AAAA,QAC/C,SAAA,EAAW,CAAA,CAAE,OAAA,EAAS,SAAA,IAAa,OAAA;AAAA,QACnC,kBAAkB,CAAA,CAAE,OAAA,EAAS,oBAAoB,CAAC,MAAA,EAAQ,QAAQ,SAAS;AAAA,OAC/E;AAAA,MACA,gBAAgB,CAAA,CAAE,cAAA;AAAA,MAClB,QAAQ,CAAA,CAAE;AAAA,KACd,CAAE,CAAA;AAAA,IACN,mBAAA,EAAqB;AAAA,GACzB;AACJ,CAAA;AAMO,MAAM,kBAAA,GAAqB,OAC9B,SAAA,EACA,OAAA,GAQI,EAAC,KAC8C;AACnD,EAAA,IAAI,SAAA,CAAU,WAAW,CAAA,EAAG;AACxB,IAAA,MAAM,IAAI,MAAM,8BAA8B,CAAA;AAAA,EAClD;AAEA,EAAA,MAAM,cAAkC,EAAC;AACzC,EAAA,KAAA,MAAW,YAAY,SAAA,EAAW;AAC9B,IAAA,IAAI;AACA,MAAA,MAAM,MAAA,GAAS,MAAM,eAAA,CAAgB,QAAQ,CAAA;AAC7C,MAAA,WAAA,CAAY,KAAK,MAAM,CAAA;AAAA,IAC3B,SAAS,KAAA,EAAO;AACZ,MAAA,MAAM,IAAI,KAAA,CAAM,CAAA,4BAAA,EAA+B,QAAQ,CAAA,GAAA,EAAM,KAAK,CAAA,CAAE,CAAA;AAAA,IACxE;AAAA,EACJ;AAEA,EAAA,WAAA,CAAY,IAAA,CAAK,CAAC,CAAA,EAAG,CAAA,KAAM;AACvB,IAAA,MAAM,KAAA,GAAQ,IAAA,CAAK,QAAA,CAAS,CAAA,CAAE,QAAQ,CAAA;AACtC,IAAA,MAAM,KAAA,GAAQ,IAAA,CAAK,QAAA,CAAS,CAAA,CAAE,QAAQ,CAAA;AACtC,IAAA,OAAO,KAAA,CAAM,cAAc,KAAK,CAAA;AAAA,EACpC,CAAC,CAAA;AAED,EAAA,MAAM,eAAA,GAAkB,YAAY,CAAC,CAAA;AACrC,EAAA,MAAM,YAAA,GAAe,EAAE,GAAG,eAAA,CAAgB,QAAA,EAAS;AAGnD,EAAA,MAAM,OAAA,GAAU,MAAM,OAAA,CAAQ,MAAA,CAAO;AAAA,IACjC,aAAa,OAAA,CAAQ,gBAAA,IAAoB,IAAA,CAAK,OAAA,CAAQ,gBAAgB,QAAQ,CAAA;AAAA,IAC9E,oBAAoB,OAAA,CAAQ;AAAA,GAC/B,CAAA;AACD,EAAA,IAAI,aAAA;AAEJ,EAAA,IAAI,QAAQ,SAAA,EAAW;AACnB,IAAA,aAAA,GAAgB,oBAAA,CAAqB,OAAA,EAAS,OAAA,CAAQ,SAAS,CAAA;AAC/D,IAAA,YAAA,CAAa,UAAU,aAAA,CAAc,IAAA;AACrC,IAAA,YAAA,CAAa,YAAY,aAAA,CAAc,EAAA;AAEvC,IAAA,IAAI,aAAA,CAAc,SAAS,WAAA,EAAa;AACpC,MAAA,MAAM,MAAA,GAAS,QAAQ,SAAA,EAAU;AACjC,MAAA,MAAM,WAAA,GAAc,UAAA,CAAY,MAAA,CAAO,eAAA,IAA8B,SAAS,CAAA;AAC9E,MAAA,MAAM,WAAA,GAAc,UAAA,CAAW,aAAA,CAAc,OAAA,CAAQ,WAAW,CAAA;AAChE,MAAA,MAAM,YAAA,GAAe,CAAC,WAAA,CAAY,UAAA,CAAW,GAAG,CAAA,IAAK,CAAC,WAAA,CAAY,KAAA,CAAM,YAAY,CAAA,GAC9E,IAAA,CAAK,OAAA,CAAQ,WAAA,EAAa,WAAW,CAAA,GACrC,WAAA;AACN,MAAA,YAAA,CAAa,WAAA,GAAc,YAAA;AAAA,IAC/B;AAAA,EACJ;AAEA,EAAA,IAAI,YAAA,GAAe,CAAA;AACnB,EAAA,IAAI,WAAA,GAAc,KAAA;AAClB,EAAA,KAAA,MAAW,KAAK,WAAA,EAAa;AACzB,IAAA,IAAI,CAAA,CAAE,SAAS,QAAA,EAAU;AACrB,MAAA,WAAA,GAAc,IAAA;AACd,MAAA,YAAA,IAAgB,aAAA,CAAc,CAAA,CAAE,QAAA,CAAS,QAAQ,CAAA;AAAA,IACrD;AAAA,EACJ;AACA,EAAA,IAAI,WAAA,IAAe,eAAe,CAAA,EAAG;AACjC,IAAA,YAAA,CAAa,QAAA,GAAW,eAAe,YAAY,CAAA;AAAA,EACvD;AAEA,EAAA,MAAM,OAAA,uBAAc,GAAA,EAAY;AAChC,EAAA,KAAA,MAAW,KAAK,WAAA,EAAa;AACzB,IAAA,IAAI,CAAA,CAAE,SAAS,IAAA,EAAM;AACjB,MAAA,KAAA,MAAW,GAAA,IAAO,CAAA,CAAE,QAAA,CAAS,IAAA,EAAM;AAC/B,QAAA,OAAA,CAAQ,IAAI,GAAG,CAAA;AAAA,MACnB;AAAA,IACJ;AAAA,EACJ;AACA,EAAA,IAAI,OAAA,CAAQ,OAAO,CAAA,EAAG;AAClB,IAAA,YAAA,CAAa,IAAA,GAAO,KAAA,CAAM,IAAA,CAAK,OAAO,EAAE,IAAA,EAAK;AAAA,EACjD;AAEA,EAAA,MAAM,aAAA,GAAgB,OAAA,CAAQ,KAAA,GACxB,OAAA,CAAQ,KAAA,GACP,gBAAgB,KAAA,GACb,CAAA,EAAG,eAAA,CAAgB,KAAK,CAAA,WAAA,CAAA,GACxB,qBAAA;AAGV,EAAA,MAAM,eAAyB,EAAC;AAChC,EAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,WAAA,CAAY,QAAQ,CAAA,EAAA,EAAK;AACzC,IAAA,MAAM,CAAA,GAAI,YAAY,CAAC,CAAA;AACvB,IAAA,MAAM,YAAA,GAAe,CAAA,CAAE,KAAA,IAAS,CAAA,KAAA,EAAQ,IAAI,CAAC,CAAA,CAAA;AAC7C,IAAA,MAAM,UAAA,GAAa,IAAA,CAAK,QAAA,CAAS,CAAA,CAAE,QAAQ,CAAA;AAE3C,IAAA,YAAA,CAAa,IAAA,CAAK,CAAA,GAAA,EAAM,YAAY,CAAA,CAAE,CAAA;AACtC,IAAA,YAAA,CAAa,IAAA,CAAK,CAAA,SAAA,EAAY,UAAU,CAAA,CAAA,CAAG,CAAA;AAC3C,IAAA,YAAA,CAAa,KAAK,EAAE,CAAA;AACpB,IAAA,YAAA,CAAa,IAAA,CAAK,EAAE,OAAO,CAAA;AAC3B,IAAA,YAAA,CAAa,KAAK,EAAE,CAAA;AAAA,EACxB;AAEA,EAAA,MAAM,eAAA,GAAkB,YAAA,CAAa,IAAA,CAAK,IAAI,CAAA;AAG9C,EAAA,IAAI,UAAA;AAEJ,EAAA,IAAI,aAAA,EAAe,SAAS,WAAA,EAAa;AACrC,IAAA,MAAM,aAAA,GAAgB,kBAAA,CAAmB,OAAsB,CAAA;AAC/D,IAAA,MAAM,OAAA,GAAUA,MAAQ,CAAO,aAAA,EAAe,OAAO,CAAA;AAErD,IAAA,MAAM,SAAA,GAAY,uBAAA,CAAwB,YAAA,EAAc,eAAA,CAAgB,QAAQ,CAAA;AAEhF,IAAA,MAAM,cAAA,GAAyC;AAAA,MAC3C,cAAA,EAAgB,eAAA;AAAA,MAChB,SAAA;AAAA,MACA,YAAY,eAAA,CAAgB;AAAA,KAChC;AAEA,IAAA,MAAM,QAAA,GAAW,OAAA,CAAQ,KAAA,CAAM,cAAc,CAAA;AAC7C,IAAA,UAAA,GAAa,OAAA,CAAQ,eAAA,CAAgB,QAAA,EAAU,cAAc,CAAA;AAE7D,IAAA,UAAA,GAAa,UAAA,CAAW,OAAA,CAAQ,OAAA,EAAS,MAAM,CAAA;AAAA,EACnD,CAAA,MAAO;AACH,IAAA,MAAM,QAAA,GAAW,IAAA,CAAK,OAAA,CAAQ,eAAA,CAAgB,QAAQ,CAAA;AACtD,IAAA,MAAM,SAAA,GAAY,4BAAA,CAA6B,eAAA,CAAgB,QAAQ,CAAA;AAEvE,IAAA,MAAM,iBAAiB,OAAA,CAAQ,KAAA,GACzB,YAAA,CAAa,OAAA,CAAQ,KAAK,CAAA,GAC1B,UAAA;AAEN,IAAA,IAAI,SAAA,EAAW;AACX,MAAA,MAAM,MAAM,SAAA,CAAU,GAAA,CAAI,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACpD,MAAA,MAAM,OAAO,SAAA,CAAU,IAAA,CAAK,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACtD,MAAA,MAAM,SAAS,SAAA,CAAU,MAAA,CAAO,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AAC1D,MAAA,UAAA,GAAa,IAAA,CAAK,IAAA,CAAK,QAAA,EAAU,CAAA,EAAG,GAAG,CAAA,CAAA,EAAI,IAAI,CAAA,EAAG,MAAM,CAAA,CAAA,EAAI,cAAc,CAAA,IAAA,CAAM,CAAA;AAAA,IACpF,CAAA,MAAO;AACH,MAAA,UAAA,GAAa,IAAA,CAAK,IAAA,CAAK,QAAA,EAAU,CAAA,EAAG,cAAc,CAAA,IAAA,CAAM,CAAA;AAAA,IAC5D;AAAA,EACJ;AAGA,EAAA,IAAI,CAAC,QAAQ,MAAA,EAAQ;AACjB,IAAA,MAAM,eAAA,GAA+B;AAAA,MACjC,EAAA,EAAI,EAAA;AAAA;AAAA,MACJ,KAAA,EAAO,aAAA;AAAA,MACP,MAAM,YAAA,CAAa,IAAA,GAAO,IAAI,IAAA,CAAK,YAAA,CAAa,IAAI,CAAA,GAAI,MAAA;AAAA,MACxD,eAAe,YAAA,CAAa,IAAA;AAAA,MAC5B,OAAA,EAAS,aAAA,EAAe,IAAA,IAAQ,YAAA,CAAa,OAAA;AAAA,MAC7C,SAAA,EAAW,aAAA,EAAe,EAAA,IAAM,YAAA,CAAa,SAAA;AAAA,MAC7C,IAAA,EAAM,YAAA,CAAa,IAAA,IAAQ,EAAC;AAAA,MAC5B,UAAU,YAAA,CAAa,QAAA;AAAA,MACvB,MAAA,EAAQ;AAAA,KACZ;AAEA,IAAA,IAAI,aAAA,EAAe;AACf,MAAA,eAAA,CAAgB,QAAA,GAAW;AAAA,QACvB,QAAQ,EAAC;AAAA,QACT,UAAU,CAAC;AAAA,UACP,IAAI,aAAA,CAAc,EAAA;AAAA,UAClB,MAAM,aAAA,CAAc,IAAA;AAAA,UACpB,IAAA,EAAM;AAAA,SACT,CAAA;AAAA,QACD,OAAO,EAAC;AAAA,QACR,WAAW;AAAC,OAChB;AAAA,IACJ;AAEA,IAAA,MAAM,aAAA,GAAgB,aAAA,CAAc,MAAA,CAAO,UAAA,EAAY,eAAe,CAAA;AACtE,IAAA,IAAI;AACA,MAAA,aAAA,CAAc,cAAc,eAAe,CAAA;AAAA,IAC/C,CAAA,SAAE;AACE,MAAA,aAAA,CAAc,KAAA,EAAM;AAAA,IACxB;AAAA,EACJ;AAEA,EAAA,OAAO,EAAE,UAAA,EAAY,OAAA,EAAS,eAAA,EAAgB;AAClD;AAMO,MAAM,cAAA,GAAiB,OAC1B,QAAA,EACA,OAAA,KAWmD;AACnD,EAAA,MAAM,OAAA,GAAU,mBAAmB,QAAQ,CAAA;AAC3C,EAAA,MAAM,aAAa,aAAA,CAAc,IAAA,CAAK,SAAS,EAAE,QAAA,EAAU,OAAO,CAAA;AAElE,EAAA,IAAI;AACA,IAAA,MAAM,cAAc,UAAA,CAAW,QAAA;AAC/B,IAAA,MAAM,UAAU,UAAA,CAAW,OAAA;AAG3B,IAAA,MAAM,OAAA,GAAU,MAAM,OAAA,CAAQ,MAAA,CAAO;AAAA,MACjC,WAAA,EAAa,OAAA,CAAQ,gBAAA,IAAoB,IAAA,CAAK,QAAQ,OAAO,CAAA;AAAA,MAC7D,oBAAoB,OAAA,CAAQ;AAAA,KAC/B,CAAA;AACD,IAAA,IAAI,aAAA;AAEJ,IAAA,IAAI,QAAQ,SAAA,EAAW;AACnB,MAAA,aAAA,GAAgB,oBAAA,CAAqB,OAAA,EAAS,OAAA,CAAQ,SAAS,CAAA;AAAA,IACnE;AAEA,IAAA,MAAM,QAAA,GAAW,OAAA,CAAQ,KAAA,IAAS,WAAA,CAAY,KAAA,IAAS,UAAA;AAGvD,IAAA,MAAM,kBAAwC,EAAC;AAE/C,IAAA,IAAI,QAAQ,KAAA,EAAO;AACf,MAAA,eAAA,CAAgB,KAAA,GAAQ,QAAA;AAAA,IAC5B;AAEA,IAAA,IAAI,aAAA,EAAe;AACf,MAAA,eAAA,CAAgB,UAAU,aAAA,CAAc,IAAA;AACxC,MAAA,eAAA,CAAgB,YAAY,aAAA,CAAc,EAAA;AAG1C,MAAA,MAAM,gBAAA,GAAmB,WAAA,CAAY,QAAA,IAAY,EAAE,QAAQ,EAAC,EAAG,QAAA,EAAU,IAAI,KAAA,EAAO,EAAC,EAAG,SAAA,EAAW,EAAC,EAAE;AACtG,MAAA,eAAA,CAAgB,QAAA,GAAW;AAAA,QACvB,MAAA,EAAQ,gBAAA,CAAiB,MAAA,IAAU,EAAC;AAAA,QACpC,UAAU,CAAC;AAAA,UACP,IAAI,aAAA,CAAc,EAAA;AAAA,UAClB,MAAM,aAAA,CAAc,IAAA;AAAA,UACpB,IAAA,EAAM;AAAA,SACT,CAAA;AAAA,QACD,KAAA,EAAO,gBAAA,CAAiB,KAAA,IAAS,EAAC;AAAA,QAClC,SAAA,EAAW,gBAAA,CAAiB,SAAA,IAAa;AAAC,OAC9C;AAAA,IACJ;AAGA,IAAA,IAAI,OAAA,CAAQ,SAAA,IAAa,OAAA,CAAQ,YAAA,EAAc;AAC3C,MAAA,MAAM,cAAc,IAAI,GAAA,CAAI,WAAA,CAAY,IAAA,IAAQ,EAAE,CAAA;AAElD,MAAA,IAAI,QAAQ,YAAA,EAAc;AACtB,QAAA,KAAA,MAAW,GAAA,IAAO,QAAQ,YAAA,EAAc;AACpC,UAAA,WAAA,CAAY,OAAO,GAAG,CAAA;AAAA,QAC1B;AAAA,MACJ;AAEA,MAAA,IAAI,QAAQ,SAAA,EAAW;AACnB,QAAA,KAAA,MAAW,GAAA,IAAO,QAAQ,SAAA,EAAW;AACjC,UAAA,WAAA,CAAY,IAAI,GAAG,CAAA;AAAA,QACvB;AAAA,MACJ;AAEA,MAAA,eAAA,CAAgB,IAAA,GAAO,KAAA,CAAM,IAAA,CAAK,WAAW,EAAE,IAAA,EAAK;AAAA,IACxD;AAGA,IAAA,IAAI,UAAA,GAAa,OAAA;AAEjB,IAAA,IAAI,aAAA,EAAe,OAAA,EAAS,WAAA,IAAe,OAAA,CAAQ,KAAA,EAAO;AACtD,MAAA,MAAM,MAAA,GAAS,QAAQ,SAAA,EAAU;AACjC,MAAA,MAAM,WAAA,GAAc,UAAA,CAAY,MAAA,CAAO,eAAA,IAA8B,SAAS,CAAA;AAE9E,MAAA,IAAI,aAAA,EAAe,SAAS,WAAA,EAAa;AACrC,QAAA,MAAM,aAAA,GAAgB,kBAAA,CAAmB,OAAA,EAAS,aAAa,CAAA;AAC/D,QAAA,MAAM,OAAA,GAAUA,MAAQ,CAAO,aAAA,EAAe,OAAO,CAAA;AAErD,QAAA,MAAM,YAAY,WAAA,CAAY,IAAA,YAAgB,OAAO,WAAA,CAAY,IAAA,uBAAW,IAAA,EAAK;AAEjF,QAAA,MAAM,cAAA,GAAyC;AAAA,UAC3C,cAAA,EAAgB,OAAA;AAAA,UAChB,SAAA;AAAA,UACA,UAAA,EAAY;AAAA,SAChB;AAEA,QAAA,MAAM,QAAA,GAAW,OAAA,CAAQ,KAAA,CAAM,cAAc,CAAA;AAE7C,QAAA,IAAI,QAAQ,KAAA,EAAO;AACf,UAAA,MAAM,WAAW,IAAA,CAAK,OAAA,CAAQ,QAAQ,eAAA,CAAgB,QAAA,EAAU,cAAc,CAAC,CAAA;AAC/E,UAAA,MAAM,SAAA,GAAY,6BAA6B,OAAO,CAAA;AACtD,UAAA,MAAM,YAAA,GAAe,YAAA,CAAa,OAAA,CAAQ,KAAK,CAAA;AAE/C,UAAA,IAAI,SAAA,EAAW;AACX,YAAA,MAAM,MAAM,SAAA,CAAU,GAAA,CAAI,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACpD,YAAA,MAAM,OAAO,SAAA,CAAU,IAAA,CAAK,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACtD,YAAA,MAAM,SAAS,SAAA,CAAU,MAAA,CAAO,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AAC1D,YAAA,UAAA,GAAa,IAAA,CAAK,IAAA,CAAK,QAAA,EAAU,CAAA,EAAG,GAAG,CAAA,CAAA,EAAI,IAAI,CAAA,EAAG,MAAM,CAAA,CAAA,EAAI,YAAY,CAAA,IAAA,CAAM,CAAA;AAAA,UAClF,CAAA,MAAO;AACH,YAAA,UAAA,GAAa,IAAA,CAAK,IAAA,CAAK,QAAA,EAAU,CAAA,EAAG,YAAY,CAAA,IAAA,CAAM,CAAA;AAAA,UAC1D;AAAA,QACJ,CAAA,MAAO;AACH,UAAA,UAAA,GAAa,OAAA,CAAQ,eAAA,CAAgB,QAAA,EAAU,cAAc,CAAA;AAC7D,UAAA,UAAA,GAAa,UAAA,CAAW,OAAA,CAAQ,OAAA,EAAS,MAAM,CAAA;AAAA,QACnD;AAAA,MACJ,CAAA,MAAA,IAAW,QAAQ,KAAA,EAAO;AACtB,QAAA,MAAM,GAAA,GAAM,IAAA,CAAK,OAAA,CAAQ,OAAO,CAAA;AAChC,QAAA,MAAM,SAAA,GAAY,6BAA6B,OAAO,CAAA;AACtD,QAAA,MAAM,YAAA,GAAe,YAAA,CAAa,OAAA,CAAQ,KAAK,CAAA;AAE/C,QAAA,IAAI,SAAA,EAAW;AACX,UAAA,MAAM,MAAM,SAAA,CAAU,GAAA,CAAI,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACpD,UAAA,MAAM,OAAO,SAAA,CAAU,IAAA,CAAK,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACtD,UAAA,MAAM,SAAS,SAAA,CAAU,MAAA,CAAO,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AAC1D,UAAA,UAAA,GAAa,IAAA,CAAK,IAAA,CAAK,GAAA,EAAK,CAAA,EAAG,GAAG,CAAA,CAAA,EAAI,IAAI,CAAA,EAAG,MAAM,CAAA,CAAA,EAAI,YAAY,CAAA,IAAA,CAAM,CAAA;AAAA,QAC7E,CAAA,MAAO;AACH,UAAA,UAAA,GAAa,IAAA,CAAK,IAAA,CAAK,GAAA,EAAK,CAAA,EAAG,YAAY,CAAA,IAAA,CAAM,CAAA;AAAA,QACrD;AAAA,MACJ;AAAA,IACJ;AAGA,IAAA,IAAI,CAAC,QAAQ,MAAA,EAAQ;AACjB,MAAA,IAAI,MAAA,CAAO,IAAA,CAAK,eAAe,CAAA,CAAE,SAAS,CAAA,EAAG;AACzC,QAAA,UAAA,CAAW,eAAe,eAAe,CAAA;AAAA,MAC7C;AAGA,MAAA,IAAI,eAAe,OAAA,EAAS;AAExB,QAAA,UAAA,CAAW,KAAA,EAAM;AAGjB,QAAA,MAAM,EAAA,CAAG,MAAM,IAAA,CAAK,OAAA,CAAQ,UAAU,CAAA,EAAG,EAAE,SAAA,EAAW,IAAA,EAAM,CAAA;AAG5D,QAAA,MAAM,EAAA,CAAG,QAAA,CAAS,OAAA,EAAS,UAAU,CAAA;AAGrC,QAAA,MAAM,EAAA,CAAG,OAAO,OAAO,CAAA;AAAA,MAC3B;AAAA,IACJ;AAEA,IAAA,OAAO,EAAE,YAAY,OAAA,EAAQ;AAAA,EACjC,CAAA,SAAE;AAEE,IAAA,IAAI;AACA,MAAA,UAAA,CAAW,KAAA,EAAM;AAAA,IACrB,CAAA,CAAA,MAAQ;AAAA,IAER;AAAA,EACJ;AACJ;AAgDO,MAAM,eAAA,GAAkB,OAAO,OAAA,KAAoE;AACtG,EAAA,MAAM;AAAA,IACF,SAAA;AAAA,IACA,KAAA,GAAQ,EAAA;AAAA,IACR,MAAA,GAAS,CAAA;AAAA,IACT,MAAA,GAAS,MAAA;AAAA,IACT,SAAA;AAAA,IACA,OAAA;AAAA,IACA,MAAA;AAAA,IACA;AAAA,GACJ,GAAI,OAAA;AAGJ,EAAA,MAAM,cAAA,GAAqC;AAAA,IACvC,SAAA;AAAA,IACA,KAAA;AAAA,IACA,MAAA;AAAA,IACA,MAAA;AAAA,IACA,MAAA;AAAA,IACA,OAAA,EAAS,SAAA;AAAA,IACT,SAAA;AAAA,IACA;AAAA,GACJ;AAEA,EAAA,MAAM,MAAA,GAAS,MAAMC,iBAAA,CAA2B,cAAc,CAAA;AAG9D,EAAA,MAAM,WAAA,GAAoC,MAAA,CAAO,WAAA,CAAY,GAAA,CAAI,CAAA,IAAA,KAAQ;AAErE,IAAA,IAAI,IAAA,GAAO,EAAA;AACX,IAAA,IAAI;AACA,MAAA,MAAM,UAAA,GAAa,cAAc,IAAA,CAAK,IAAA,CAAK,UAAU,EAAE,QAAA,EAAU,MAAM,CAAA;AACvE,MAAA,IAAA,GAAO,WAAW,QAAA,CAAS,EAAA;AAC3B,MAAA,UAAA,CAAW,KAAA,EAAM;AAAA,IACrB,CAAA,CAAA,MAAQ;AAEJ,MAAA,IAAA,GAAO,EAAA;AAAA,IACX;AAEA,IAAA,OAAO;AAAA,MACH,MAAM,IAAA,CAAK,QAAA;AAAA,MACX,QAAA,EAAU,IAAA,CAAK,QAAA,CAAS,IAAA,CAAK,QAAQ,CAAA;AAAA,MACrC,IAAA;AAAA,MACA,IAAA,EAAM,IAAA,CAAK,IAAA,YAAgB,IAAA,GAAO,IAAA,CAAK,IAAA,CAAK,WAAA,EAAY,CAAE,KAAA,CAAM,GAAG,CAAA,CAAE,CAAC,CAAA,GAAI,EAAA;AAAA,MAC1E,IAAA,EAAM,MAAA;AAAA;AAAA,MACN,OAAO,IAAA,CAAK,KAAA;AAAA,MACZ,gBAAA,EAAkB,KAAA;AAAA;AAAA,MAClB,SAAA,EAAW,IAAA,CAAK,IAAA,oBAAQ,IAAI,IAAA,EAAK;AAAA,MACjC,QAAQ,IAAA,CAAK,MAAA;AAAA,MACb,cAAA,EAAgB,MAAA;AAAA;AAAA,MAChB,WAAA,EAAa,KAAK,cAAA,EAAgB,MAAA;AAAA,MAClC,QAAA,EAAU,KAAK,OAAA,GAAU;AAAA,QACrB,UAAU,CAAC;AAAA,UACP,IAAI,IAAA,CAAK,OAAA;AAAA,UACT,MAAM,IAAA,CAAK;AAAA,SACd;AAAA,OACL,GAAI;AAAA,KACR;AAAA,EACJ,CAAC,CAAA;AAED,EAAA,OAAO;AAAA,IACH,WAAA;AAAA,IACA,OAAO,MAAA,CAAO,KAAA;AAAA,IACd,SAAS,MAAA,CAAO,OAAA;AAAA,IAChB,KAAA;AAAA,IACA;AAAA,GACJ;AACJ;;;;"}
@@ -0,0 +1,431 @@
1
+ import * as fs from 'fs/promises';
2
+ import * as path from 'node:path';
3
+ import { getLogger } from './index41.js';
4
+ import { extractTimestampFromFilename, slugifyTitle } from './index36.js';
5
+ import { PklTranscript } from '@redaksjon/protokoll-format';
6
+ import { ensurePklExtension } from './index38.js';
7
+
8
+ const FEEDBACK_TOOLS = [
9
+ {
10
+ name: "correct_text",
11
+ description: "Replace text in the transcript. Use this to fix misspellings, wrong terms, or incorrect names.",
12
+ parameters: {
13
+ find: { type: "string", description: "The text to find in the transcript", required: true },
14
+ replace: { type: "string", description: "The text to replace it with", required: true },
15
+ replace_all: { type: "boolean", description: "Replace all occurrences (default: true)" }
16
+ }
17
+ },
18
+ {
19
+ name: "add_term",
20
+ description: "Add a new term to the context so it will be recognized in future transcripts.",
21
+ parameters: {
22
+ term: { type: "string", description: "The correct term/abbreviation", required: true },
23
+ definition: { type: "string", description: "What the term means", required: true },
24
+ sounds_like: { type: "array", items: { type: "string" }, description: "Phonetic variations" },
25
+ context: { type: "string", description: "Additional context about when this term is used" }
26
+ }
27
+ },
28
+ {
29
+ name: "add_person",
30
+ description: "Add a new person to the context for future name recognition.",
31
+ parameters: {
32
+ name: { type: "string", description: "The correct full name", required: true },
33
+ sounds_like: { type: "array", items: { type: "string" }, description: "Phonetic variations", required: true },
34
+ role: { type: "string", description: "Their role or title" },
35
+ company: { type: "string", description: "Company they work for" },
36
+ context: { type: "string", description: "Additional context about this person" }
37
+ }
38
+ },
39
+ {
40
+ name: "change_project",
41
+ description: "Change the project assignment of this transcript.",
42
+ parameters: {
43
+ project_id: { type: "string", description: "The project ID to assign", required: true }
44
+ }
45
+ },
46
+ {
47
+ name: "change_title",
48
+ description: "Change the title of this transcript.",
49
+ parameters: {
50
+ new_title: { type: "string", description: "The new title for the transcript", required: true }
51
+ }
52
+ },
53
+ {
54
+ name: "provide_help",
55
+ description: "Provide helpful information to the user about what kinds of feedback they can give.",
56
+ parameters: {
57
+ topic: { type: "string", description: "The topic to help with", enum: ["terms", "people", "projects", "corrections", "general"] }
58
+ }
59
+ },
60
+ {
61
+ name: "complete",
62
+ description: "Call this when you have finished processing all the feedback.",
63
+ parameters: {
64
+ summary: { type: "string", description: "A summary of what was done", required: true }
65
+ }
66
+ }
67
+ ];
68
+ const executeTool = async (toolName, args, feedbackCtx) => {
69
+ const logger = getLogger();
70
+ switch (toolName) {
71
+ case "correct_text": {
72
+ const find = String(args.find);
73
+ const replace = String(args.replace);
74
+ const replaceAll = args.replace_all !== false;
75
+ if (!feedbackCtx.transcriptContent.includes(find)) {
76
+ return {
77
+ success: false,
78
+ message: `Text "${find}" not found in transcript.`
79
+ };
80
+ }
81
+ const occurrences = feedbackCtx.transcriptContent.split(find).length - 1;
82
+ if (replaceAll) {
83
+ feedbackCtx.transcriptContent = feedbackCtx.transcriptContent.split(find).join(replace);
84
+ } else {
85
+ feedbackCtx.transcriptContent = feedbackCtx.transcriptContent.replace(find, replace);
86
+ }
87
+ const changeCount = replaceAll ? occurrences : 1;
88
+ feedbackCtx.changes.push({
89
+ type: "text_correction",
90
+ description: `Replaced "${find}" with "${replace}" (${changeCount} occurrence${changeCount > 1 ? "s" : ""})`,
91
+ details: { find, replace, count: changeCount }
92
+ });
93
+ return {
94
+ success: true,
95
+ message: `Replaced ${changeCount} occurrence(s) of "${find}" with "${replace}".`
96
+ };
97
+ }
98
+ case "add_term": {
99
+ const term = String(args.term);
100
+ const definition = String(args.definition);
101
+ const soundsLike = args.sounds_like;
102
+ const termContext = args.context;
103
+ const id = term.toLowerCase().replace(/[^a-z0-9]+/g, "-").replace(/^-|-$/g, "");
104
+ const existing = feedbackCtx.context.getTerm(id);
105
+ if (existing) {
106
+ return {
107
+ success: false,
108
+ message: `Term "${term}" already exists in context.`
109
+ };
110
+ }
111
+ const newTerm = {
112
+ id,
113
+ name: term,
114
+ type: "term",
115
+ expansion: definition,
116
+ sounds_like: soundsLike,
117
+ domain: termContext
118
+ };
119
+ if (!feedbackCtx.dryRun) {
120
+ await feedbackCtx.context.saveEntity(newTerm);
121
+ }
122
+ feedbackCtx.changes.push({
123
+ type: "term_added",
124
+ description: `Added term "${term}" to context`,
125
+ details: { term, definition, sounds_like: soundsLike }
126
+ });
127
+ return {
128
+ success: true,
129
+ message: `Added term "${term}" to context.`,
130
+ data: { id, term, definition }
131
+ };
132
+ }
133
+ case "add_person": {
134
+ const name = String(args.name);
135
+ const soundsLike = args.sounds_like;
136
+ const role = args.role;
137
+ const company = args.company;
138
+ const personContext = args.context;
139
+ const id = name.toLowerCase().replace(/[^a-z0-9]+/g, "-").replace(/^-|-$/g, "");
140
+ const existing = feedbackCtx.context.getPerson(id);
141
+ if (existing) {
142
+ return {
143
+ success: false,
144
+ message: `Person "${name}" already exists in context.`
145
+ };
146
+ }
147
+ const newPerson = {
148
+ id,
149
+ name,
150
+ type: "person",
151
+ sounds_like: soundsLike,
152
+ role,
153
+ company,
154
+ context: personContext
155
+ };
156
+ if (!feedbackCtx.dryRun) {
157
+ await feedbackCtx.context.saveEntity(newPerson);
158
+ }
159
+ feedbackCtx.changes.push({
160
+ type: "person_added",
161
+ description: `Added person "${name}" to context`,
162
+ details: { name, sounds_like: soundsLike, role, company }
163
+ });
164
+ return {
165
+ success: true,
166
+ message: `Added person "${name}" to context.`,
167
+ data: { id, name, sounds_like: soundsLike }
168
+ };
169
+ }
170
+ case "change_project": {
171
+ const projectId = String(args.project_id);
172
+ const project = feedbackCtx.context.getProject(projectId);
173
+ if (!project) {
174
+ const available = feedbackCtx.context.getAllProjects().map((p) => p.id);
175
+ return {
176
+ success: false,
177
+ message: `Project "${projectId}" not found. Available: ${available.join(", ")}`
178
+ };
179
+ }
180
+ const metadataRegex = /\*\*Project\*\*: .+/;
181
+ const projectIdRegex = /\*\*Project ID\*\*: `.+`/;
182
+ if (metadataRegex.test(feedbackCtx.transcriptContent)) {
183
+ feedbackCtx.transcriptContent = feedbackCtx.transcriptContent.replace(
184
+ metadataRegex,
185
+ `**Project**: ${project.name}`
186
+ );
187
+ }
188
+ if (projectIdRegex.test(feedbackCtx.transcriptContent)) {
189
+ feedbackCtx.transcriptContent = feedbackCtx.transcriptContent.replace(
190
+ projectIdRegex,
191
+ `**Project ID**: \`${project.id}\``
192
+ );
193
+ }
194
+ feedbackCtx.changes.push({
195
+ type: "project_changed",
196
+ description: `Changed project to "${project.name}" (${project.id})`,
197
+ details: { project_id: projectId, project_name: project.name, routing: project.routing }
198
+ });
199
+ return {
200
+ success: true,
201
+ message: `Changed project to "${project.name}".`,
202
+ data: { project_id: projectId, destination: project.routing?.destination }
203
+ };
204
+ }
205
+ case "change_title": {
206
+ const newTitle = String(args.new_title);
207
+ feedbackCtx.changes.push({
208
+ type: "title_changed",
209
+ description: `Changed title to "${newTitle}"`,
210
+ details: { new_title: newTitle, slug: slugifyTitle(newTitle) }
211
+ });
212
+ return {
213
+ success: true,
214
+ message: `Changed title to "${newTitle}".`,
215
+ data: { new_title: newTitle }
216
+ };
217
+ }
218
+ case "provide_help": {
219
+ const topic = String(args.topic || "general");
220
+ let helpText = "";
221
+ switch (topic) {
222
+ case "terms":
223
+ helpText = "You can teach me about abbreviations, acronyms, and technical terms.";
224
+ break;
225
+ case "people":
226
+ helpText = "You can teach me about people whose names were transcribed incorrectly.";
227
+ break;
228
+ case "projects":
229
+ helpText = "You can tell me if a transcript belongs to a different project.";
230
+ break;
231
+ case "corrections":
232
+ helpText = "You can ask me to fix any text in the transcript.";
233
+ break;
234
+ default:
235
+ helpText = "I can help with terms, names, projects, and general corrections.";
236
+ }
237
+ return {
238
+ success: true,
239
+ message: helpText
240
+ };
241
+ }
242
+ case "complete": {
243
+ const summary = String(args.summary);
244
+ return {
245
+ success: true,
246
+ message: summary,
247
+ data: { complete: true }
248
+ };
249
+ }
250
+ default:
251
+ logger.warn("Unknown tool: %s", toolName);
252
+ return {
253
+ success: false,
254
+ message: `Unknown tool: ${toolName}`
255
+ };
256
+ }
257
+ };
258
+ const buildFeedbackSystemPrompt = (transcriptPreview, availableProjects) => {
259
+ const toolDescriptions = FEEDBACK_TOOLS.map(
260
+ (t) => `- ${t.name}: ${t.description}`
261
+ ).join("\n");
262
+ return `You are an intelligent feedback processor for a transcription system.
263
+
264
+ ## Current Transcript Preview
265
+ ${transcriptPreview.substring(0, 1e3)}${transcriptPreview.length > 1e3 ? "..." : ""}
266
+
267
+ ## Available Projects
268
+ ${availableProjects.length > 0 ? availableProjects.join(", ") : "(no projects configured)"}
269
+
270
+ ## Available Tools
271
+ ${toolDescriptions}
272
+
273
+ ## Rules
274
+ - Understand the feedback and identify necessary actions
275
+ - Execute tools in order: text corrections, then context entities, then metadata
276
+ - Always call 'complete' when finished with a summary
277
+ - For name/term corrections: BOTH fix the text AND add to context`;
278
+ };
279
+ const processFeedback = async (feedback, feedbackCtx, reasoning) => {
280
+ const logger = getLogger();
281
+ const projects = feedbackCtx.context.getAllProjects().map((p) => `${p.id} (${p.name})`);
282
+ const systemPrompt = buildFeedbackSystemPrompt(feedbackCtx.transcriptContent, projects);
283
+ const tools = FEEDBACK_TOOLS.map((t) => ({
284
+ type: "function",
285
+ function: {
286
+ name: t.name,
287
+ description: t.description,
288
+ parameters: {
289
+ type: "object",
290
+ properties: Object.fromEntries(
291
+ Object.entries(t.parameters).map(([key, param]) => [
292
+ key,
293
+ {
294
+ type: param.type,
295
+ description: param.description,
296
+ ...param.enum ? { enum: param.enum } : {},
297
+ ...param.items ? { items: param.items } : {}
298
+ }
299
+ ])
300
+ ),
301
+ required: Object.entries(t.parameters).filter(([_, p]) => p.required).map(([key]) => key)
302
+ }
303
+ }
304
+ }));
305
+ let iterations = 0;
306
+ const maxIterations = 10;
307
+ const conversationHistory = [
308
+ { role: "system", content: systemPrompt },
309
+ { role: "user", content: feedback }
310
+ ];
311
+ while (iterations < maxIterations) {
312
+ iterations++;
313
+ logger.debug("Feedback processing iteration %d", iterations);
314
+ try {
315
+ const response = await reasoning.completeWithTools({
316
+ messages: conversationHistory,
317
+ tools
318
+ });
319
+ if (response.tool_calls && response.tool_calls.length > 0) {
320
+ conversationHistory.push({
321
+ role: "assistant",
322
+ content: response.content || "",
323
+ tool_calls: response.tool_calls.map((tc) => ({
324
+ id: tc.id,
325
+ function: {
326
+ name: tc.function.name,
327
+ arguments: tc.function.arguments
328
+ }
329
+ }))
330
+ });
331
+ for (const toolCall of response.tool_calls) {
332
+ const toolName = toolCall.function.name;
333
+ let args;
334
+ try {
335
+ args = JSON.parse(toolCall.function.arguments);
336
+ } catch {
337
+ args = {};
338
+ }
339
+ const result = await executeTool(toolName, args, feedbackCtx);
340
+ conversationHistory.push({
341
+ role: "tool",
342
+ content: JSON.stringify(result),
343
+ tool_call_id: toolCall.id
344
+ });
345
+ if (toolName === "complete") {
346
+ return;
347
+ }
348
+ }
349
+ } else {
350
+ return;
351
+ }
352
+ } catch (error) {
353
+ logger.error("Error during feedback processing", { error });
354
+ throw error;
355
+ }
356
+ }
357
+ logger.warn("Feedback processing reached max iterations");
358
+ };
359
+ const applyChanges = async (feedbackCtx) => {
360
+ const logger = getLogger();
361
+ const pklPath = ensurePklExtension(feedbackCtx.transcriptPath);
362
+ let newPath = pklPath;
363
+ let moved = false;
364
+ const titleChange = feedbackCtx.changes.find((c) => c.type === "title_changed");
365
+ if (titleChange) {
366
+ const slug = titleChange.details.slug;
367
+ const timestamp = extractTimestampFromFilename(pklPath);
368
+ const dir = path.dirname(pklPath);
369
+ if (timestamp) {
370
+ const timeStr = `${timestamp.hour.toString().padStart(2, "0")}${timestamp.minute.toString().padStart(2, "0")}`;
371
+ newPath = path.join(dir, `${timestamp.day}-${timeStr}-${slug}.pkl`);
372
+ } else {
373
+ newPath = path.join(dir, `${slug}.pkl`);
374
+ }
375
+ }
376
+ const projectChange = feedbackCtx.changes.find((c) => c.type === "project_changed");
377
+ if (projectChange && projectChange.details.routing) {
378
+ const routing = projectChange.details.routing;
379
+ if (routing.destination) {
380
+ let dest = routing.destination;
381
+ if (dest.startsWith("~")) {
382
+ dest = path.join(process.env.HOME || "", dest.slice(1));
383
+ }
384
+ const now = /* @__PURE__ */ new Date();
385
+ const year = now.getFullYear().toString();
386
+ const month = (now.getMonth() + 1).toString().padStart(2, "0");
387
+ let structuredPath = dest;
388
+ const structure = routing.structure || "month";
389
+ if (structure === "year") {
390
+ structuredPath = path.join(dest, year);
391
+ } else if (structure === "month") {
392
+ structuredPath = path.join(dest, year, month);
393
+ } else if (structure === "day") {
394
+ const day = now.getDate().toString().padStart(2, "0");
395
+ structuredPath = path.join(dest, year, month, day);
396
+ }
397
+ const filename = path.basename(newPath);
398
+ newPath = path.join(structuredPath, filename);
399
+ moved = true;
400
+ }
401
+ }
402
+ await fs.mkdir(path.dirname(newPath), { recursive: true });
403
+ if (!feedbackCtx.dryRun) {
404
+ const transcript = PklTranscript.open(pklPath, { readOnly: false });
405
+ try {
406
+ if (feedbackCtx.transcriptContent !== feedbackCtx.originalContent) {
407
+ transcript.updateContent(feedbackCtx.transcriptContent);
408
+ }
409
+ if (titleChange) {
410
+ transcript.updateMetadata({ title: titleChange.details.new_title });
411
+ }
412
+ if (projectChange) {
413
+ transcript.updateMetadata({
414
+ projectId: projectChange.details.project_id,
415
+ project: projectChange.details.project_name
416
+ });
417
+ }
418
+ } finally {
419
+ transcript.close();
420
+ }
421
+ if (newPath !== pklPath) {
422
+ await fs.copyFile(pklPath, newPath);
423
+ await fs.unlink(pklPath);
424
+ }
425
+ }
426
+ logger.info("Applied %d changes to transcript", feedbackCtx.changes.length);
427
+ return { newPath, moved };
428
+ };
429
+
430
+ export { FEEDBACK_TOOLS, applyChanges, buildFeedbackSystemPrompt, executeTool, processFeedback };
431
+ //# sourceMappingURL=index37.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index37.js","sources":["../src/transcript/feedback.ts"],"sourcesContent":["/**\n * Feedback Operations\n * \n * Core business logic for processing natural language feedback on transcripts.\n * Extracted from CLI to provide reusable functions for MCP tools.\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'node:path';\nimport * as Context from '@redaksjon/context';\nimport * as Reasoning from '../reasoning';\nimport * as Logging from '../logging';\nimport { slugifyTitle, extractTimestampFromFilename } from './operations';\nimport { PklTranscript } from '@redaksjon/protokoll-format';\nimport { ensurePklExtension } from './pkl-utils';\n\n/**\n * Tool definitions for feedback processor\n */\nexport interface FeedbackTool {\n name: string;\n description: string;\n parameters: Record<string, {\n type: string;\n description: string;\n required?: boolean;\n enum?: string[];\n items?: { type: string };\n }>;\n}\n\nexport const FEEDBACK_TOOLS: FeedbackTool[] = [\n {\n name: 'correct_text',\n description: 'Replace text in the transcript. Use this to fix misspellings, wrong terms, or incorrect names.',\n parameters: {\n find: { type: 'string', description: 'The text to find in the transcript', required: true },\n replace: { type: 'string', description: 'The text to replace it with', required: true },\n replace_all: { type: 'boolean', description: 'Replace all occurrences (default: true)' },\n },\n },\n {\n name: 'add_term',\n description: 'Add a new term to the context so it will be recognized in future transcripts.',\n parameters: {\n term: { type: 'string', description: 'The correct term/abbreviation', required: true },\n definition: { type: 'string', description: 'What the term means', required: true },\n sounds_like: { type: 'array', items: { type: 'string' }, description: 'Phonetic variations' },\n context: { type: 'string', description: 'Additional context about when this term is used' },\n },\n },\n {\n name: 'add_person',\n description: 'Add a new person to the context for future name recognition.',\n parameters: {\n name: { type: 'string', description: 'The correct full name', required: true },\n sounds_like: { type: 'array', items: { type: 'string' }, description: 'Phonetic variations', required: true },\n role: { type: 'string', description: 'Their role or title' },\n company: { type: 'string', description: 'Company they work for' },\n context: { type: 'string', description: 'Additional context about this person' },\n },\n },\n {\n name: 'change_project',\n description: 'Change the project assignment of this transcript.',\n parameters: {\n project_id: { type: 'string', description: 'The project ID to assign', required: true },\n },\n },\n {\n name: 'change_title',\n description: 'Change the title of this transcript.',\n parameters: {\n new_title: { type: 'string', description: 'The new title for the transcript', required: true },\n },\n },\n {\n name: 'provide_help',\n description: 'Provide helpful information to the user about what kinds of feedback they can give.',\n parameters: {\n topic: { type: 'string', description: 'The topic to help with', enum: ['terms', 'people', 'projects', 'corrections', 'general'] },\n },\n },\n {\n name: 'complete',\n description: 'Call this when you have finished processing all the feedback.',\n parameters: {\n summary: { type: 'string', description: 'A summary of what was done', required: true },\n },\n },\n];\n\n/**\n * Tool execution result\n */\nexport interface ToolResult {\n success: boolean;\n message: string;\n data?: Record<string, unknown>;\n}\n\n/**\n * Feedback processing context\n */\nexport interface FeedbackContext {\n transcriptPath: string;\n transcriptContent: string;\n originalContent: string;\n context: Context.ContextInstance;\n changes: FeedbackChange[];\n verbose: boolean;\n dryRun: boolean;\n}\n\nexport interface FeedbackChange {\n type: 'text_correction' | 'term_added' | 'person_added' | 'project_changed' | 'title_changed';\n description: string;\n details: Record<string, unknown>;\n}\n\n/**\n * Execute a feedback tool\n */\nexport const executeTool = async (\n toolName: string,\n args: Record<string, unknown>,\n feedbackCtx: FeedbackContext\n): Promise<ToolResult> => {\n const logger = Logging.getLogger();\n \n switch (toolName) {\n case 'correct_text': {\n const find = String(args.find);\n const replace = String(args.replace);\n const replaceAll = args.replace_all !== false;\n \n if (!feedbackCtx.transcriptContent.includes(find)) {\n return {\n success: false,\n message: `Text \"${find}\" not found in transcript.`,\n };\n }\n \n const occurrences = feedbackCtx.transcriptContent.split(find).length - 1;\n \n if (replaceAll) {\n feedbackCtx.transcriptContent = feedbackCtx.transcriptContent.split(find).join(replace);\n } else {\n feedbackCtx.transcriptContent = feedbackCtx.transcriptContent.replace(find, replace);\n }\n \n const changeCount = replaceAll ? occurrences : 1;\n \n feedbackCtx.changes.push({\n type: 'text_correction',\n description: `Replaced \"${find}\" with \"${replace}\" (${changeCount} occurrence${changeCount > 1 ? 's' : ''})`,\n details: { find, replace, count: changeCount },\n });\n \n return {\n success: true,\n message: `Replaced ${changeCount} occurrence(s) of \"${find}\" with \"${replace}\".`,\n };\n }\n \n case 'add_term': {\n const term = String(args.term);\n const definition = String(args.definition);\n const soundsLike = args.sounds_like as string[] | undefined;\n const termContext = args.context as string | undefined;\n \n const id = term.toLowerCase().replace(/[^a-z0-9]+/g, '-').replace(/^-|-$/g, '');\n \n const existing = feedbackCtx.context.getTerm(id);\n if (existing) {\n return {\n success: false,\n message: `Term \"${term}\" already exists in context.`,\n };\n }\n \n const newTerm: Context.Term = {\n id,\n name: term,\n type: 'term',\n expansion: definition,\n sounds_like: soundsLike,\n domain: termContext,\n };\n \n if (!feedbackCtx.dryRun) {\n await feedbackCtx.context.saveEntity(newTerm);\n }\n \n feedbackCtx.changes.push({\n type: 'term_added',\n description: `Added term \"${term}\" to context`,\n details: { term, definition, sounds_like: soundsLike },\n });\n \n return {\n success: true,\n message: `Added term \"${term}\" to context.`,\n data: { id, term, definition },\n };\n }\n \n case 'add_person': {\n const name = String(args.name);\n const soundsLike = args.sounds_like as string[];\n const role = args.role as string | undefined;\n const company = args.company as string | undefined;\n const personContext = args.context as string | undefined;\n \n const id = name.toLowerCase().replace(/[^a-z0-9]+/g, '-').replace(/^-|-$/g, '');\n \n const existing = feedbackCtx.context.getPerson(id);\n if (existing) {\n return {\n success: false,\n message: `Person \"${name}\" already exists in context.`,\n };\n }\n \n const newPerson: Context.Person = {\n id,\n name,\n type: 'person',\n sounds_like: soundsLike,\n role,\n company,\n context: personContext,\n };\n \n if (!feedbackCtx.dryRun) {\n await feedbackCtx.context.saveEntity(newPerson);\n }\n \n feedbackCtx.changes.push({\n type: 'person_added',\n description: `Added person \"${name}\" to context`,\n details: { name, sounds_like: soundsLike, role, company },\n });\n \n return {\n success: true,\n message: `Added person \"${name}\" to context.`,\n data: { id, name, sounds_like: soundsLike },\n };\n }\n \n case 'change_project': {\n const projectId = String(args.project_id);\n \n const project = feedbackCtx.context.getProject(projectId);\n if (!project) {\n const available = feedbackCtx.context.getAllProjects().map(p => p.id);\n return {\n success: false,\n message: `Project \"${projectId}\" not found. Available: ${available.join(', ')}`,\n };\n }\n \n const metadataRegex = /\\*\\*Project\\*\\*: .+/;\n const projectIdRegex = /\\*\\*Project ID\\*\\*: `.+`/;\n \n if (metadataRegex.test(feedbackCtx.transcriptContent)) {\n feedbackCtx.transcriptContent = feedbackCtx.transcriptContent.replace(\n metadataRegex,\n `**Project**: ${project.name}`\n );\n }\n \n if (projectIdRegex.test(feedbackCtx.transcriptContent)) {\n feedbackCtx.transcriptContent = feedbackCtx.transcriptContent.replace(\n projectIdRegex,\n `**Project ID**: \\`${project.id}\\``\n );\n }\n \n feedbackCtx.changes.push({\n type: 'project_changed',\n description: `Changed project to \"${project.name}\" (${project.id})`,\n details: { project_id: projectId, project_name: project.name, routing: project.routing },\n });\n \n return {\n success: true,\n message: `Changed project to \"${project.name}\".`,\n data: { project_id: projectId, destination: project.routing?.destination },\n };\n }\n \n case 'change_title': {\n const newTitle = String(args.new_title);\n \n // For PKL format, title is stored in metadata, not in content\n // We just track the change here - applyChanges will update the PKL metadata\n feedbackCtx.changes.push({\n type: 'title_changed',\n description: `Changed title to \"${newTitle}\"`,\n details: { new_title: newTitle, slug: slugifyTitle(newTitle) },\n });\n \n return {\n success: true,\n message: `Changed title to \"${newTitle}\".`,\n data: { new_title: newTitle },\n };\n }\n \n case 'provide_help': {\n const topic = String(args.topic || 'general');\n let helpText = '';\n \n switch (topic) {\n case 'terms':\n helpText = 'You can teach me about abbreviations, acronyms, and technical terms.';\n break;\n case 'people':\n helpText = 'You can teach me about people whose names were transcribed incorrectly.';\n break;\n case 'projects':\n helpText = 'You can tell me if a transcript belongs to a different project.';\n break;\n case 'corrections':\n helpText = 'You can ask me to fix any text in the transcript.';\n break;\n default:\n helpText = 'I can help with terms, names, projects, and general corrections.';\n }\n \n return {\n success: true,\n message: helpText,\n };\n }\n \n case 'complete': {\n const summary = String(args.summary);\n return {\n success: true,\n message: summary,\n data: { complete: true },\n };\n }\n \n default:\n logger.warn('Unknown tool: %s', toolName);\n return {\n success: false,\n message: `Unknown tool: ${toolName}`,\n };\n }\n};\n\n/**\n * Build system prompt for feedback agent\n */\nexport const buildFeedbackSystemPrompt = (\n transcriptPreview: string,\n availableProjects: string[]\n): string => {\n const toolDescriptions = FEEDBACK_TOOLS.map(t => \n `- ${t.name}: ${t.description}`\n ).join('\\n');\n \n return `You are an intelligent feedback processor for a transcription system.\n\n## Current Transcript Preview\n${transcriptPreview.substring(0, 1000)}${transcriptPreview.length > 1000 ? '...' : ''}\n\n## Available Projects\n${availableProjects.length > 0 ? availableProjects.join(', ') : '(no projects configured)'}\n\n## Available Tools\n${toolDescriptions}\n\n## Rules\n- Understand the feedback and identify necessary actions\n- Execute tools in order: text corrections, then context entities, then metadata\n- Always call 'complete' when finished with a summary\n- For name/term corrections: BOTH fix the text AND add to context`;\n};\n\n/**\n * Process feedback using agentic model\n */\nexport const processFeedback = async (\n feedback: string,\n feedbackCtx: FeedbackContext,\n reasoning: Reasoning.ReasoningInstance\n): Promise<void> => {\n const logger = Logging.getLogger();\n \n const projects = feedbackCtx.context.getAllProjects().map(p => `${p.id} (${p.name})`);\n const systemPrompt = buildFeedbackSystemPrompt(feedbackCtx.transcriptContent, projects);\n \n const tools = FEEDBACK_TOOLS.map(t => ({\n type: 'function' as const,\n function: {\n name: t.name,\n description: t.description,\n parameters: {\n type: 'object',\n properties: Object.fromEntries(\n Object.entries(t.parameters).map(([key, param]) => [\n key,\n {\n type: param.type,\n description: param.description,\n ...(param.enum ? { enum: param.enum } : {}),\n ...(param.items ? { items: param.items } : {}),\n },\n ])\n ),\n required: Object.entries(t.parameters)\n .filter(([_, p]) => p.required)\n .map(([key]) => key),\n },\n },\n }));\n \n let iterations = 0;\n const maxIterations = 10;\n const conversationHistory: Array<{\n role: 'system' | 'user' | 'assistant' | 'tool';\n content: string;\n tool_call_id?: string;\n tool_calls?: Array<{ id: string; function: { name: string; arguments: string } }>;\n }> = [\n { role: 'system', content: systemPrompt },\n { role: 'user', content: feedback },\n ];\n \n while (iterations < maxIterations) {\n iterations++;\n logger.debug('Feedback processing iteration %d', iterations);\n \n try {\n const response = await reasoning.completeWithTools({\n messages: conversationHistory,\n tools,\n });\n \n if (response.tool_calls && response.tool_calls.length > 0) {\n conversationHistory.push({\n role: 'assistant',\n content: response.content || '',\n tool_calls: response.tool_calls.map(tc => ({\n id: tc.id,\n function: {\n name: tc.function.name,\n arguments: tc.function.arguments,\n },\n })),\n });\n \n for (const toolCall of response.tool_calls) {\n const toolName = toolCall.function.name;\n let args: Record<string, unknown>;\n \n try {\n args = JSON.parse(toolCall.function.arguments);\n } catch {\n args = {};\n }\n \n const result = await executeTool(toolName, args, feedbackCtx);\n \n conversationHistory.push({\n role: 'tool',\n content: JSON.stringify(result),\n tool_call_id: toolCall.id,\n });\n \n if (toolName === 'complete') {\n return;\n }\n }\n } else {\n return;\n }\n } catch (error) {\n logger.error('Error during feedback processing', { error });\n throw error;\n }\n }\n \n logger.warn('Feedback processing reached max iterations');\n};\n\n/**\n * Apply changes and save transcript\n */\nexport const applyChanges = async (\n feedbackCtx: FeedbackContext\n): Promise<{ newPath: string; moved: boolean }> => {\n const logger = Logging.getLogger();\n \n // Ensure we're working with PKL files\n const pklPath = ensurePklExtension(feedbackCtx.transcriptPath);\n let newPath = pklPath;\n let moved = false;\n \n const titleChange = feedbackCtx.changes.find(c => c.type === 'title_changed');\n if (titleChange) {\n const slug = titleChange.details.slug as string;\n const timestamp = extractTimestampFromFilename(pklPath);\n const dir = path.dirname(pklPath);\n \n if (timestamp) {\n const timeStr = `${timestamp.hour.toString().padStart(2, '0')}${timestamp.minute.toString().padStart(2, '0')}`;\n newPath = path.join(dir, `${timestamp.day}-${timeStr}-${slug}.pkl`);\n } else {\n newPath = path.join(dir, `${slug}.pkl`);\n }\n }\n \n const projectChange = feedbackCtx.changes.find(c => c.type === 'project_changed');\n if (projectChange && projectChange.details.routing) {\n const routing = projectChange.details.routing as { destination?: string; structure?: string };\n if (routing.destination) {\n let dest = routing.destination;\n if (dest.startsWith('~')) {\n dest = path.join(process.env.HOME || '', dest.slice(1));\n }\n \n const now = new Date();\n const year = now.getFullYear().toString();\n const month = (now.getMonth() + 1).toString().padStart(2, '0');\n \n let structuredPath = dest;\n const structure = routing.structure || 'month';\n if (structure === 'year') {\n structuredPath = path.join(dest, year);\n } else if (structure === 'month') {\n structuredPath = path.join(dest, year, month);\n } else if (structure === 'day') {\n const day = now.getDate().toString().padStart(2, '0');\n structuredPath = path.join(dest, year, month, day);\n }\n \n const filename = path.basename(newPath);\n newPath = path.join(structuredPath, filename);\n moved = true;\n }\n }\n \n await fs.mkdir(path.dirname(newPath), { recursive: true });\n \n if (!feedbackCtx.dryRun) {\n // Open the PKL transcript and apply changes\n const transcript = PklTranscript.open(pklPath, { readOnly: false });\n try {\n // Update content if it was modified\n if (feedbackCtx.transcriptContent !== feedbackCtx.originalContent) {\n transcript.updateContent(feedbackCtx.transcriptContent);\n }\n \n // Update title if it was changed\n if (titleChange) {\n transcript.updateMetadata({ title: titleChange.details.new_title as string });\n }\n \n // Update project if it was changed\n if (projectChange) {\n transcript.updateMetadata({ \n projectId: projectChange.details.project_id as string,\n project: projectChange.details.project_name as string,\n });\n }\n } finally {\n transcript.close();\n }\n \n // Move file if path changed\n if (newPath !== pklPath) {\n await fs.copyFile(pklPath, newPath);\n await fs.unlink(pklPath);\n }\n }\n \n logger.info('Applied %d changes to transcript', feedbackCtx.changes.length);\n \n return { newPath, moved };\n};\n"],"names":["Logging.getLogger"],"mappings":";;;;;;;AA+BO,MAAM,cAAA,GAAiC;AAAA,EAC1C;AAAA,IACI,IAAA,EAAM,cAAA;AAAA,IACN,WAAA,EAAa,gGAAA;AAAA,IACb,UAAA,EAAY;AAAA,MACR,MAAM,EAAE,IAAA,EAAM,UAAU,WAAA,EAAa,oCAAA,EAAsC,UAAU,IAAA,EAAK;AAAA,MAC1F,SAAS,EAAE,IAAA,EAAM,UAAU,WAAA,EAAa,6BAAA,EAA+B,UAAU,IAAA,EAAK;AAAA,MACtF,WAAA,EAAa,EAAE,IAAA,EAAM,SAAA,EAAW,aAAa,yCAAA;AAA0C;AAC3F,GACJ;AAAA,EACA;AAAA,IACI,IAAA,EAAM,UAAA;AAAA,IACN,WAAA,EAAa,+EAAA;AAAA,IACb,UAAA,EAAY;AAAA,MACR,MAAM,EAAE,IAAA,EAAM,UAAU,WAAA,EAAa,+BAAA,EAAiC,UAAU,IAAA,EAAK;AAAA,MACrF,YAAY,EAAE,IAAA,EAAM,UAAU,WAAA,EAAa,qBAAA,EAAuB,UAAU,IAAA,EAAK;AAAA,MACjF,WAAA,EAAa,EAAE,IAAA,EAAM,OAAA,EAAS,KAAA,EAAO,EAAE,IAAA,EAAM,QAAA,EAAS,EAAG,WAAA,EAAa,qBAAA,EAAsB;AAAA,MAC5F,OAAA,EAAS,EAAE,IAAA,EAAM,QAAA,EAAU,aAAa,iDAAA;AAAkD;AAC9F,GACJ;AAAA,EACA;AAAA,IACI,IAAA,EAAM,YAAA;AAAA,IACN,WAAA,EAAa,8DAAA;AAAA,IACb,UAAA,EAAY;AAAA,MACR,MAAM,EAAE,IAAA,EAAM,UAAU,WAAA,EAAa,uBAAA,EAAyB,UAAU,IAAA,EAAK;AAAA,MAC7E,WAAA,EAAa,EAAE,IAAA,EAAM,OAAA,EAAS,KAAA,EAAO,EAAE,IAAA,EAAM,QAAA,EAAS,EAAG,WAAA,EAAa,qBAAA,EAAuB,QAAA,EAAU,IAAA,EAAK;AAAA,MAC5G,IAAA,EAAM,EAAE,IAAA,EAAM,QAAA,EAAU,aAAa,qBAAA,EAAsB;AAAA,MAC3D,OAAA,EAAS,EAAE,IAAA,EAAM,QAAA,EAAU,aAAa,uBAAA,EAAwB;AAAA,MAChE,OAAA,EAAS,EAAE,IAAA,EAAM,QAAA,EAAU,aAAa,sCAAA;AAAuC;AACnF,GACJ;AAAA,EACA;AAAA,IACI,IAAA,EAAM,gBAAA;AAAA,IACN,WAAA,EAAa,mDAAA;AAAA,IACb,UAAA,EAAY;AAAA,MACR,YAAY,EAAE,IAAA,EAAM,UAAU,WAAA,EAAa,0BAAA,EAA4B,UAAU,IAAA;AAAK;AAC1F,GACJ;AAAA,EACA;AAAA,IACI,IAAA,EAAM,cAAA;AAAA,IACN,WAAA,EAAa,sCAAA;AAAA,IACb,UAAA,EAAY;AAAA,MACR,WAAW,EAAE,IAAA,EAAM,UAAU,WAAA,EAAa,kCAAA,EAAoC,UAAU,IAAA;AAAK;AACjG,GACJ;AAAA,EACA;AAAA,IACI,IAAA,EAAM,cAAA;AAAA,IACN,WAAA,EAAa,qFAAA;AAAA,IACb,UAAA,EAAY;AAAA,MACR,KAAA,EAAO,EAAE,IAAA,EAAM,QAAA,EAAU,WAAA,EAAa,wBAAA,EAA0B,IAAA,EAAM,CAAC,OAAA,EAAS,QAAA,EAAU,UAAA,EAAY,aAAA,EAAe,SAAS,CAAA;AAAE;AACpI,GACJ;AAAA,EACA;AAAA,IACI,IAAA,EAAM,UAAA;AAAA,IACN,WAAA,EAAa,+DAAA;AAAA,IACb,UAAA,EAAY;AAAA,MACR,SAAS,EAAE,IAAA,EAAM,UAAU,WAAA,EAAa,4BAAA,EAA8B,UAAU,IAAA;AAAK;AACzF;AAER;AAiCO,MAAM,WAAA,GAAc,OACvB,QAAA,EACA,IAAA,EACA,WAAA,KACsB;AACtB,EAAA,MAAM,MAAA,GAASA,SAAQ,EAAU;AAEjC,EAAA,QAAQ,QAAA;AAAU,IACd,KAAK,cAAA,EAAgB;AACjB,MAAA,MAAM,IAAA,GAAO,MAAA,CAAO,IAAA,CAAK,IAAI,CAAA;AAC7B,MAAA,MAAM,OAAA,GAAU,MAAA,CAAO,IAAA,CAAK,OAAO,CAAA;AACnC,MAAA,MAAM,UAAA,GAAa,KAAK,WAAA,KAAgB,KAAA;AAExC,MAAA,IAAI,CAAC,WAAA,CAAY,iBAAA,CAAkB,QAAA,CAAS,IAAI,CAAA,EAAG;AAC/C,QAAA,OAAO;AAAA,UACH,OAAA,EAAS,KAAA;AAAA,UACT,OAAA,EAAS,SAAS,IAAI,CAAA,0BAAA;AAAA,SAC1B;AAAA,MACJ;AAEA,MAAA,MAAM,cAAc,WAAA,CAAY,iBAAA,CAAkB,KAAA,CAAM,IAAI,EAAE,MAAA,GAAS,CAAA;AAEvE,MAAA,IAAI,UAAA,EAAY;AACZ,QAAA,WAAA,CAAY,oBAAoB,WAAA,CAAY,iBAAA,CAAkB,MAAM,IAAI,CAAA,CAAE,KAAK,OAAO,CAAA;AAAA,MAC1F,CAAA,MAAO;AACH,QAAA,WAAA,CAAY,iBAAA,GAAoB,WAAA,CAAY,iBAAA,CAAkB,OAAA,CAAQ,MAAM,OAAO,CAAA;AAAA,MACvF;AAEA,MAAA,MAAM,WAAA,GAAc,aAAa,WAAA,GAAc,CAAA;AAE/C,MAAA,WAAA,CAAY,QAAQ,IAAA,CAAK;AAAA,QACrB,IAAA,EAAM,iBAAA;AAAA,QACN,WAAA,EAAa,CAAA,UAAA,EAAa,IAAI,CAAA,QAAA,EAAW,OAAO,CAAA,GAAA,EAAM,WAAW,CAAA,WAAA,EAAc,WAAA,GAAc,CAAA,GAAI,GAAA,GAAM,EAAE,CAAA,CAAA,CAAA;AAAA,QACzG,OAAA,EAAS,EAAE,IAAA,EAAM,OAAA,EAAS,OAAO,WAAA;AAAY,OAChD,CAAA;AAED,MAAA,OAAO;AAAA,QACH,OAAA,EAAS,IAAA;AAAA,QACT,SAAS,CAAA,SAAA,EAAY,WAAW,CAAA,mBAAA,EAAsB,IAAI,WAAW,OAAO,CAAA,EAAA;AAAA,OAChF;AAAA,IACJ;AAAA,IAEA,KAAK,UAAA,EAAY;AACb,MAAA,MAAM,IAAA,GAAO,MAAA,CAAO,IAAA,CAAK,IAAI,CAAA;AAC7B,MAAA,MAAM,UAAA,GAAa,MAAA,CAAO,IAAA,CAAK,UAAU,CAAA;AACzC,MAAA,MAAM,aAAa,IAAA,CAAK,WAAA;AACxB,MAAA,MAAM,cAAc,IAAA,CAAK,OAAA;AAEzB,MAAA,MAAM,EAAA,GAAK,IAAA,CAAK,WAAA,EAAY,CAAE,OAAA,CAAQ,eAAe,GAAG,CAAA,CAAE,OAAA,CAAQ,QAAA,EAAU,EAAE,CAAA;AAE9E,MAAA,MAAM,QAAA,GAAW,WAAA,CAAY,OAAA,CAAQ,OAAA,CAAQ,EAAE,CAAA;AAC/C,MAAA,IAAI,QAAA,EAAU;AACV,QAAA,OAAO;AAAA,UACH,OAAA,EAAS,KAAA;AAAA,UACT,OAAA,EAAS,SAAS,IAAI,CAAA,4BAAA;AAAA,SAC1B;AAAA,MACJ;AAEA,MAAA,MAAM,OAAA,GAAwB;AAAA,QAC1B,EAAA;AAAA,QACA,IAAA,EAAM,IAAA;AAAA,QACN,IAAA,EAAM,MAAA;AAAA,QACN,SAAA,EAAW,UAAA;AAAA,QACX,WAAA,EAAa,UAAA;AAAA,QACb,MAAA,EAAQ;AAAA,OACZ;AAEA,MAAA,IAAI,CAAC,YAAY,MAAA,EAAQ;AACrB,QAAA,MAAM,WAAA,CAAY,OAAA,CAAQ,UAAA,CAAW,OAAO,CAAA;AAAA,MAChD;AAEA,MAAA,WAAA,CAAY,QAAQ,IAAA,CAAK;AAAA,QACrB,IAAA,EAAM,YAAA;AAAA,QACN,WAAA,EAAa,eAAe,IAAI,CAAA,YAAA,CAAA;AAAA,QAChC,OAAA,EAAS,EAAE,IAAA,EAAM,UAAA,EAAY,aAAa,UAAA;AAAW,OACxD,CAAA;AAED,MAAA,OAAO;AAAA,QACH,OAAA,EAAS,IAAA;AAAA,QACT,OAAA,EAAS,eAAe,IAAI,CAAA,aAAA,CAAA;AAAA,QAC5B,IAAA,EAAM,EAAE,EAAA,EAAI,IAAA,EAAM,UAAA;AAAW,OACjC;AAAA,IACJ;AAAA,IAEA,KAAK,YAAA,EAAc;AACf,MAAA,MAAM,IAAA,GAAO,MAAA,CAAO,IAAA,CAAK,IAAI,CAAA;AAC7B,MAAA,MAAM,aAAa,IAAA,CAAK,WAAA;AACxB,MAAA,MAAM,OAAO,IAAA,CAAK,IAAA;AAClB,MAAA,MAAM,UAAU,IAAA,CAAK,OAAA;AACrB,MAAA,MAAM,gBAAgB,IAAA,CAAK,OAAA;AAE3B,MAAA,MAAM,EAAA,GAAK,IAAA,CAAK,WAAA,EAAY,CAAE,OAAA,CAAQ,eAAe,GAAG,CAAA,CAAE,OAAA,CAAQ,QAAA,EAAU,EAAE,CAAA;AAE9E,MAAA,MAAM,QAAA,GAAW,WAAA,CAAY,OAAA,CAAQ,SAAA,CAAU,EAAE,CAAA;AACjD,MAAA,IAAI,QAAA,EAAU;AACV,QAAA,OAAO;AAAA,UACH,OAAA,EAAS,KAAA;AAAA,UACT,OAAA,EAAS,WAAW,IAAI,CAAA,4BAAA;AAAA,SAC5B;AAAA,MACJ;AAEA,MAAA,MAAM,SAAA,GAA4B;AAAA,QAC9B,EAAA;AAAA,QACA,IAAA;AAAA,QACA,IAAA,EAAM,QAAA;AAAA,QACN,WAAA,EAAa,UAAA;AAAA,QACb,IAAA;AAAA,QACA,OAAA;AAAA,QACA,OAAA,EAAS;AAAA,OACb;AAEA,MAAA,IAAI,CAAC,YAAY,MAAA,EAAQ;AACrB,QAAA,MAAM,WAAA,CAAY,OAAA,CAAQ,UAAA,CAAW,SAAS,CAAA;AAAA,MAClD;AAEA,MAAA,WAAA,CAAY,QAAQ,IAAA,CAAK;AAAA,QACrB,IAAA,EAAM,cAAA;AAAA,QACN,WAAA,EAAa,iBAAiB,IAAI,CAAA,YAAA,CAAA;AAAA,QAClC,SAAS,EAAE,IAAA,EAAM,WAAA,EAAa,UAAA,EAAY,MAAM,OAAA;AAAQ,OAC3D,CAAA;AAED,MAAA,OAAO;AAAA,QACH,OAAA,EAAS,IAAA;AAAA,QACT,OAAA,EAAS,iBAAiB,IAAI,CAAA,aAAA,CAAA;AAAA,QAC9B,IAAA,EAAM,EAAE,EAAA,EAAI,IAAA,EAAM,aAAa,UAAA;AAAW,OAC9C;AAAA,IACJ;AAAA,IAEA,KAAK,gBAAA,EAAkB;AACnB,MAAA,MAAM,SAAA,GAAY,MAAA,CAAO,IAAA,CAAK,UAAU,CAAA;AAExC,MAAA,MAAM,OAAA,GAAU,WAAA,CAAY,OAAA,CAAQ,UAAA,CAAW,SAAS,CAAA;AACxD,MAAA,IAAI,CAAC,OAAA,EAAS;AACV,QAAA,MAAM,SAAA,GAAY,YAAY,OAAA,CAAQ,cAAA,GAAiB,GAAA,CAAI,CAAA,CAAA,KAAK,EAAE,EAAE,CAAA;AACpE,QAAA,OAAO;AAAA,UACH,OAAA,EAAS,KAAA;AAAA,UACT,SAAS,CAAA,SAAA,EAAY,SAAS,2BAA2B,SAAA,CAAU,IAAA,CAAK,IAAI,CAAC,CAAA;AAAA,SACjF;AAAA,MACJ;AAEA,MAAA,MAAM,aAAA,GAAgB,qBAAA;AACtB,MAAA,MAAM,cAAA,GAAiB,0BAAA;AAEvB,MAAA,IAAI,aAAA,CAAc,IAAA,CAAK,WAAA,CAAY,iBAAiB,CAAA,EAAG;AACnD,QAAA,WAAA,CAAY,iBAAA,GAAoB,YAAY,iBAAA,CAAkB,OAAA;AAAA,UAC1D,aAAA;AAAA,UACA,CAAA,aAAA,EAAgB,QAAQ,IAAI,CAAA;AAAA,SAChC;AAAA,MACJ;AAEA,MAAA,IAAI,cAAA,CAAe,IAAA,CAAK,WAAA,CAAY,iBAAiB,CAAA,EAAG;AACpD,QAAA,WAAA,CAAY,iBAAA,GAAoB,YAAY,iBAAA,CAAkB,OAAA;AAAA,UAC1D,cAAA;AAAA,UACA,CAAA,kBAAA,EAAqB,QAAQ,EAAE,CAAA,EAAA;AAAA,SACnC;AAAA,MACJ;AAEA,MAAA,WAAA,CAAY,QAAQ,IAAA,CAAK;AAAA,QACrB,IAAA,EAAM,iBAAA;AAAA,QACN,aAAa,CAAA,oBAAA,EAAuB,OAAA,CAAQ,IAAI,CAAA,GAAA,EAAM,QAAQ,EAAE,CAAA,CAAA,CAAA;AAAA,QAChE,OAAA,EAAS,EAAE,UAAA,EAAY,SAAA,EAAW,cAAc,OAAA,CAAQ,IAAA,EAAM,OAAA,EAAS,OAAA,CAAQ,OAAA;AAAQ,OAC1F,CAAA;AAED,MAAA,OAAO;AAAA,QACH,OAAA,EAAS,IAAA;AAAA,QACT,OAAA,EAAS,CAAA,oBAAA,EAAuB,OAAA,CAAQ,IAAI,CAAA,EAAA,CAAA;AAAA,QAC5C,MAAM,EAAE,UAAA,EAAY,WAAW,WAAA,EAAa,OAAA,CAAQ,SAAS,WAAA;AAAY,OAC7E;AAAA,IACJ;AAAA,IAEA,KAAK,cAAA,EAAgB;AACjB,MAAA,MAAM,QAAA,GAAW,MAAA,CAAO,IAAA,CAAK,SAAS,CAAA;AAItC,MAAA,WAAA,CAAY,QAAQ,IAAA,CAAK;AAAA,QACrB,IAAA,EAAM,eAAA;AAAA,QACN,WAAA,EAAa,qBAAqB,QAAQ,CAAA,CAAA,CAAA;AAAA,QAC1C,SAAS,EAAE,SAAA,EAAW,UAAU,IAAA,EAAM,YAAA,CAAa,QAAQ,CAAA;AAAE,OAChE,CAAA;AAED,MAAA,OAAO;AAAA,QACH,OAAA,EAAS,IAAA;AAAA,QACT,OAAA,EAAS,qBAAqB,QAAQ,CAAA,EAAA,CAAA;AAAA,QACtC,IAAA,EAAM,EAAE,SAAA,EAAW,QAAA;AAAS,OAChC;AAAA,IACJ;AAAA,IAEA,KAAK,cAAA,EAAgB;AACjB,MAAA,MAAM,KAAA,GAAQ,MAAA,CAAO,IAAA,CAAK,KAAA,IAAS,SAAS,CAAA;AAC5C,MAAA,IAAI,QAAA,GAAW,EAAA;AAEf,MAAA,QAAQ,KAAA;AAAO,QACX,KAAK,OAAA;AACD,UAAA,QAAA,GAAW,sEAAA;AACX,UAAA;AAAA,QACJ,KAAK,QAAA;AACD,UAAA,QAAA,GAAW,yEAAA;AACX,UAAA;AAAA,QACJ,KAAK,UAAA;AACD,UAAA,QAAA,GAAW,iEAAA;AACX,UAAA;AAAA,QACJ,KAAK,aAAA;AACD,UAAA,QAAA,GAAW,mDAAA;AACX,UAAA;AAAA,QACJ;AACI,UAAA,QAAA,GAAW,kEAAA;AAAA;AAGnB,MAAA,OAAO;AAAA,QACH,OAAA,EAAS,IAAA;AAAA,QACT,OAAA,EAAS;AAAA,OACb;AAAA,IACJ;AAAA,IAEA,KAAK,UAAA,EAAY;AACb,MAAA,MAAM,OAAA,GAAU,MAAA,CAAO,IAAA,CAAK,OAAO,CAAA;AACnC,MAAA,OAAO;AAAA,QACH,OAAA,EAAS,IAAA;AAAA,QACT,OAAA,EAAS,OAAA;AAAA,QACT,IAAA,EAAM,EAAE,QAAA,EAAU,IAAA;AAAK,OAC3B;AAAA,IACJ;AAAA,IAEA;AACI,MAAA,MAAA,CAAO,IAAA,CAAK,oBAAoB,QAAQ,CAAA;AACxC,MAAA,OAAO;AAAA,QACH,OAAA,EAAS,KAAA;AAAA,QACT,OAAA,EAAS,iBAAiB,QAAQ,CAAA;AAAA,OACtC;AAAA;AAEZ;AAKO,MAAM,yBAAA,GAA4B,CACrC,iBAAA,EACA,iBAAA,KACS;AACT,EAAA,MAAM,mBAAmB,cAAA,CAAe,GAAA;AAAA,IAAI,OACxC,CAAA,EAAA,EAAK,CAAA,CAAE,IAAI,CAAA,EAAA,EAAK,EAAE,WAAW,CAAA;AAAA,GACjC,CAAE,KAAK,IAAI,CAAA;AAEX,EAAA,OAAO,CAAA;;AAAA;AAAA,EAGT,iBAAA,CAAkB,SAAA,CAAU,CAAA,EAAG,GAAI,CAAC,GAAG,iBAAA,CAAkB,MAAA,GAAS,GAAA,GAAO,KAAA,GAAQ,EAAE;;AAAA;AAAA,EAGnF,kBAAkB,MAAA,GAAS,CAAA,GAAI,kBAAkB,IAAA,CAAK,IAAI,IAAI,0BAA0B;;AAAA;AAAA,EAGxF,gBAAgB;;AAAA;AAAA;AAAA;AAAA;AAAA,iEAAA,CAAA;AAOlB;AAKO,MAAM,eAAA,GAAkB,OAC3B,QAAA,EACA,WAAA,EACA,SAAA,KACgB;AAChB,EAAA,MAAM,MAAA,GAASA,SAAQ,EAAU;AAEjC,EAAA,MAAM,QAAA,GAAW,WAAA,CAAY,OAAA,CAAQ,cAAA,EAAe,CAAE,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,EAAG,CAAA,CAAE,EAAE,CAAA,EAAA,EAAK,CAAA,CAAE,IAAI,CAAA,CAAA,CAAG,CAAA;AACpF,EAAA,MAAM,YAAA,GAAe,yBAAA,CAA0B,WAAA,CAAY,iBAAA,EAAmB,QAAQ,CAAA;AAEtF,EAAA,MAAM,KAAA,GAAQ,cAAA,CAAe,GAAA,CAAI,CAAA,CAAA,MAAM;AAAA,IACnC,IAAA,EAAM,UAAA;AAAA,IACN,QAAA,EAAU;AAAA,MACN,MAAM,CAAA,CAAE,IAAA;AAAA,MACR,aAAa,CAAA,CAAE,WAAA;AAAA,MACf,UAAA,EAAY;AAAA,QACR,IAAA,EAAM,QAAA;AAAA,QACN,YAAY,MAAA,CAAO,WAAA;AAAA,UACf,MAAA,CAAO,OAAA,CAAQ,CAAA,CAAE,UAAU,CAAA,CAAE,IAAI,CAAC,CAAC,GAAA,EAAK,KAAK,CAAA,KAAM;AAAA,YAC/C,GAAA;AAAA,YACA;AAAA,cACI,MAAM,KAAA,CAAM,IAAA;AAAA,cACZ,aAAa,KAAA,CAAM,WAAA;AAAA,cACnB,GAAI,MAAM,IAAA,GAAO,EAAE,MAAM,KAAA,CAAM,IAAA,KAAS,EAAC;AAAA,cACzC,GAAI,MAAM,KAAA,GAAQ,EAAE,OAAO,KAAA,CAAM,KAAA,KAAU;AAAC;AAChD,WACH;AAAA,SACL;AAAA,QACA,QAAA,EAAU,OAAO,OAAA,CAAQ,CAAA,CAAE,UAAU,CAAA,CAChC,MAAA,CAAO,CAAC,CAAC,CAAA,EAAG,CAAC,CAAA,KAAM,CAAA,CAAE,QAAQ,CAAA,CAC7B,GAAA,CAAI,CAAC,CAAC,GAAG,MAAM,GAAG;AAAA;AAC3B;AACJ,GACJ,CAAE,CAAA;AAEF,EAAA,IAAI,UAAA,GAAa,CAAA;AACjB,EAAA,MAAM,aAAA,GAAgB,EAAA;AACtB,EAAA,MAAM,mBAAA,GAKD;AAAA,IACD,EAAE,IAAA,EAAM,QAAA,EAAU,OAAA,EAAS,YAAA,EAAa;AAAA,IACxC,EAAE,IAAA,EAAM,MAAA,EAAQ,OAAA,EAAS,QAAA;AAAS,GACtC;AAEA,EAAA,OAAO,aAAa,aAAA,EAAe;AAC/B,IAAA,UAAA,EAAA;AACA,IAAA,MAAA,CAAO,KAAA,CAAM,oCAAoC,UAAU,CAAA;AAE3D,IAAA,IAAI;AACA,MAAA,MAAM,QAAA,GAAW,MAAM,SAAA,CAAU,iBAAA,CAAkB;AAAA,QAC/C,QAAA,EAAU,mBAAA;AAAA,QACV;AAAA,OACH,CAAA;AAED,MAAA,IAAI,QAAA,CAAS,UAAA,IAAc,QAAA,CAAS,UAAA,CAAW,SAAS,CAAA,EAAG;AACvD,QAAA,mBAAA,CAAoB,IAAA,CAAK;AAAA,UACrB,IAAA,EAAM,WAAA;AAAA,UACN,OAAA,EAAS,SAAS,OAAA,IAAW,EAAA;AAAA,UAC7B,UAAA,EAAY,QAAA,CAAS,UAAA,CAAW,GAAA,CAAI,CAAA,EAAA,MAAO;AAAA,YACvC,IAAI,EAAA,CAAG,EAAA;AAAA,YACP,QAAA,EAAU;AAAA,cACN,IAAA,EAAM,GAAG,QAAA,CAAS,IAAA;AAAA,cAClB,SAAA,EAAW,GAAG,QAAA,CAAS;AAAA;AAC3B,WACJ,CAAE;AAAA,SACL,CAAA;AAED,QAAA,KAAA,MAAW,QAAA,IAAY,SAAS,UAAA,EAAY;AACxC,UAAA,MAAM,QAAA,GAAW,SAAS,QAAA,CAAS,IAAA;AACnC,UAAA,IAAI,IAAA;AAEJ,UAAA,IAAI;AACA,YAAA,IAAA,GAAO,IAAA,CAAK,KAAA,CAAM,QAAA,CAAS,QAAA,CAAS,SAAS,CAAA;AAAA,UACjD,CAAA,CAAA,MAAQ;AACJ,YAAA,IAAA,GAAO,EAAC;AAAA,UACZ;AAEA,UAAA,MAAM,MAAA,GAAS,MAAM,WAAA,CAAY,QAAA,EAAU,MAAM,WAAW,CAAA;AAE5D,UAAA,mBAAA,CAAoB,IAAA,CAAK;AAAA,YACrB,IAAA,EAAM,MAAA;AAAA,YACN,OAAA,EAAS,IAAA,CAAK,SAAA,CAAU,MAAM,CAAA;AAAA,YAC9B,cAAc,QAAA,CAAS;AAAA,WAC1B,CAAA;AAED,UAAA,IAAI,aAAa,UAAA,EAAY;AACzB,YAAA;AAAA,UACJ;AAAA,QACJ;AAAA,MACJ,CAAA,MAAO;AACH,QAAA;AAAA,MACJ;AAAA,IACJ,SAAS,KAAA,EAAO;AACZ,MAAA,MAAA,CAAO,KAAA,CAAM,kCAAA,EAAoC,EAAE,KAAA,EAAO,CAAA;AAC1D,MAAA,MAAM,KAAA;AAAA,IACV;AAAA,EACJ;AAEA,EAAA,MAAA,CAAO,KAAK,4CAA4C,CAAA;AAC5D;AAKO,MAAM,YAAA,GAAe,OACxB,WAAA,KAC+C;AAC/C,EAAA,MAAM,MAAA,GAASA,SAAQ,EAAU;AAGjC,EAAA,MAAM,OAAA,GAAU,kBAAA,CAAmB,WAAA,CAAY,cAAc,CAAA;AAC7D,EAAA,IAAI,OAAA,GAAU,OAAA;AACd,EAAA,IAAI,KAAA,GAAQ,KAAA;AAEZ,EAAA,MAAM,cAAc,WAAA,CAAY,OAAA,CAAQ,KAAK,CAAA,CAAA,KAAK,CAAA,CAAE,SAAS,eAAe,CAAA;AAC5E,EAAA,IAAI,WAAA,EAAa;AACb,IAAA,MAAM,IAAA,GAAO,YAAY,OAAA,CAAQ,IAAA;AACjC,IAAA,MAAM,SAAA,GAAY,6BAA6B,OAAO,CAAA;AACtD,IAAA,MAAM,GAAA,GAAM,IAAA,CAAK,OAAA,CAAQ,OAAO,CAAA;AAEhC,IAAA,IAAI,SAAA,EAAW;AACX,MAAA,MAAM,UAAU,CAAA,EAAG,SAAA,CAAU,KAAK,QAAA,EAAS,CAAE,SAAS,CAAA,EAAG,GAAG,CAAC,CAAA,EAAG,UAAU,MAAA,CAAO,QAAA,GAAW,QAAA,CAAS,CAAA,EAAG,GAAG,CAAC,CAAA,CAAA;AAC5G,MAAA,OAAA,GAAU,IAAA,CAAK,IAAA,CAAK,GAAA,EAAK,CAAA,EAAG,SAAA,CAAU,GAAG,CAAA,CAAA,EAAI,OAAO,CAAA,CAAA,EAAI,IAAI,CAAA,IAAA,CAAM,CAAA;AAAA,IACtE,CAAA,MAAO;AACH,MAAA,OAAA,GAAU,IAAA,CAAK,IAAA,CAAK,GAAA,EAAK,CAAA,EAAG,IAAI,CAAA,IAAA,CAAM,CAAA;AAAA,IAC1C;AAAA,EACJ;AAEA,EAAA,MAAM,gBAAgB,WAAA,CAAY,OAAA,CAAQ,KAAK,CAAA,CAAA,KAAK,CAAA,CAAE,SAAS,iBAAiB,CAAA;AAChF,EAAA,IAAI,aAAA,IAAiB,aAAA,CAAc,OAAA,CAAQ,OAAA,EAAS;AAChD,IAAA,MAAM,OAAA,GAAU,cAAc,OAAA,CAAQ,OAAA;AACtC,IAAA,IAAI,QAAQ,WAAA,EAAa;AACrB,MAAA,IAAI,OAAO,OAAA,CAAQ,WAAA;AACnB,MAAA,IAAI,IAAA,CAAK,UAAA,CAAW,GAAG,CAAA,EAAG;AACtB,QAAA,IAAA,GAAO,IAAA,CAAK,KAAK,OAAA,CAAQ,GAAA,CAAI,QAAQ,EAAA,EAAI,IAAA,CAAK,KAAA,CAAM,CAAC,CAAC,CAAA;AAAA,MAC1D;AAEA,MAAA,MAAM,GAAA,uBAAU,IAAA,EAAK;AACrB,MAAA,MAAM,IAAA,GAAO,GAAA,CAAI,WAAA,EAAY,CAAE,QAAA,EAAS;AACxC,MAAA,MAAM,KAAA,GAAA,CAAS,IAAI,QAAA,EAAS,GAAI,GAAG,QAAA,EAAS,CAAE,QAAA,CAAS,CAAA,EAAG,GAAG,CAAA;AAE7D,MAAA,IAAI,cAAA,GAAiB,IAAA;AACrB,MAAA,MAAM,SAAA,GAAY,QAAQ,SAAA,IAAa,OAAA;AACvC,MAAA,IAAI,cAAc,MAAA,EAAQ;AACtB,QAAA,cAAA,GAAiB,IAAA,CAAK,IAAA,CAAK,IAAA,EAAM,IAAI,CAAA;AAAA,MACzC,CAAA,MAAA,IAAW,cAAc,OAAA,EAAS;AAC9B,QAAA,cAAA,GAAiB,IAAA,CAAK,IAAA,CAAK,IAAA,EAAM,IAAA,EAAM,KAAK,CAAA;AAAA,MAChD,CAAA,MAAA,IAAW,cAAc,KAAA,EAAO;AAC5B,QAAA,MAAM,GAAA,GAAM,IAAI,OAAA,EAAQ,CAAE,UAAS,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACpD,QAAA,cAAA,GAAiB,IAAA,CAAK,IAAA,CAAK,IAAA,EAAM,IAAA,EAAM,OAAO,GAAG,CAAA;AAAA,MACrD;AAEA,MAAA,MAAM,QAAA,GAAW,IAAA,CAAK,QAAA,CAAS,OAAO,CAAA;AACtC,MAAA,OAAA,GAAU,IAAA,CAAK,IAAA,CAAK,cAAA,EAAgB,QAAQ,CAAA;AAC5C,MAAA,KAAA,GAAQ,IAAA;AAAA,IACZ;AAAA,EACJ;AAEA,EAAA,MAAM,EAAA,CAAG,MAAM,IAAA,CAAK,OAAA,CAAQ,OAAO,CAAA,EAAG,EAAE,SAAA,EAAW,IAAA,EAAM,CAAA;AAEzD,EAAA,IAAI,CAAC,YAAY,MAAA,EAAQ;AAErB,IAAA,MAAM,aAAa,aAAA,CAAc,IAAA,CAAK,SAAS,EAAE,QAAA,EAAU,OAAO,CAAA;AAClE,IAAA,IAAI;AAEA,MAAA,IAAI,WAAA,CAAY,iBAAA,KAAsB,WAAA,CAAY,eAAA,EAAiB;AAC/D,QAAA,UAAA,CAAW,aAAA,CAAc,YAAY,iBAAiB,CAAA;AAAA,MAC1D;AAGA,MAAA,IAAI,WAAA,EAAa;AACb,QAAA,UAAA,CAAW,eAAe,EAAE,KAAA,EAAO,WAAA,CAAY,OAAA,CAAQ,WAAqB,CAAA;AAAA,MAChF;AAGA,MAAA,IAAI,aAAA,EAAe;AACf,QAAA,UAAA,CAAW,cAAA,CAAe;AAAA,UACtB,SAAA,EAAW,cAAc,OAAA,CAAQ,UAAA;AAAA,UACjC,OAAA,EAAS,cAAc,OAAA,CAAQ;AAAA,SAClC,CAAA;AAAA,MACL;AAAA,IACJ,CAAA,SAAE;AACE,MAAA,UAAA,CAAW,KAAA,EAAM;AAAA,IACrB;AAGA,IAAA,IAAI,YAAY,OAAA,EAAS;AACrB,MAAA,MAAM,EAAA,CAAG,QAAA,CAAS,OAAA,EAAS,OAAO,CAAA;AAClC,MAAA,MAAM,EAAA,CAAG,OAAO,OAAO,CAAA;AAAA,IAC3B;AAAA,EACJ;AAEA,EAAA,MAAA,CAAO,IAAA,CAAK,kCAAA,EAAoC,WAAA,CAAY,OAAA,CAAQ,MAAM,CAAA;AAE1E,EAAA,OAAO,EAAE,SAAS,KAAA,EAAM;AAC5B;;;;"}