@redaksjon/protokoll-engine 0.1.7 → 0.1.8-dev.20260221045945.bbdc773
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.
- package/dist/index33.js +1 -1
- package/dist/index34.js +2 -2
- package/dist/index35.js +4 -4
- package/dist/index36.js +1 -1
- package/dist/index38.js +12 -13
- package/dist/index38.js.map +1 -1
- package/dist/index53.js +48 -5
- package/dist/index53.js.map +1 -1
- package/dist/index54.js +34 -46
- package/dist/index54.js.map +1 -1
- package/dist/index55.js +280 -35
- package/dist/index55.js.map +1 -1
- package/dist/index56.js +137 -258
- package/dist/index56.js.map +1 -1
- package/dist/index57.js +60 -142
- package/dist/index57.js.map +1 -1
- package/dist/index58.js +70 -73
- package/dist/index58.js.map +1 -1
- package/dist/index59.js +3 -73
- package/dist/index59.js.map +1 -1
- package/dist/index60.js +5 -148
- package/dist/index60.js.map +1 -1
- package/dist/index61.js +4 -4
- package/dist/index62.js +148 -5
- package/dist/index62.js.map +1 -1
- package/dist/transcript/operations.d.ts.map +1 -1
- package/package.json +1 -1
package/dist/index33.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import * as Context from '@redaksjon/context';
|
|
2
2
|
import { create as create$1 } from './index6.js';
|
|
3
|
-
import { create as create$2 } from './
|
|
3
|
+
import { create as create$2 } from './index59.js';
|
|
4
4
|
import { create as create$3 } from './index11.js';
|
|
5
5
|
import { create as create$4 } from './index3.js';
|
|
6
6
|
import { create as create$5 } from './index2.js';
|
package/dist/index34.js
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import { getLogger } from './index47.js';
|
|
2
2
|
import { create as create$1 } from './index13.js';
|
|
3
3
|
import { create as create$2 } from './index14.js';
|
|
4
|
-
import { transcribeAudio } from './
|
|
5
|
-
import { stringifyJSON } from './
|
|
4
|
+
import { transcribeAudio } from './index53.js';
|
|
5
|
+
import { stringifyJSON } from './index54.js';
|
|
6
6
|
import path__default from 'node:path';
|
|
7
7
|
import { create as create$4 } from './index5.js';
|
|
8
8
|
import { create as create$3 } from './index2.js';
|
package/dist/index35.js
CHANGED
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import { getLogger } from './index47.js';
|
|
2
2
|
import { create as create$1 } from './index13.js';
|
|
3
|
-
import { create as create$2 } from './
|
|
4
|
-
import { create as create$3 } from './
|
|
5
|
-
import { create as create$4 } from './
|
|
6
|
-
import { stringifyJSON } from './
|
|
3
|
+
import { create as create$2 } from './index55.js';
|
|
4
|
+
import { create as create$3 } from './index56.js';
|
|
5
|
+
import { create as create$4 } from './index57.js';
|
|
6
|
+
import { stringifyJSON } from './index54.js';
|
|
7
7
|
import path from 'path';
|
|
8
8
|
|
|
9
9
|
const create = (config, contextInstance) => {
|
package/dist/index36.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { getLogger } from './index47.js';
|
|
2
2
|
import { create as create$3 } from './index14.js';
|
|
3
3
|
import { create as create$1 } from './index13.js';
|
|
4
|
-
import { create as create$2 } from './
|
|
4
|
+
import { create as create$2 } from './index58.js';
|
|
5
5
|
import { DEFAULT_INTERMEDIATE_DIRECTORY } from './index18.js';
|
|
6
6
|
import path__default from 'node:path';
|
|
7
7
|
|
package/dist/index38.js
CHANGED
|
@@ -7,6 +7,10 @@ import { findProjectResilient } from './index16.js';
|
|
|
7
7
|
import { PklTranscript, listTranscripts as listTranscripts$1 } from '@redaksjon/protokoll-format';
|
|
8
8
|
import { ensurePklExtension } from './index40.js';
|
|
9
9
|
|
|
10
|
+
const UUID_PATTERN = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i;
|
|
11
|
+
function looksLikeUuid(value) {
|
|
12
|
+
return UUID_PATTERN.test(value);
|
|
13
|
+
}
|
|
10
14
|
function isUuidInput(input) {
|
|
11
15
|
return /^[a-f0-9]{8}/.test(input);
|
|
12
16
|
}
|
|
@@ -413,25 +417,20 @@ const listTranscripts = async (options) => {
|
|
|
413
417
|
const transcript = PklTranscript.open(item.filePath, { readOnly: true });
|
|
414
418
|
const meta = transcript.metadata;
|
|
415
419
|
uuid = meta.id;
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
projects: meta.entities.projects?.map((e) => ({ id: e.id, name: e.name })),
|
|
420
|
-
terms: meta.entities.terms?.map((e) => ({ id: e.id, name: e.name })),
|
|
421
|
-
companies: meta.entities.companies?.map((e) => ({ id: e.id, name: e.name }))
|
|
422
|
-
};
|
|
423
|
-
} else if (item.project) {
|
|
420
|
+
const mappedProjects = meta.entities?.projects?.map((e) => ({ id: e.id, name: e.name }));
|
|
421
|
+
const projectEntries = mappedProjects && mappedProjects.length > 0 ? mappedProjects : item.project && !looksLikeUuid(item.project) ? [{ id: meta.projectId || item.project, name: item.project }] : void 0;
|
|
422
|
+
if (meta.entities || projectEntries) {
|
|
424
423
|
entities = {
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
}
|
|
424
|
+
people: meta.entities?.people?.map((e) => ({ id: e.id, name: e.name })),
|
|
425
|
+
projects: projectEntries,
|
|
426
|
+
terms: meta.entities?.terms?.map((e) => ({ id: e.id, name: e.name })),
|
|
427
|
+
companies: meta.entities?.companies?.map((e) => ({ id: e.id, name: e.name }))
|
|
429
428
|
};
|
|
430
429
|
}
|
|
431
430
|
transcript.close();
|
|
432
431
|
} catch {
|
|
433
432
|
uuid = "";
|
|
434
|
-
if (item.project) {
|
|
433
|
+
if (item.project && !looksLikeUuid(item.project)) {
|
|
435
434
|
entities = {
|
|
436
435
|
projects: [{ id: item.project, name: item.project }]
|
|
437
436
|
};
|
package/dist/index38.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index38.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, undefined);\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: 'enhanced',\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 if (targetProject?.routing?.destination) {\n const routingConfig = buildRoutingConfig(context, targetProject);\n const routing = Routing.create(routingConfig, context, undefined);\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?: import('@redaksjon/protokoll-format').TranscriptStatus;\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 /** Project name - used as fallback when projectId is also set (matches transcripts with project name but no projectId) */\n project?: 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 project,\n } = options;\n \n // Use the storage API from protokoll-format\n // Pass projectId for UUID-based filtering; project (name) as fallback for transcripts without projectId\n const storageOptions: StorageListOptions = {\n directory,\n limit,\n offset,\n sortBy,\n search,\n projectId,\n project,\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 let uuid = '';\n let entities: TranscriptListItem['entities'];\n try {\n const transcript = PklTranscript.open(item.filePath, { readOnly: true });\n const meta = transcript.metadata;\n uuid = meta.id;\n if (meta.entities) {\n entities = {\n people: meta.entities.people?.map(e => ({ id: e.id, name: e.name })),\n projects: meta.entities.projects?.map(e => ({ id: e.id, name: e.name })),\n terms: meta.entities.terms?.map(e => ({ id: e.id, name: e.name })),\n companies: meta.entities.companies?.map(e => ({ id: e.id, name: e.name })),\n };\n } else if (item.project) {\n entities = {\n projects: [{\n id: meta.projectId || item.project,\n name: item.project,\n }],\n };\n }\n transcript.close();\n } catch {\n uuid = '';\n if (item.project) {\n entities = {\n projects: [{ id: item.project, name: item.project }],\n };\n }\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,\n title: item.title,\n hasRawTranscript: false,\n createdAt: item.date || new Date(),\n status: item.status,\n openTasksCount: undefined,\n contentSize: item.contentPreview?.length,\n entities,\n };\n });\n \n return {\n transcripts,\n total: result.total,\n hasMore: result.hasMore,\n limit,\n offset,\n };\n};\n\n/**\n * Validate status transitions for transcript lifecycle\n * \n * Ensures status changes follow valid workflow:\n * - Upload workflow: uploaded → transcribing → initial → enhanced → reviewed → closed\n * - Error can occur at any point\n * - Error status allows retry (back to uploaded or transcribing)\n * \n * @param from - Current status\n * @param to - Desired status\n * @returns true if transition is valid, false otherwise\n */\nexport function isValidStatusTransition(\n from: import('@redaksjon/protokoll-format').TranscriptStatus | undefined,\n to: import('@redaksjon/protokoll-format').TranscriptStatus\n): boolean {\n // If no current status, any status is valid (initial creation)\n if (!from) {\n return true;\n }\n \n // Define valid transitions for each status\n const validTransitions: Record<\n import('@redaksjon/protokoll-format').TranscriptStatus,\n import('@redaksjon/protokoll-format').TranscriptStatus[]\n > = {\n 'uploaded': ['transcribing', 'error'],\n 'transcribing': ['initial', 'error'],\n 'error': ['uploaded', 'transcribing'], // Allow retry\n 'initial': ['enhanced', 'in_progress', 'error'],\n 'enhanced': ['reviewed', 'in_progress', 'error'],\n 'reviewed': ['closed', 'in_progress', 'error'],\n 'in_progress': ['initial', 'enhanced', 'reviewed', 'closed', 'error'],\n 'closed': ['archived', 'in_progress', 'error'],\n 'archived': ['closed', 'error'], // Allow un-archiving\n };\n \n return validTransitions[from]?.includes(to) ?? false;\n}\n"],"names":["Routing.create","listTranscriptsFromStorage"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index38.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/** UUID v4 pattern — used to detect corrupted project fields */\nconst UUID_PATTERN = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i;\n\nfunction looksLikeUuid(value: string): boolean {\n return UUID_PATTERN.test(value);\n}\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, undefined);\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: 'enhanced',\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 if (targetProject?.routing?.destination) {\n const routingConfig = buildRoutingConfig(context, targetProject);\n const routing = Routing.create(routingConfig, context, undefined);\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?: import('@redaksjon/protokoll-format').TranscriptStatus;\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 /** Project name - used as fallback when projectId is also set (matches transcripts with project name but no projectId) */\n project?: 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 project,\n } = options;\n \n // Use the storage API from protokoll-format\n // Pass projectId for UUID-based filtering; project (name) as fallback for transcripts without projectId\n const storageOptions: StorageListOptions = {\n directory,\n limit,\n offset,\n sortBy,\n search,\n projectId,\n project,\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 let uuid = '';\n let entities: TranscriptListItem['entities'];\n try {\n const transcript = PklTranscript.open(item.filePath, { readOnly: true });\n const meta = transcript.metadata;\n uuid = meta.id;\n const mappedProjects = meta.entities?.projects?.map(e => ({ id: e.id, name: e.name }));\n // If entities.projects is missing/empty but the scalar project field is set,\n // synthesise a project entry so the list view can display it correctly.\n const projectEntries = (mappedProjects && mappedProjects.length > 0)\n ? mappedProjects\n : (item.project && !looksLikeUuid(item.project))\n ? [{ id: meta.projectId || item.project, name: item.project }]\n : undefined;\n\n if (meta.entities || projectEntries) {\n entities = {\n people: meta.entities?.people?.map(e => ({ id: e.id, name: e.name })),\n projects: projectEntries,\n terms: meta.entities?.terms?.map(e => ({ id: e.id, name: e.name })),\n companies: meta.entities?.companies?.map(e => ({ id: e.id, name: e.name })),\n };\n }\n transcript.close();\n } catch {\n uuid = '';\n if (item.project && !looksLikeUuid(item.project)) {\n entities = {\n projects: [{ id: item.project, name: item.project }],\n };\n }\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,\n title: item.title,\n hasRawTranscript: false,\n createdAt: item.date || new Date(),\n status: item.status,\n openTasksCount: undefined,\n contentSize: item.contentPreview?.length,\n entities,\n };\n });\n \n return {\n transcripts,\n total: result.total,\n hasMore: result.hasMore,\n limit,\n offset,\n };\n};\n\n/**\n * Validate status transitions for transcript lifecycle\n * \n * Ensures status changes follow valid workflow:\n * - Upload workflow: uploaded → transcribing → initial → enhanced → reviewed → closed\n * - Error can occur at any point\n * - Error status allows retry (back to uploaded or transcribing)\n * \n * @param from - Current status\n * @param to - Desired status\n * @returns true if transition is valid, false otherwise\n */\nexport function isValidStatusTransition(\n from: import('@redaksjon/protokoll-format').TranscriptStatus | undefined,\n to: import('@redaksjon/protokoll-format').TranscriptStatus\n): boolean {\n // If no current status, any status is valid (initial creation)\n if (!from) {\n return true;\n }\n \n // Define valid transitions for each status\n const validTransitions: Record<\n import('@redaksjon/protokoll-format').TranscriptStatus,\n import('@redaksjon/protokoll-format').TranscriptStatus[]\n > = {\n 'uploaded': ['transcribing', 'error'],\n 'transcribing': ['initial', 'error'],\n 'error': ['uploaded', 'transcribing'], // Allow retry\n 'initial': ['enhanced', 'in_progress', 'error'],\n 'enhanced': ['reviewed', 'in_progress', 'error'],\n 'reviewed': ['closed', 'in_progress', 'error'],\n 'in_progress': ['initial', 'enhanced', 'reviewed', 'closed', 'error'],\n 'closed': ['archived', 'in_progress', 'error'],\n 'archived': ['closed', 'error'], // Allow un-archiving\n };\n \n return validTransitions[from]?.includes(to) ?? false;\n}\n"],"names":["Routing.create","listTranscriptsFromStorage"],"mappings":"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"}
|
package/dist/index53.js
CHANGED
|
@@ -1,8 +1,51 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { OpenAI } from 'openai';
|
|
2
|
+
import { create } from './index13.js';
|
|
3
|
+
import { getLogger } from './index47.js';
|
|
4
|
+
import { DEFAULT_TRANSCRIPTION_MODEL } from './index18.js';
|
|
2
5
|
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
+
class OpenAIError extends Error {
|
|
7
|
+
constructor(message) {
|
|
8
|
+
super(message);
|
|
9
|
+
this.name = "OpenAIError";
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
async function transcribeAudio(filePath, options = {}) {
|
|
13
|
+
const logger = getLogger();
|
|
14
|
+
const storage$1 = create({ log: logger.debug });
|
|
15
|
+
try {
|
|
16
|
+
const apiKey = process.env.OPENAI_API_KEY;
|
|
17
|
+
if (!apiKey) {
|
|
18
|
+
throw new OpenAIError("OPENAI_API_KEY environment variable is not set");
|
|
19
|
+
}
|
|
20
|
+
const openai = new OpenAI({
|
|
21
|
+
apiKey
|
|
22
|
+
});
|
|
23
|
+
const model = options.model || DEFAULT_TRANSCRIPTION_MODEL;
|
|
24
|
+
const fileName = filePath.split("/").pop() || filePath;
|
|
25
|
+
logger.debug("Transcribing: %s (full path: %s)", fileName, filePath);
|
|
26
|
+
const startTime = Date.now();
|
|
27
|
+
const audioStream = await storage$1.readStream(filePath);
|
|
28
|
+
const transcription = await openai.audio.transcriptions.create({
|
|
29
|
+
model,
|
|
30
|
+
file: audioStream,
|
|
31
|
+
response_format: "json"
|
|
32
|
+
});
|
|
33
|
+
if (!transcription) {
|
|
34
|
+
throw new OpenAIError("No transcription received from OpenAI");
|
|
35
|
+
}
|
|
36
|
+
const duration = ((Date.now() - startTime) / 1e3).toFixed(1);
|
|
37
|
+
logger.info("%s (%ss, %d chars)", model, duration, transcription.text?.length || 0);
|
|
38
|
+
if (options.debug && options.debugFile) {
|
|
39
|
+
await storage$1.writeFile(options.debugFile, JSON.stringify(transcription, null, 2), "utf8");
|
|
40
|
+
logger.debug("Wrote debug file to %s", options.debugFile);
|
|
41
|
+
}
|
|
42
|
+
logger.debug("Received transcription from OpenAI: %s", transcription);
|
|
43
|
+
return transcription;
|
|
44
|
+
} catch (error) {
|
|
45
|
+
logger.error("Error transcribing audio file: %s %s", error.message, error.stack);
|
|
46
|
+
throw new OpenAIError(`Failed to transcribe audio: ${error.message}`);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
6
49
|
|
|
7
|
-
export {
|
|
50
|
+
export { OpenAIError, transcribeAudio };
|
|
8
51
|
//# sourceMappingURL=index53.js.map
|
package/dist/index53.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index53.js","sources":["../src/
|
|
1
|
+
{"version":3,"file":"index53.js","sources":["../src/util/openai.ts"],"sourcesContent":["import { OpenAI } from 'openai';\nimport { ChatCompletionCreateParamsNonStreaming, ChatCompletionMessageParam } from 'openai/resources/chat/completions';\nimport * as Storage from '@/util/storage';\nimport { getLogger } from '@/logging';\nimport { DEFAULT_MODEL, DEFAULT_TRANSCRIPTION_MODEL } from '@/constants';\n\nexport interface Transcription {\n text: string;\n}\n\nexport class OpenAIError extends Error {\n constructor(message: string) {\n super(message);\n this.name = 'OpenAIError';\n }\n}\n\n\nexport async function createCompletion(messages: ChatCompletionMessageParam[], options: { responseFormat?: any, model?: string, reasoningLevel?: 'none' | 'low' | 'medium' | 'high', maxTokens?: number, debug?: boolean, debugFile?: string, reason?: string } = {}): Promise<string | any> {\n const logger = getLogger();\n const storage = Storage.create({ log: logger.debug });\n try {\n const apiKey = process.env.OPENAI_API_KEY;\n if (!apiKey) {\n throw new OpenAIError('OPENAI_API_KEY environment variable is not set');\n }\n\n const openai = new OpenAI({\n apiKey: apiKey,\n });\n\n const model = options.model || DEFAULT_MODEL;\n \n // Check if model supports reasoning_effort\n const supportsReasoning = model.includes('gpt-5') || \n model.includes('o1') || model.includes('o3');\n const isReasoningCall = supportsReasoning && options.reasoningLevel && options.reasoningLevel !== 'none';\n \n logger.debug('Sending prompt to OpenAI: %j', messages);\n\n const startTime = Date.now();\n \n const requestParams: Record<string, unknown> = {\n model,\n messages,\n max_completion_tokens: options.maxTokens || 10000,\n response_format: options.responseFormat,\n };\n \n if (isReasoningCall) {\n requestParams.reasoning_effort = options.reasoningLevel;\n logger.debug('Using reasoning_effort: %s', options.reasoningLevel);\n }\n \n const completion = await openai.chat.completions.create(\n requestParams as unknown as ChatCompletionCreateParamsNonStreaming\n );\n const duration = ((Date.now() - startTime) / 1000).toFixed(1);\n\n // Log token usage with reason if provided\n const usage = completion.usage;\n const reasonSuffix = options.reason ? ` - ${options.reason}` : '';\n if (usage) {\n logger.info('%s (%ss, %d→%d tokens)%s', \n model, duration, usage.prompt_tokens, usage.completion_tokens, reasonSuffix);\n } else {\n logger.info('%s (%ss)%s', model, duration, reasonSuffix);\n }\n\n if (options.debug && options.debugFile) {\n await storage.writeFile(options.debugFile, JSON.stringify(completion, null, 2), 'utf8');\n logger.debug('Wrote debug file to %s', options.debugFile);\n }\n\n const response = completion.choices[0]?.message?.content?.trim();\n if (!response) {\n // Log the full completion object to help debug\n logger.error('Empty response from OpenAI. Full completion object: %j', completion);\n throw new OpenAIError('No response received from OpenAI');\n }\n\n logger.debug('Received response from OpenAI: %s', response);\n if (options.responseFormat) {\n return JSON.parse(response);\n } else {\n return response;\n }\n\n } catch (error: any) {\n logger.error('Error calling OpenAI API: %s %s', error.message, error.stack);\n throw new OpenAIError(`Failed to create completion: ${error.message}`);\n }\n}\n\nexport async function transcribeAudio(filePath: string, options: { model?: string, debug?: boolean, debugFile?: string } = {}): Promise<Transcription> {\n const logger = getLogger();\n const storage = Storage.create({ log: logger.debug });\n try {\n const apiKey = process.env.OPENAI_API_KEY;\n if (!apiKey) {\n throw new OpenAIError('OPENAI_API_KEY environment variable is not set');\n }\n\n const openai = new OpenAI({\n apiKey: apiKey,\n });\n\n const model = options.model || DEFAULT_TRANSCRIPTION_MODEL;\n const fileName = filePath.split('/').pop() || filePath;\n logger.debug('Transcribing: %s (full path: %s)', fileName, filePath);\n\n const startTime = Date.now();\n const audioStream = await storage.readStream(filePath);\n const transcription = await openai.audio.transcriptions.create({\n model,\n file: audioStream,\n response_format: \"json\",\n });\n \n if (!transcription) {\n throw new OpenAIError('No transcription received from OpenAI');\n }\n \n const duration = ((Date.now() - startTime) / 1000).toFixed(1);\n logger.info('%s (%ss, %d chars)', model, duration, transcription.text?.length || 0);\n\n if (options.debug && options.debugFile) {\n await storage.writeFile(options.debugFile, JSON.stringify(transcription, null, 2), 'utf8');\n logger.debug('Wrote debug file to %s', options.debugFile);\n }\n\n logger.debug('Received transcription from OpenAI: %s', transcription);\n return transcription;\n\n } catch (error: any) {\n logger.error('Error transcribing audio file: %s %s', error.message, error.stack);\n throw new OpenAIError(`Failed to transcribe audio: ${error.message}`);\n }\n}\n"],"names":["storage","Storage.create"],"mappings":";;;;;AAUO,MAAM,oBAAoB,KAAA,CAAM;AAAA,EACnC,YAAY,OAAA,EAAiB;AACzB,IAAA,KAAA,CAAM,OAAO,CAAA;AACb,IAAA,IAAA,CAAK,IAAA,GAAO,aAAA;AAAA,EAChB;AACJ;AA+EA,eAAsB,eAAA,CAAgB,QAAA,EAAkB,OAAA,GAAmE,EAAC,EAA2B;AACnJ,EAAA,MAAM,SAAS,SAAA,EAAU;AACzB,EAAA,MAAMA,YAAUC,MAAQ,CAAO,EAAE,GAAA,EAAK,MAAA,CAAO,OAAO,CAAA;AACpD,EAAA,IAAI;AACA,IAAA,MAAM,MAAA,GAAS,QAAQ,GAAA,CAAI,cAAA;AAC3B,IAAA,IAAI,CAAC,MAAA,EAAQ;AACT,MAAA,MAAM,IAAI,YAAY,gDAAgD,CAAA;AAAA,IAC1E;AAEA,IAAA,MAAM,MAAA,GAAS,IAAI,MAAA,CAAO;AAAA,MACtB;AAAA,KACH,CAAA;AAED,IAAA,MAAM,KAAA,GAAQ,QAAQ,KAAA,IAAS,2BAAA;AAC/B,IAAA,MAAM,WAAW,QAAA,CAAS,KAAA,CAAM,GAAG,CAAA,CAAE,KAAI,IAAK,QAAA;AAC9C,IAAA,MAAA,CAAO,KAAA,CAAM,kCAAA,EAAoC,QAAA,EAAU,QAAQ,CAAA;AAEnE,IAAA,MAAM,SAAA,GAAY,KAAK,GAAA,EAAI;AAC3B,IAAA,MAAM,WAAA,GAAc,MAAMD,SAAA,CAAQ,UAAA,CAAW,QAAQ,CAAA;AACrD,IAAA,MAAM,aAAA,GAAgB,MAAM,MAAA,CAAO,KAAA,CAAM,eAAe,MAAA,CAAO;AAAA,MAC3D,KAAA;AAAA,MACA,IAAA,EAAM,WAAA;AAAA,MACN,eAAA,EAAiB;AAAA,KACpB,CAAA;AAED,IAAA,IAAI,CAAC,aAAA,EAAe;AAChB,MAAA,MAAM,IAAI,YAAY,uCAAuC,CAAA;AAAA,IACjE;AAEA,IAAA,MAAM,aAAa,IAAA,CAAK,GAAA,KAAQ,SAAA,IAAa,GAAA,EAAM,QAAQ,CAAC,CAAA;AAC5D,IAAA,MAAA,CAAO,KAAK,oBAAA,EAAsB,KAAA,EAAO,UAAU,aAAA,CAAc,IAAA,EAAM,UAAU,CAAC,CAAA;AAElF,IAAA,IAAI,OAAA,CAAQ,KAAA,IAAS,OAAA,CAAQ,SAAA,EAAW;AACpC,MAAA,MAAMA,SAAA,CAAQ,SAAA,CAAU,OAAA,CAAQ,SAAA,EAAW,IAAA,CAAK,UAAU,aAAA,EAAe,IAAA,EAAM,CAAC,CAAA,EAAG,MAAM,CAAA;AACzF,MAAA,MAAA,CAAO,KAAA,CAAM,wBAAA,EAA0B,OAAA,CAAQ,SAAS,CAAA;AAAA,IAC5D;AAEA,IAAA,MAAA,CAAO,KAAA,CAAM,0CAA0C,aAAa,CAAA;AACpE,IAAA,OAAO,aAAA;AAAA,EAEX,SAAS,KAAA,EAAY;AACjB,IAAA,MAAA,CAAO,KAAA,CAAM,sCAAA,EAAwC,KAAA,CAAM,OAAA,EAAS,MAAM,KAAK,CAAA;AAC/E,IAAA,MAAM,IAAI,WAAA,CAAY,CAAA,4BAAA,EAA+B,KAAA,CAAM,OAAO,CAAA,CAAE,CAAA;AAAA,EACxE;AACJ;;;;"}
|
package/dist/index54.js
CHANGED
|
@@ -1,51 +1,39 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
if (!apiKey) {
|
|
18
|
-
throw new OpenAIError("OPENAI_API_KEY environment variable is not set");
|
|
1
|
+
const stringifyJSON = function(obj) {
|
|
2
|
+
const arrOfKeyVals = [];
|
|
3
|
+
const arrVals = [];
|
|
4
|
+
let objKeys = [];
|
|
5
|
+
if (typeof obj === "number" || typeof obj === "boolean" || obj === null)
|
|
6
|
+
return "" + obj;
|
|
7
|
+
else if (typeof obj === "string")
|
|
8
|
+
return '"' + obj + '"';
|
|
9
|
+
else if (Array.isArray(obj)) {
|
|
10
|
+
if (obj[0] === void 0)
|
|
11
|
+
return "[]";
|
|
12
|
+
else {
|
|
13
|
+
obj.forEach(function(el) {
|
|
14
|
+
arrVals.push(stringifyJSON(el));
|
|
15
|
+
});
|
|
16
|
+
return "[" + arrVals + "]";
|
|
19
17
|
}
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
18
|
+
} else if (obj instanceof Object) {
|
|
19
|
+
objKeys = Object.keys(obj);
|
|
20
|
+
objKeys.forEach(function(key) {
|
|
21
|
+
const keyOut = '"' + key + '":';
|
|
22
|
+
const keyValOut = obj[key];
|
|
23
|
+
if (keyValOut instanceof Function || keyValOut === void 0)
|
|
24
|
+
arrOfKeyVals.push("");
|
|
25
|
+
else if (typeof keyValOut === "string")
|
|
26
|
+
arrOfKeyVals.push(keyOut + '"' + keyValOut + '"');
|
|
27
|
+
else if (typeof keyValOut === "boolean" || typeof keyValOut === "number" || keyValOut === null)
|
|
28
|
+
arrOfKeyVals.push(keyOut + keyValOut);
|
|
29
|
+
else if (keyValOut instanceof Object) {
|
|
30
|
+
arrOfKeyVals.push(keyOut + stringifyJSON(keyValOut));
|
|
31
|
+
}
|
|
32
32
|
});
|
|
33
|
-
|
|
34
|
-
throw new OpenAIError("No transcription received from OpenAI");
|
|
35
|
-
}
|
|
36
|
-
const duration = ((Date.now() - startTime) / 1e3).toFixed(1);
|
|
37
|
-
logger.info("%s (%ss, %d chars)", model, duration, transcription.text?.length || 0);
|
|
38
|
-
if (options.debug && options.debugFile) {
|
|
39
|
-
await storage$1.writeFile(options.debugFile, JSON.stringify(transcription, null, 2), "utf8");
|
|
40
|
-
logger.debug("Wrote debug file to %s", options.debugFile);
|
|
41
|
-
}
|
|
42
|
-
logger.debug("Received transcription from OpenAI: %s", transcription);
|
|
43
|
-
return transcription;
|
|
44
|
-
} catch (error) {
|
|
45
|
-
logger.error("Error transcribing audio file: %s %s", error.message, error.stack);
|
|
46
|
-
throw new OpenAIError(`Failed to transcribe audio: ${error.message}`);
|
|
33
|
+
return "{" + arrOfKeyVals + "}";
|
|
47
34
|
}
|
|
48
|
-
|
|
35
|
+
return "";
|
|
36
|
+
};
|
|
49
37
|
|
|
50
|
-
export {
|
|
38
|
+
export { stringifyJSON };
|
|
51
39
|
//# sourceMappingURL=index54.js.map
|
package/dist/index54.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index54.js","sources":["../src/util/
|
|
1
|
+
{"version":3,"file":"index54.js","sources":["../src/util/general.ts"],"sourcesContent":["// Utility function for deep merging two objects.\nexport function deepMerge(target: any, source: any): any {\n for (const key in source) {\n if (Object.prototype.hasOwnProperty.call(source, key)) {\n // Block prototype-polluting keys\n if (key === '__proto__' || key === 'constructor') {\n continue;\n }\n if (source[key] && typeof source[key] === 'object' && !Array.isArray(source[key])) {\n if (!target[key]) {\n target[key] = {};\n }\n deepMerge(target[key], source[key]);\n } else {\n target[key] = source[key];\n }\n }\n }\n return target;\n}\n\n//Recursive implementation of jSON.stringify;\nexport const stringifyJSON = function (obj: any): string {\n\n const arrOfKeyVals: string[] = [];\n const arrVals: string[] = [];\n let objKeys: string[] = [];\n\n /*********CHECK FOR PRIMITIVE TYPES**********/\n if (typeof obj === 'number' || typeof obj === 'boolean' || obj === null)\n return '' + obj;\n else if (typeof obj === 'string')\n return '\"' + obj + '\"';\n\n /*********CHECK FOR ARRAY**********/\n else if (Array.isArray(obj)) {\n //check for empty array\n if (obj[0] === undefined)\n return '[]';\n else {\n obj.forEach(function (el) {\n arrVals.push(stringifyJSON(el));\n });\n return '[' + arrVals + ']';\n }\n }\n /*********CHECK FOR OBJECT**********/\n else if (obj instanceof Object) {\n //get object keys\n objKeys = Object.keys(obj);\n //set key output;\n objKeys.forEach(function (key) {\n const keyOut = '\"' + key + '\":';\n const keyValOut = obj[key];\n //skip functions and undefined properties\n if (keyValOut instanceof Function || keyValOut === undefined)\n arrOfKeyVals.push('');\n else if (typeof keyValOut === 'string')\n arrOfKeyVals.push(keyOut + '\"' + keyValOut + '\"');\n else if (typeof keyValOut === 'boolean' || typeof keyValOut === 'number' || keyValOut === null)\n arrOfKeyVals.push(keyOut + keyValOut);\n //check for nested objects, call recursively until no more objects\n else if (keyValOut instanceof Object) {\n arrOfKeyVals.push(keyOut + stringifyJSON(keyValOut));\n }\n });\n return '{' + arrOfKeyVals + '}';\n }\n return '';\n};"],"names":[],"mappings":"AAsBO,MAAM,aAAA,GAAgB,SAAU,GAAA,EAAkB;AAErD,EAAA,MAAM,eAAyB,EAAC;AAChC,EAAA,MAAM,UAAoB,EAAC;AAC3B,EAAA,IAAI,UAAoB,EAAC;AAGzB,EAAA,IAAI,OAAO,GAAA,KAAQ,QAAA,IAAY,OAAO,GAAA,KAAQ,aAAa,GAAA,KAAQ,IAAA;AAC/D,IAAA,OAAO,EAAA,GAAK,GAAA;AAAA,OAAA,IACP,OAAO,GAAA,KAAQ,QAAA;AACpB,IAAA,OAAO,MAAM,GAAA,GAAM,GAAA;AAAA,OAAA,IAGd,KAAA,CAAM,OAAA,CAAQ,GAAG,CAAA,EAAG;AAEzB,IAAA,IAAI,GAAA,CAAI,CAAC,CAAA,KAAM,MAAA;AACX,MAAA,OAAO,IAAA;AAAA,SACN;AACD,MAAA,GAAA,CAAI,OAAA,CAAQ,SAAU,EAAA,EAAI;AACtB,QAAA,OAAA,CAAQ,IAAA,CAAK,aAAA,CAAc,EAAE,CAAC,CAAA;AAAA,MAClC,CAAC,CAAA;AACD,MAAA,OAAO,MAAM,OAAA,GAAU,GAAA;AAAA,IAC3B;AAAA,EACJ,CAAA,MAAA,IAES,eAAe,MAAA,EAAQ;AAE5B,IAAA,OAAA,GAAU,MAAA,CAAO,KAAK,GAAG,CAAA;AAEzB,IAAA,OAAA,CAAQ,OAAA,CAAQ,SAAU,GAAA,EAAK;AAC3B,MAAA,MAAM,MAAA,GAAS,MAAM,GAAA,GAAM,IAAA;AAC3B,MAAA,MAAM,SAAA,GAAY,IAAI,GAAG,CAAA;AAEzB,MAAA,IAAI,SAAA,YAAqB,YAAY,SAAA,KAAc,MAAA;AAC/C,QAAA,YAAA,CAAa,KAAK,EAAE,CAAA;AAAA,WAAA,IACf,OAAO,SAAA,KAAc,QAAA;AAC1B,QAAA,YAAA,CAAa,IAAA,CAAK,MAAA,GAAS,GAAA,GAAM,SAAA,GAAY,GAAG,CAAA;AAAA,WAAA,IAC3C,OAAO,SAAA,KAAc,SAAA,IAAa,OAAO,SAAA,KAAc,YAAY,SAAA,KAAc,IAAA;AACtF,QAAA,YAAA,CAAa,IAAA,CAAK,SAAS,SAAS,CAAA;AAAA,WAAA,IAE/B,qBAAqB,MAAA,EAAQ;AAClC,QAAA,YAAA,CAAa,IAAA,CAAK,MAAA,GAAS,aAAA,CAAc,SAAS,CAAC,CAAA;AAAA,MACvD;AAAA,IACJ,CAAC,CAAA;AACD,IAAA,OAAO,MAAM,YAAA,GAAe,GAAA;AAAA,EAChC;AACA,EAAA,OAAO,EAAA;AACX;;;;"}
|