@eldrforge/kodrdriv 1.2.0 ā 1.2.2
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/arguments.js +60 -9
- package/dist/arguments.js.map +1 -1
- package/dist/commands/commit.js +16 -6
- package/dist/commands/commit.js.map +1 -1
- package/dist/commands/release.js +9 -3
- package/dist/commands/release.js.map +1 -1
- package/dist/commands/review.js +354 -143
- package/dist/commands/review.js.map +1 -1
- package/dist/constants.js +21 -7
- package/dist/constants.js.map +1 -1
- package/dist/content/issues.js +1 -1
- package/dist/prompt/commit.js +6 -0
- package/dist/prompt/commit.js.map +1 -1
- package/dist/prompt/instructions/review.md +13 -3
- package/dist/types.js +44 -4
- package/dist/types.js.map +1 -1
- package/dist/util/general.js +21 -5
- package/dist/util/general.js.map +1 -1
- package/dist/util/openai.js +70 -9
- package/dist/util/openai.js.map +1 -1
- package/dist/util/validation.js +24 -1
- package/dist/util/validation.js.map +1 -1
- package/package.json +2 -2
- package/test-review-flow.sh +15 -0
- package/test-sort-files/alpha.md +3 -0
- package/test-sort-files/middle.txt +3 -0
- package/test-sort-files/zebra.txt +3 -0
package/dist/util/general.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"general.js","sources":["../../src/util/general.ts"],"sourcesContent":["import path from 'path';\nimport * as Storage from './storage';\nimport { getLogger } from '../logging';\n// eslint-disable-next-line no-restricted-imports\nimport * as fs from 'fs';\n\n// 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 if (key === \"__proto__\" || key === \"constructor\") {\n continue; // Skip prototype-polluting keys\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, options: { depth: number } = { depth: 0 }): string {\n\n if (options.depth > 10) {\n return '{\"error\": \"Maximum depth reached\"}';\n }\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, { depth: options.depth + 1 }));\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, { depth: options.depth + 1 }));\n }\n });\n return '{' + arrOfKeyVals + '}';\n }\n return '';\n};\n\nexport const incrementPatchVersion = (version: string): string => {\n // Remove 'v' prefix if present\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split into major.minor.patch and handle pre-release identifiers\n const parts = cleanVersion.split('.');\n if (parts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n // Handle pre-release versions like \"4.6.24-dev.0\"\n // Split the patch part on '-' to separate patch number from pre-release\n const patchPart = parts[2];\n let patchNumber: number;\n\n if (patchPart.startsWith('-')) {\n // Handle negative patch numbers like \"-1\" or \"-5\"\n patchNumber = parseInt(patchPart, 10);\n } else {\n // Handle normal patch numbers with possible pre-release like \"24-dev.0\"\n const patchComponents = patchPart.split('-');\n patchNumber = parseInt(patchComponents[0], 10);\n }\n\n if (isNaN(patchNumber)) {\n throw new Error(`Invalid patch version: ${patchPart}`);\n }\n\n // Increment the patch number and rebuild the version\n const newPatchNumber = patchNumber + 1;\n\n // For pre-release versions, we'll create a clean release version by dropping the pre-release identifier\n const newVersion = `${parts[0]}.${parts[1]}.${newPatchNumber}`;\n\n return newVersion;\n};\n\nexport const incrementMinorVersion = (version: string): string => {\n // Remove 'v' prefix if present\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split into major.minor.patch and handle pre-release identifiers\n const parts = cleanVersion.split('.');\n if (parts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n const majorNumber = parseInt(parts[0], 10);\n const minorPart = parts[1];\n\n // Handle pre-release versions on minor like \"23-dev.0\"\n const minorComponents = minorPart.split('-');\n const minorNumber = parseInt(minorComponents[0], 10);\n\n if (isNaN(majorNumber) || isNaN(minorNumber)) {\n throw new Error(`Invalid version numbers in: ${version}`);\n }\n\n // Increment the minor number and reset patch to 0\n const newMinorNumber = minorNumber + 1;\n const newVersion = `${majorNumber}.${newMinorNumber}.0`;\n\n return newVersion;\n};\n\nexport const incrementMajorVersion = (version: string): string => {\n // Remove 'v' prefix if present\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split into major.minor.patch and handle pre-release identifiers\n const parts = cleanVersion.split('.');\n if (parts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n const majorPart = parts[0];\n\n // Handle pre-release versions on major like \"4-dev.0\"\n const majorComponents = majorPart.split('-');\n const majorNumber = parseInt(majorComponents[0], 10);\n\n if (isNaN(majorNumber)) {\n throw new Error(`Invalid major version number in: ${version}`);\n }\n\n // Increment the major number and reset minor and patch to 0\n const newMajorNumber = majorNumber + 1;\n const newVersion = `${newMajorNumber}.0.0`;\n\n return newVersion;\n};\n\nexport const validateVersionString = (version: string): boolean => {\n // Remove 'v' prefix if present\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Basic semver regex pattern\n const semverPattern = /^\\d+\\.\\d+\\.\\d+$/;\n return semverPattern.test(cleanVersion);\n};\n\nexport const calculateTargetVersion = (currentVersion: string, targetVersion: string): string => {\n switch (targetVersion.toLowerCase()) {\n case 'patch':\n return incrementPatchVersion(currentVersion);\n case 'minor':\n return incrementMinorVersion(currentVersion);\n case 'major':\n return incrementMajorVersion(currentVersion);\n default:\n // Explicit version provided\n if (!validateVersionString(targetVersion)) {\n throw new Error(`Invalid version format: ${targetVersion}. Expected format: \"x.y.z\" or one of: \"patch\", \"minor\", \"major\"`);\n }\n return targetVersion.startsWith('v') ? targetVersion.slice(1) : targetVersion;\n }\n};\n\nexport const checkIfTagExists = async (tagName: string): Promise<boolean> => {\n const { runSecure, validateGitRef } = await import('./child');\n try {\n // Validate tag name to prevent injection\n if (!validateGitRef(tagName)) {\n throw new Error(`Invalid tag name: ${tagName}`);\n }\n const { stdout } = await runSecure('git', ['tag', '-l', tagName]);\n return stdout.trim() === tagName;\n } catch {\n // If git command fails, assume tag doesn't exist\n return false;\n }\n};\n\nexport const confirmVersionInteractively = async (currentVersion: string, proposedVersion: string, targetVersionInput?: string): Promise<string> => {\n const { getUserChoice, getUserTextInput, requireTTY } = await import('./interactive');\n const { getLogger } = await import('../logging');\n\n requireTTY('Interactive version confirmation requires a terminal.');\n\n const logger = getLogger();\n logger.info(`\\nš¦ Version Confirmation:`);\n logger.info(` Current version: ${currentVersion}`);\n logger.info(` Proposed version: ${proposedVersion}`);\n if (targetVersionInput) {\n logger.info(` Target input: ${targetVersionInput}`);\n }\n\n const choices = [\n { key: 'c', label: `Confirm ${proposedVersion}` },\n { key: 'e', label: 'Enter custom version' },\n { key: 'a', label: 'Abort publish' }\n ];\n\n const choice = await getUserChoice('\\nš¤ Confirm the version for this release:', choices);\n\n switch (choice) {\n case 'c':\n return proposedVersion;\n case 'e': {\n const customVersion = await getUserTextInput('\\nš Enter the version number (e.g., \"4.30.0\"):');\n if (!validateVersionString(customVersion)) {\n throw new Error(`Invalid version format: ${customVersion}. Expected format: \"x.y.z\"`);\n }\n const cleanCustomVersion = customVersion.startsWith('v') ? customVersion.slice(1) : customVersion;\n logger.info(`ā
Using custom version: ${cleanCustomVersion}`);\n return cleanCustomVersion;\n }\n case 'a':\n throw new Error('Release aborted by user');\n default:\n throw new Error(`Unexpected choice: ${choice}`);\n }\n};\n\nexport const getOutputPath = (outputDirectory: string, filename: string): string => {\n return path.join(outputDirectory, filename);\n};\n\nexport const getTimestampedFilename = (baseName: string, extension: string = '.json'): string => {\n const now = new Date();\n\n // Format as YYMMdd-HHmm (e.g., 250701-1030)\n const yy = now.getFullYear().toString().slice(-2);\n const mm = (now.getMonth() + 1).toString().padStart(2, '0');\n const dd = now.getDate().toString().padStart(2, '0');\n const hh = now.getHours().toString().padStart(2, '0');\n const min = now.getMinutes().toString().padStart(2, '0');\n\n const timestamp = `${yy}${mm}${dd}-${hh}${min}`;\n\n return `${timestamp}-${baseName}${extension}`;\n};\n\nexport const getTimestampedRequestFilename = (baseName: string): string => {\n return getTimestampedFilename(baseName, '.request.json');\n};\n\nexport const getTimestampedResponseFilename = (baseName: string): string => {\n return getTimestampedFilename(baseName, '.response.json');\n};\n\nexport const getTimestampedCommitFilename = (): string => {\n return getTimestampedFilename('commit-message', '.md');\n};\n\nexport const getTimestampedReleaseNotesFilename = (): string => {\n return getTimestampedFilename('release-notes', '.md');\n};\n\nexport const getTimestampedAudioFilename = (): string => {\n return getTimestampedFilename('audio-recording', '.wav');\n};\n\nexport const getTimestampedTranscriptFilename = (): string => {\n return getTimestampedFilename('audio-transcript', '.md');\n};\n\nexport const getTimestampedReviewFilename = (): string => {\n return getTimestampedFilename('review-analysis', '.md');\n};\n\nexport const getTimestampedReviewNotesFilename = (): string => {\n return getTimestampedFilename('review-notes', '.md');\n};\n\nexport const getTimestampedArchivedAudioFilename = (originalExtension: string = '.wav'): string => {\n return getTimestampedFilename('review-audio', originalExtension);\n};\n\nexport const getTimestampedArchivedTranscriptFilename = (): string => {\n return getTimestampedFilename('review-transcript', '.md');\n};\n\n/**\n * Archives an audio file and its transcription to the output/kodrdriv directory\n * @param originalAudioPath - Path to the original audio file\n * @param transcriptionText - The raw transcription text\n * @param outputDirectory - Base output directory (default: 'output')\n * @returns Object containing the paths where files were archived\n */\nexport const archiveAudio = async (\n originalAudioPath: string,\n transcriptionText: string,\n outputDirectory: string = 'output'\n): Promise<{ audioPath: string; transcriptPath: string }> => {\n const logger = getLogger();\n const storage = Storage.create({ log: logger.debug });\n\n try {\n // Ensure the output directory exists (should already be output/kodrdriv)\n await storage.ensureDirectory(outputDirectory);\n\n // Get file extension from original audio file\n const originalExtension = path.extname(originalAudioPath);\n\n // Generate timestamped filenames\n const archivedAudioFilename = getTimestampedArchivedAudioFilename(originalExtension);\n const archivedTranscriptFilename = getTimestampedArchivedTranscriptFilename();\n\n // Full paths for archived files - directly in the output directory\n const archivedAudioPath = path.join(outputDirectory, archivedAudioFilename);\n const archivedTranscriptPath = path.join(outputDirectory, archivedTranscriptFilename);\n\n // Copy audio file if it exists\n if (await storage.isFileReadable(originalAudioPath)) {\n // Read original audio file as buffer using fs directly for binary files\n const audioBuffer = await fs.promises.readFile(originalAudioPath);\n await storage.writeFile(archivedAudioPath, audioBuffer, 'binary');\n logger.debug('Archived audio file to: %s', archivedAudioPath);\n } else {\n logger.warn('Original audio file not found or not readable: %s', originalAudioPath);\n }\n\n // Save transcription text\n const transcriptContent = `# Audio Transcription Archive\\n\\n**Original Audio File:** ${originalAudioPath}\\n**Archived:** ${new Date().toISOString()}\\n\\n## Transcription\\n\\n${transcriptionText}`;\n await storage.writeFile(archivedTranscriptPath, transcriptContent, 'utf8');\n logger.debug('Archived transcription to: %s', archivedTranscriptPath);\n\n logger.info('š Audio archived successfully - Audio: %s, Transcript: %s', archivedAudioFilename, archivedTranscriptFilename);\n\n return {\n audioPath: archivedAudioPath,\n transcriptPath: archivedTranscriptPath\n };\n\n } catch (error: any) {\n logger.error('Failed to archive audio: %s', error.message);\n throw new Error(`Audio archiving failed: ${error.message}`);\n }\n};\n"],"names":["stringifyJSON","obj","options","depth","arrOfKeyVals","arrVals","objKeys","Array","isArray","undefined","forEach","el","push","Object","keys","key","keyOut","keyValOut","Function","incrementPatchVersion","version","cleanVersion","startsWith","slice","parts","split","length","Error","patchPart","patchNumber","parseInt","patchComponents","isNaN","newPatchNumber","newVersion","incrementMinorVersion","majorNumber","minorPart","minorComponents","minorNumber","newMinorNumber","incrementMajorVersion","majorPart","majorComponents","newMajorNumber","validateVersionString","semverPattern","test","calculateTargetVersion","currentVersion","targetVersion","toLowerCase","checkIfTagExists","tagName","runSecure","validateGitRef","stdout","trim","confirmVersionInteractively","proposedVersion","targetVersionInput","getUserChoice","getUserTextInput","requireTTY","getLogger","logger","info","choices","label","choice","customVersion","cleanCustomVersion","getOutputPath","outputDirectory","filename","path","join","getTimestampedFilename","baseName","extension","now","Date","yy","getFullYear","toString","mm","getMonth","padStart","dd","getDate","hh","getHours","min","getMinutes","timestamp","getTimestampedRequestFilename","getTimestampedResponseFilename","getTimestampedCommitFilename","getTimestampedReleaseNotesFilename","getTimestampedAudioFilename","getTimestampedReviewFilename","getTimestampedReviewNotesFilename","getTimestampedArchivedAudioFilename","originalExtension","getTimestampedArchivedTranscriptFilename","archiveAudio","originalAudioPath","transcriptionText","storage","Storage","log","debug","ensureDirectory","extname","archivedAudioFilename","archivedTranscriptFilename","archivedAudioPath","archivedTranscriptPath","isFileReadable","audioBuffer","fs","promises","readFile","writeFile","warn","transcriptContent","toISOString","audioPath","transcriptPath","error","message"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"general.js","sources":["../../src/util/general.ts"],"sourcesContent":["import path from 'path';\nimport * as Storage from './storage';\nimport { getLogger } from '../logging';\n// eslint-disable-next-line no-restricted-imports\nimport * as fs from 'fs';\n\n// 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 if (key === \"__proto__\" || key === \"constructor\") {\n continue; // Skip prototype-polluting keys\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, options: { depth: number } = { depth: 0 }): string {\n\n if (options.depth > 10) {\n return '{\"error\": \"Maximum depth reached\"}';\n }\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, { depth: options.depth + 1 }));\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, { depth: options.depth + 1 }));\n }\n });\n return '{' + arrOfKeyVals + '}';\n }\n return '';\n};\n\nexport const incrementPatchVersion = (version: string): string => {\n // Remove 'v' prefix if present\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split into major.minor.patch and handle pre-release identifiers\n const parts = cleanVersion.split('.');\n if (parts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n // Handle pre-release versions like \"4.6.24-dev.0\"\n // Split the patch part on '-' to separate patch number from pre-release\n const patchPart = parts[2];\n let patchNumber: number;\n let originalPatchString: string;\n let hasPrerelease = false;\n\n if (patchPart.startsWith('-')) {\n // Handle negative patch numbers like \"-1\" or \"-5\" or \"-1-dev.0\"\n const negativeComponents = patchPart.split('-');\n // For \"-1-dev.0\", negativeComponents will be ['', '1', 'dev.0']\n if (negativeComponents.length > 2) {\n // This is a negative number with pre-release like \"-1-dev.0\"\n originalPatchString = `-${negativeComponents[1]}`;\n patchNumber = parseInt(`-${negativeComponents[1]}`, 10);\n hasPrerelease = true;\n } else {\n // This is just a negative number like \"-1\"\n patchNumber = parseInt(patchPart, 10);\n originalPatchString = patchPart;\n }\n } else {\n // Handle normal patch numbers with possible pre-release like \"24-dev.0\"\n const patchComponents = patchPart.split('-');\n originalPatchString = patchComponents[0];\n patchNumber = parseInt(patchComponents[0], 10);\n hasPrerelease = patchComponents.length > 1;\n }\n\n if (isNaN(patchNumber)) {\n throw new Error(`Invalid patch version: ${patchPart}`);\n }\n\n // For pre-release versions, graduate to the base version (drop pre-release identifier)\n // For stable versions, increment the patch number\n const newPatchNumber = hasPrerelease ? originalPatchString : (patchNumber + 1).toString();\n\n // Create clean release version\n const newVersion = `${parts[0]}.${parts[1]}.${newPatchNumber}`;\n\n return newVersion;\n};\n\nexport const incrementMinorVersion = (version: string): string => {\n // Remove 'v' prefix if present\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split into major.minor.patch and handle pre-release identifiers\n const parts = cleanVersion.split('.');\n if (parts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n const majorNumber = parseInt(parts[0], 10);\n const minorPart = parts[1];\n\n // Handle pre-release versions on minor like \"23-dev.0\"\n const minorComponents = minorPart.split('-');\n const minorNumber = parseInt(minorComponents[0], 10);\n\n if (isNaN(majorNumber) || isNaN(minorNumber)) {\n throw new Error(`Invalid version numbers in: ${version}`);\n }\n\n // Increment the minor number and reset patch to 0\n const newMinorNumber = minorNumber + 1;\n const newVersion = `${majorNumber}.${newMinorNumber}.0`;\n\n return newVersion;\n};\n\nexport const incrementMajorVersion = (version: string): string => {\n // Remove 'v' prefix if present\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split into major.minor.patch and handle pre-release identifiers\n const parts = cleanVersion.split('.');\n if (parts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n const majorPart = parts[0];\n\n // Handle pre-release versions on major like \"4-dev.0\"\n const majorComponents = majorPart.split('-');\n const majorNumber = parseInt(majorComponents[0], 10);\n\n if (isNaN(majorNumber)) {\n throw new Error(`Invalid major version number in: ${version}`);\n }\n\n // Increment the major number and reset minor and patch to 0\n const newMajorNumber = majorNumber + 1;\n const newVersion = `${newMajorNumber}.0.0`;\n\n return newVersion;\n};\n\nexport const validateVersionString = (version: string): boolean => {\n // Remove 'v' prefix if present\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Basic semver regex pattern\n const semverPattern = /^\\d+\\.\\d+\\.\\d+$/;\n return semverPattern.test(cleanVersion);\n};\n\nexport const calculateTargetVersion = (currentVersion: string, targetVersion: string): string => {\n switch (targetVersion.toLowerCase()) {\n case 'patch':\n return incrementPatchVersion(currentVersion);\n case 'minor':\n return incrementMinorVersion(currentVersion);\n case 'major':\n return incrementMajorVersion(currentVersion);\n default:\n // Explicit version provided\n if (!validateVersionString(targetVersion)) {\n throw new Error(`Invalid version format: ${targetVersion}. Expected format: \"x.y.z\" or one of: \"patch\", \"minor\", \"major\"`);\n }\n return targetVersion.startsWith('v') ? targetVersion.slice(1) : targetVersion;\n }\n};\n\nexport const checkIfTagExists = async (tagName: string): Promise<boolean> => {\n const { runSecure, validateGitRef } = await import('./child');\n try {\n // Validate tag name to prevent injection\n if (!validateGitRef(tagName)) {\n throw new Error(`Invalid tag name: ${tagName}`);\n }\n const { stdout } = await runSecure('git', ['tag', '-l', tagName]);\n return stdout.trim() === tagName;\n } catch {\n // If git command fails, assume tag doesn't exist\n return false;\n }\n};\n\nexport const confirmVersionInteractively = async (currentVersion: string, proposedVersion: string, targetVersionInput?: string): Promise<string> => {\n const { getUserChoice, getUserTextInput, requireTTY } = await import('./interactive');\n const { getLogger } = await import('../logging');\n\n requireTTY('Interactive version confirmation requires a terminal.');\n\n const logger = getLogger();\n logger.info(`\\nš¦ Version Confirmation:`);\n logger.info(` Current version: ${currentVersion}`);\n logger.info(` Proposed version: ${proposedVersion}`);\n if (targetVersionInput) {\n logger.info(` Target input: ${targetVersionInput}`);\n }\n\n const choices = [\n { key: 'c', label: `Confirm ${proposedVersion}` },\n { key: 'e', label: 'Enter custom version' },\n { key: 'a', label: 'Abort publish' }\n ];\n\n const choice = await getUserChoice('\\nš¤ Confirm the version for this release:', choices);\n\n switch (choice) {\n case 'c':\n return proposedVersion;\n case 'e': {\n const customVersion = await getUserTextInput('\\nš Enter the version number (e.g., \"4.30.0\"):');\n if (!validateVersionString(customVersion)) {\n throw new Error(`Invalid version format: ${customVersion}. Expected format: \"x.y.z\"`);\n }\n const cleanCustomVersion = customVersion.startsWith('v') ? customVersion.slice(1) : customVersion;\n logger.info(`ā
Using custom version: ${cleanCustomVersion}`);\n return cleanCustomVersion;\n }\n case 'a':\n throw new Error('Release aborted by user');\n default:\n throw new Error(`Unexpected choice: ${choice}`);\n }\n};\n\nexport const getOutputPath = (outputDirectory: string, filename: string): string => {\n return path.join(outputDirectory, filename);\n};\n\nexport const getTimestampedFilename = (baseName: string, extension: string = '.json'): string => {\n const now = new Date();\n\n // Format as YYMMdd-HHmm (e.g., 250701-1030)\n const yy = now.getFullYear().toString().slice(-2);\n const mm = (now.getMonth() + 1).toString().padStart(2, '0');\n const dd = now.getDate().toString().padStart(2, '0');\n const hh = now.getHours().toString().padStart(2, '0');\n const min = now.getMinutes().toString().padStart(2, '0');\n\n const timestamp = `${yy}${mm}${dd}-${hh}${min}`;\n\n return `${timestamp}-${baseName}${extension}`;\n};\n\nexport const getTimestampedRequestFilename = (baseName: string): string => {\n return getTimestampedFilename(baseName, '.request.json');\n};\n\nexport const getTimestampedResponseFilename = (baseName: string): string => {\n return getTimestampedFilename(baseName, '.response.json');\n};\n\nexport const getTimestampedCommitFilename = (): string => {\n return getTimestampedFilename('commit-message', '.md');\n};\n\nexport const getTimestampedReleaseNotesFilename = (): string => {\n return getTimestampedFilename('release-notes', '.md');\n};\n\nexport const getTimestampedAudioFilename = (): string => {\n return getTimestampedFilename('audio-recording', '.wav');\n};\n\nexport const getTimestampedTranscriptFilename = (): string => {\n return getTimestampedFilename('audio-transcript', '.md');\n};\n\nexport const getTimestampedReviewFilename = (): string => {\n return getTimestampedFilename('review-analysis', '.md');\n};\n\nexport const getTimestampedReviewNotesFilename = (): string => {\n return getTimestampedFilename('review-notes', '.md');\n};\n\nexport const getTimestampedArchivedAudioFilename = (originalExtension: string = '.wav'): string => {\n return getTimestampedFilename('review-audio', originalExtension);\n};\n\nexport const getTimestampedArchivedTranscriptFilename = (): string => {\n return getTimestampedFilename('review-transcript', '.md');\n};\n\n/**\n * Archives an audio file and its transcription to the output/kodrdriv directory\n * @param originalAudioPath - Path to the original audio file\n * @param transcriptionText - The raw transcription text\n * @param outputDirectory - Base output directory (default: 'output')\n * @returns Object containing the paths where files were archived\n */\nexport const archiveAudio = async (\n originalAudioPath: string,\n transcriptionText: string,\n outputDirectory: string = 'output'\n): Promise<{ audioPath: string; transcriptPath: string }> => {\n const logger = getLogger();\n const storage = Storage.create({ log: logger.debug });\n\n try {\n // Ensure the output directory exists (should already be output/kodrdriv)\n await storage.ensureDirectory(outputDirectory);\n\n // Get file extension from original audio file\n const originalExtension = path.extname(originalAudioPath);\n\n // Generate timestamped filenames\n const archivedAudioFilename = getTimestampedArchivedAudioFilename(originalExtension);\n const archivedTranscriptFilename = getTimestampedArchivedTranscriptFilename();\n\n // Full paths for archived files - directly in the output directory\n const archivedAudioPath = path.join(outputDirectory, archivedAudioFilename);\n const archivedTranscriptPath = path.join(outputDirectory, archivedTranscriptFilename);\n\n // Copy audio file if it exists\n if (await storage.isFileReadable(originalAudioPath)) {\n // Read original audio file as buffer using fs directly for binary files\n const audioBuffer = await fs.promises.readFile(originalAudioPath);\n await storage.writeFile(archivedAudioPath, audioBuffer, 'binary');\n logger.debug('Archived audio file to: %s', archivedAudioPath);\n } else {\n logger.warn('Original audio file not found or not readable: %s', originalAudioPath);\n }\n\n // Save transcription text\n const transcriptContent = `# Audio Transcription Archive\\n\\n**Original Audio File:** ${originalAudioPath}\\n**Archived:** ${new Date().toISOString()}\\n\\n## Transcription\\n\\n${transcriptionText}`;\n await storage.writeFile(archivedTranscriptPath, transcriptContent, 'utf8');\n logger.debug('Archived transcription to: %s', archivedTranscriptPath);\n\n logger.info('š Audio archived successfully - Audio: %s, Transcript: %s', archivedAudioFilename, archivedTranscriptFilename);\n\n return {\n audioPath: archivedAudioPath,\n transcriptPath: archivedTranscriptPath\n };\n\n } catch (error: any) {\n logger.error('Failed to archive audio: %s', error.message);\n throw new Error(`Audio archiving failed: ${error.message}`);\n }\n};\n"],"names":["stringifyJSON","obj","options","depth","arrOfKeyVals","arrVals","objKeys","Array","isArray","undefined","forEach","el","push","Object","keys","key","keyOut","keyValOut","Function","incrementPatchVersion","version","cleanVersion","startsWith","slice","parts","split","length","Error","patchPart","patchNumber","originalPatchString","hasPrerelease","negativeComponents","parseInt","patchComponents","isNaN","newPatchNumber","toString","newVersion","incrementMinorVersion","majorNumber","minorPart","minorComponents","minorNumber","newMinorNumber","incrementMajorVersion","majorPart","majorComponents","newMajorNumber","validateVersionString","semverPattern","test","calculateTargetVersion","currentVersion","targetVersion","toLowerCase","checkIfTagExists","tagName","runSecure","validateGitRef","stdout","trim","confirmVersionInteractively","proposedVersion","targetVersionInput","getUserChoice","getUserTextInput","requireTTY","getLogger","logger","info","choices","label","choice","customVersion","cleanCustomVersion","getOutputPath","outputDirectory","filename","path","join","getTimestampedFilename","baseName","extension","now","Date","yy","getFullYear","mm","getMonth","padStart","dd","getDate","hh","getHours","min","getMinutes","timestamp","getTimestampedRequestFilename","getTimestampedResponseFilename","getTimestampedCommitFilename","getTimestampedReleaseNotesFilename","getTimestampedAudioFilename","getTimestampedReviewFilename","getTimestampedReviewNotesFilename","getTimestampedArchivedAudioFilename","originalExtension","getTimestampedArchivedTranscriptFilename","archiveAudio","originalAudioPath","transcriptionText","storage","Storage","log","debug","ensureDirectory","extname","archivedAudioFilename","archivedTranscriptFilename","archivedAudioPath","archivedTranscriptPath","isFileReadable","audioBuffer","fs","promises","readFile","writeFile","warn","transcriptContent","toISOString","audioPath","transcriptPath","error","message"],"mappings":"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"}
|
package/dist/util/openai.js
CHANGED
|
@@ -41,6 +41,54 @@ function _define_property(obj, key, value) {
|
|
|
41
41
|
// Return command-specific model if available, otherwise global model
|
|
42
42
|
return commandModel || config.model || 'gpt-4o-mini';
|
|
43
43
|
}
|
|
44
|
+
/**
|
|
45
|
+
* Get the appropriate OpenAI reasoning level based on command-specific configuration
|
|
46
|
+
* Command-specific reasoning overrides the global reasoning setting
|
|
47
|
+
*/ function getOpenAIReasoningForCommand(config, commandName) {
|
|
48
|
+
let commandReasoning;
|
|
49
|
+
switch(commandName){
|
|
50
|
+
case 'commit':
|
|
51
|
+
case 'audio-commit':
|
|
52
|
+
var _config_commit;
|
|
53
|
+
commandReasoning = (_config_commit = config.commit) === null || _config_commit === void 0 ? void 0 : _config_commit.openaiReasoning;
|
|
54
|
+
break;
|
|
55
|
+
case 'release':
|
|
56
|
+
var _config_release;
|
|
57
|
+
commandReasoning = (_config_release = config.release) === null || _config_release === void 0 ? void 0 : _config_release.openaiReasoning;
|
|
58
|
+
break;
|
|
59
|
+
case 'review':
|
|
60
|
+
case 'audio-review':
|
|
61
|
+
var _config_review;
|
|
62
|
+
commandReasoning = (_config_review = config.review) === null || _config_review === void 0 ? void 0 : _config_review.openaiReasoning;
|
|
63
|
+
break;
|
|
64
|
+
}
|
|
65
|
+
// Return command-specific reasoning if available, otherwise global reasoning
|
|
66
|
+
return commandReasoning || config.openaiReasoning || 'low';
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Get the appropriate OpenAI max output tokens based on command-specific configuration
|
|
70
|
+
* Command-specific max output tokens overrides the global setting
|
|
71
|
+
*/ function getOpenAIMaxOutputTokensForCommand(config, commandName) {
|
|
72
|
+
let commandMaxOutputTokens;
|
|
73
|
+
switch(commandName){
|
|
74
|
+
case 'commit':
|
|
75
|
+
case 'audio-commit':
|
|
76
|
+
var _config_commit;
|
|
77
|
+
commandMaxOutputTokens = (_config_commit = config.commit) === null || _config_commit === void 0 ? void 0 : _config_commit.openaiMaxOutputTokens;
|
|
78
|
+
break;
|
|
79
|
+
case 'release':
|
|
80
|
+
var _config_release;
|
|
81
|
+
commandMaxOutputTokens = (_config_release = config.release) === null || _config_release === void 0 ? void 0 : _config_release.openaiMaxOutputTokens;
|
|
82
|
+
break;
|
|
83
|
+
case 'review':
|
|
84
|
+
case 'audio-review':
|
|
85
|
+
var _config_review;
|
|
86
|
+
commandMaxOutputTokens = (_config_review = config.review) === null || _config_review === void 0 ? void 0 : _config_review.openaiMaxOutputTokens;
|
|
87
|
+
break;
|
|
88
|
+
}
|
|
89
|
+
// Return command-specific max output tokens if available, otherwise global setting
|
|
90
|
+
return commandMaxOutputTokens || config.openaiMaxOutputTokens || 10000;
|
|
91
|
+
}
|
|
44
92
|
class OpenAIError extends Error {
|
|
45
93
|
constructor(message, isTokenLimitError = false){
|
|
46
94
|
super(message), _define_property(this, "isTokenLimitError", void 0), this.isTokenLimitError = isTokenLimitError;
|
|
@@ -60,8 +108,12 @@ function isRateLimitError(error) {
|
|
|
60
108
|
if (error.status === 429 || error.code === 'rate_limit_exceeded') {
|
|
61
109
|
return true;
|
|
62
110
|
}
|
|
63
|
-
|
|
64
|
-
|
|
111
|
+
// Only check message if it exists
|
|
112
|
+
if (error.message) {
|
|
113
|
+
const message = error.message.toLowerCase();
|
|
114
|
+
return message.includes('rate limit exceeded') || message.includes('too many requests') || message.includes('quota exceeded') || message.includes('rate') && message.includes('limit');
|
|
115
|
+
}
|
|
116
|
+
return false;
|
|
65
117
|
}
|
|
66
118
|
async function createCompletion(messages, options = {
|
|
67
119
|
model: "gpt-4o-mini"
|
|
@@ -86,27 +138,36 @@ async function createCompletion(messages, options = {
|
|
|
86
138
|
const modelToUse = options.model || "gpt-4o-mini";
|
|
87
139
|
logger.info('š¤ Making request to OpenAI using model: %s', modelToUse);
|
|
88
140
|
logger.debug('Sending prompt to OpenAI: %j', messages);
|
|
89
|
-
|
|
90
|
-
|
|
141
|
+
var _options_openaiMaxOutputTokens, _ref;
|
|
142
|
+
// Use openaiMaxOutputTokens if specified (highest priority), otherwise fall back to maxTokens, or default to 10000
|
|
143
|
+
const maxCompletionTokens = (_ref = (_options_openaiMaxOutputTokens = options.openaiMaxOutputTokens) !== null && _options_openaiMaxOutputTokens !== void 0 ? _options_openaiMaxOutputTokens : options.maxTokens) !== null && _ref !== void 0 ? _ref : 10000;
|
|
91
144
|
// Save request debug file if enabled
|
|
92
145
|
if (options.debug && (options.debugRequestFile || options.debugFile)) {
|
|
93
146
|
const requestData = {
|
|
94
147
|
model: modelToUse,
|
|
95
148
|
messages,
|
|
96
149
|
max_completion_tokens: maxCompletionTokens,
|
|
97
|
-
response_format: options.responseFormat
|
|
150
|
+
response_format: options.responseFormat,
|
|
151
|
+
reasoning_effort: options.openaiReasoning
|
|
98
152
|
};
|
|
99
153
|
const debugFile = options.debugRequestFile || options.debugFile;
|
|
100
154
|
await storage.writeFile(debugFile, JSON.stringify(requestData, null, 2), 'utf8');
|
|
101
155
|
logger.debug('Wrote request debug file to %s', debugFile);
|
|
102
156
|
}
|
|
103
|
-
//
|
|
104
|
-
const
|
|
157
|
+
// Prepare the API call options
|
|
158
|
+
const apiOptions = {
|
|
105
159
|
model: modelToUse,
|
|
106
160
|
messages,
|
|
107
161
|
max_completion_tokens: maxCompletionTokens,
|
|
108
162
|
response_format: options.responseFormat
|
|
109
|
-
}
|
|
163
|
+
};
|
|
164
|
+
// Add reasoning parameter if specified and model supports it
|
|
165
|
+
if (options.openaiReasoning && (modelToUse.includes('gpt-5') || modelToUse.includes('o3'))) {
|
|
166
|
+
apiOptions.reasoning_effort = options.openaiReasoning;
|
|
167
|
+
logger.debug('Using OpenAI reasoning level: %s', options.openaiReasoning);
|
|
168
|
+
}
|
|
169
|
+
// Add timeout wrapper to the OpenAI API call
|
|
170
|
+
const completionPromise = openai.chat.completions.create(apiOptions);
|
|
110
171
|
// Create timeout promise with proper cleanup to prevent memory leaks
|
|
111
172
|
let timeoutId = null;
|
|
112
173
|
const timeoutPromise = new Promise((_, reject)=>{
|
|
@@ -280,5 +341,5 @@ async function transcribeAudio(filePath, options = {
|
|
|
280
341
|
}
|
|
281
342
|
}
|
|
282
343
|
|
|
283
|
-
export { OpenAIError, createCompletion, createCompletionWithRetry, getModelForCommand, isRateLimitError, isTokenLimitError, transcribeAudio };
|
|
344
|
+
export { OpenAIError, createCompletion, createCompletionWithRetry, getModelForCommand, getOpenAIMaxOutputTokensForCommand, getOpenAIReasoningForCommand, isRateLimitError, isTokenLimitError, transcribeAudio };
|
|
284
345
|
//# sourceMappingURL=openai.js.map
|
package/dist/util/openai.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"openai.js","sources":["../../src/util/openai.ts"],"sourcesContent":["import { OpenAI } from 'openai';\nimport { ChatCompletionMessageParam } from 'openai/resources';\nimport * as Storage from './storage';\nimport { getLogger } from '../logging';\nimport { archiveAudio } from './general';\nimport { Config } from '../types';\nimport { safeJsonParse } from './validation';\n// eslint-disable-next-line no-restricted-imports\nimport fs from 'fs';\n\nexport interface Transcription {\n text: string;\n}\n\n/**\n * Get the appropriate model to use based on command-specific configuration\n * Command-specific model overrides the global model setting\n */\nexport function getModelForCommand(config: Config, commandName: string): string {\n let commandModel: string | undefined;\n\n switch (commandName) {\n case 'commit':\n case 'audio-commit':\n commandModel = config.commit?.model;\n break;\n case 'release':\n commandModel = config.release?.model;\n break;\n case 'review':\n case 'audio-review':\n commandModel = config.review?.model;\n break;\n default:\n // For other commands, just use global model\n break;\n }\n\n // Return command-specific model if available, otherwise global model\n return commandModel || config.model || 'gpt-4o-mini';\n}\n\nexport class OpenAIError extends Error {\n constructor(message: string, public readonly isTokenLimitError: boolean = false) {\n super(message);\n this.name = 'OpenAIError';\n }\n}\n\n// Check if an error is a token limit exceeded error\nexport function isTokenLimitError(error: any): boolean {\n if (!error?.message) return false;\n\n const message = error.message.toLowerCase();\n return message.includes('maximum context length') ||\n message.includes('context_length_exceeded') ||\n message.includes('token limit') ||\n message.includes('too many tokens') ||\n message.includes('reduce the length');\n}\n\n// Check if an error is a rate limit error\nexport function isRateLimitError(error: any): boolean {\n if (!error?.message && !error?.code && !error?.status) return false;\n\n // Check for OpenAI specific rate limit indicators\n if (error.status === 429 || error.code === 'rate_limit_exceeded') {\n return true;\n }\n\n const message = error.message.toLowerCase();\n return message.includes('rate limit exceeded') ||\n message.includes('too many requests') ||\n message.includes('quota exceeded') ||\n (message.includes('rate') && message.includes('limit'));\n}\n\nexport async function createCompletion(messages: ChatCompletionMessageParam[], options: { responseFormat?: any, model?: string, debug?: boolean, debugFile?: string, debugRequestFile?: string, debugResponseFile?: string, maxTokens?: number } = { model: \"gpt-4o-mini\" }): Promise<string | any> {\n const logger = getLogger();\n const storage = Storage.create({ log: logger.debug });\n let openai: OpenAI | null = null;\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 // Create the client which we'll close in the finally block.\n const timeoutMs = parseInt(process.env.OPENAI_TIMEOUT_MS || '300000'); // Default to 5 minutes\n openai = new OpenAI({\n apiKey: apiKey,\n timeout: timeoutMs,\n });\n\n const modelToUse = options.model || \"gpt-4o-mini\";\n logger.info('š¤ Making request to OpenAI using model: %s', modelToUse);\n logger.debug('Sending prompt to OpenAI: %j', messages);\n\n // Use provided maxTokens or default to 10000\n const maxCompletionTokens = options.maxTokens || 10000;\n\n // Save request debug file if enabled\n if (options.debug && (options.debugRequestFile || options.debugFile)) {\n const requestData = {\n model: modelToUse,\n messages,\n max_completion_tokens: maxCompletionTokens,\n response_format: options.responseFormat,\n };\n const debugFile = options.debugRequestFile || options.debugFile;\n await storage.writeFile(debugFile!, JSON.stringify(requestData, null, 2), 'utf8');\n logger.debug('Wrote request debug file to %s', debugFile);\n }\n\n // Add timeout wrapper to the OpenAI API call\n const completionPromise = openai.chat.completions.create({\n model: modelToUse,\n messages,\n max_completion_tokens: maxCompletionTokens,\n response_format: options.responseFormat,\n });\n\n // Create timeout promise with proper cleanup to prevent memory leaks\n let timeoutId: NodeJS.Timeout | null = null;\n const timeoutPromise = new Promise<never>((_, reject) => {\n const timeoutMs = parseInt(process.env.OPENAI_TIMEOUT_MS || '300000'); // Default to 5 minutes\n timeoutId = setTimeout(() => reject(new OpenAIError(`OpenAI API call timed out after ${timeoutMs/1000} seconds`)), timeoutMs);\n });\n\n let completion;\n try {\n completion = await Promise.race([completionPromise, timeoutPromise]);\n } finally {\n // Clear the timeout to prevent memory leaks\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n }\n\n // Save response debug file if enabled\n if (options.debug && (options.debugResponseFile || options.debugFile)) {\n const debugFile = options.debugResponseFile || options.debugFile;\n await storage.writeFile(debugFile!, JSON.stringify(completion, null, 2), 'utf8');\n logger.debug('Wrote response debug file to %s', debugFile);\n }\n\n const response = completion.choices[0]?.message?.content?.trim();\n if (!response) {\n throw new OpenAIError('No response received from OpenAI');\n }\n\n logger.debug('Received response from OpenAI: %s...', response.substring(0, 30));\n if (options.responseFormat) {\n return safeJsonParse(response, 'OpenAI API 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 const isTokenError = isTokenLimitError(error);\n throw new OpenAIError(`Failed to create completion: ${error.message}`, isTokenError);\n } finally {\n // OpenAI client cleanup is handled automatically by the library\n // No manual cleanup needed for newer versions\n }\n}\n\n// Create completion with automatic retry on token limit errors\nexport async function createCompletionWithRetry(\n messages: ChatCompletionMessageParam[],\n options: { responseFormat?: any, model?: string, debug?: boolean, debugFile?: string, debugRequestFile?: string, debugResponseFile?: string, maxTokens?: number } = { model: \"gpt-4o-mini\" },\n retryCallback?: (attempt: number) => Promise<ChatCompletionMessageParam[]>\n): Promise<string | any> {\n const logger = getLogger();\n const maxRetries = 3;\n\n for (let attempt = 1; attempt <= maxRetries; attempt++) {\n try {\n const messagesToSend = attempt === 1 ? messages : (retryCallback ? await retryCallback(attempt) : messages);\n return await createCompletion(messagesToSend, options);\n } catch (error: any) {\n if (error instanceof OpenAIError && error.isTokenLimitError && attempt < maxRetries && retryCallback) {\n logger.warn('Token limit exceeded on attempt %d/%d, retrying with reduced content...', attempt, maxRetries);\n // Add exponential backoff for token limit errors\n const backoffMs = Math.min(1000 * Math.pow(2, attempt - 1), 10000);\n await new Promise(resolve => setTimeout(resolve, backoffMs));\n continue;\n } else if (isRateLimitError(error) && attempt < maxRetries) {\n // Handle rate limiting with exponential backoff\n const backoffMs = Math.min(2000 * Math.pow(2, attempt - 1), 15000); // More reasonable backoff: 2s, 4s, 8s, max 15s\n logger.warn(`Rate limit hit on attempt ${attempt}/${maxRetries}, waiting ${backoffMs}ms before retry...`);\n await new Promise(resolve => setTimeout(resolve, backoffMs));\n continue;\n }\n throw error;\n }\n }\n\n // This should never be reached, but TypeScript requires it\n throw new OpenAIError('Max retries exceeded');\n}\n\nexport async function transcribeAudio(filePath: string, options: { model?: string, debug?: boolean, debugFile?: string, debugRequestFile?: string, debugResponseFile?: string, outputDirectory?: string } = { model: \"whisper-1\" }): Promise<Transcription> {\n const logger = getLogger();\n const storage = Storage.create({ log: logger.debug });\n let openai: OpenAI | null = null;\n let audioStream: fs.ReadStream | null = null;\n let streamClosed = false;\n\n // Helper function to safely close the stream\n const closeAudioStream = () => {\n if (audioStream && !streamClosed) {\n try {\n // Only call destroy if it exists and the stream isn't already destroyed\n if (typeof audioStream.destroy === 'function' && !audioStream.destroyed) {\n audioStream.destroy();\n }\n streamClosed = true;\n logger.debug('Audio stream closed successfully');\n } catch (streamErr) {\n logger.debug('Failed to destroy audio read stream: %s', (streamErr as Error).message);\n streamClosed = true; // Mark as closed even if destroy failed\n }\n }\n };\n\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 openai = new OpenAI({\n apiKey: apiKey,\n });\n\n logger.debug('Transcribing audio file: %s', filePath);\n\n // Save request debug file if enabled\n if (options.debug && (options.debugRequestFile || options.debugFile)) {\n const requestData = {\n model: options.model || \"whisper-1\",\n file: filePath, // Can't serialize the stream, so just save the file path\n response_format: \"json\",\n };\n const debugFile = options.debugRequestFile || options.debugFile;\n await storage.writeFile(debugFile!, JSON.stringify(requestData, null, 2), 'utf8');\n logger.debug('Wrote request debug file to %s', debugFile);\n }\n\n audioStream = await storage.readStream(filePath);\n\n // Set up error handler for the stream to ensure cleanup on stream errors\n // Only add handler if the stream has the 'on' method (real streams)\n if (audioStream && typeof audioStream.on === 'function') {\n audioStream.on('error', (streamError) => {\n logger.error('Audio stream error: %s', streamError.message);\n closeAudioStream();\n });\n }\n\n let transcription;\n try {\n transcription = await openai.audio.transcriptions.create({\n model: options.model || \"whisper-1\",\n file: audioStream,\n response_format: \"json\",\n });\n // Close the stream immediately after successful API call to prevent race conditions\n closeAudioStream();\n } catch (apiError) {\n // Close the stream immediately if the API call fails\n closeAudioStream();\n throw apiError;\n }\n\n // Save response debug file if enabled\n if (options.debug && (options.debugResponseFile || options.debugFile)) {\n const debugFile = options.debugResponseFile || options.debugFile;\n await storage.writeFile(debugFile!, JSON.stringify(transcription, null, 2), 'utf8');\n logger.debug('Wrote response debug file to %s', debugFile);\n }\n\n const response = transcription;\n if (!response) {\n throw new OpenAIError('No transcription received from OpenAI');\n }\n\n logger.debug('Received transcription from OpenAI: %s', response);\n\n // Archive the audio file and transcription\n try {\n const outputDir = options.outputDirectory || 'output';\n await archiveAudio(filePath, response.text, outputDir);\n } catch (archiveError: any) {\n // Don't fail the transcription if archiving fails, just log the error\n logger.warn('Failed to archive audio file: %s', archiveError.message);\n }\n\n return response;\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 } finally {\n // Ensure the audio stream is properly closed to release file handles\n closeAudioStream();\n // OpenAI client cleanup is handled automatically by the library\n // No manual cleanup needed for newer versions\n }\n}\n"],"names":["getModelForCommand","config","commandName","commandModel","commit","model","release","review","OpenAIError","Error","message","isTokenLimitError","name","error","toLowerCase","includes","isRateLimitError","code","status","createCompletion","messages","options","logger","getLogger","storage","Storage","log","debug","openai","completion","apiKey","process","env","OPENAI_API_KEY","timeoutMs","parseInt","OPENAI_TIMEOUT_MS","OpenAI","timeout","modelToUse","info","maxCompletionTokens","maxTokens","debugRequestFile","debugFile","requestData","max_completion_tokens","response_format","responseFormat","writeFile","JSON","stringify","completionPromise","chat","completions","create","timeoutId","timeoutPromise","Promise","_","reject","setTimeout","race","clearTimeout","debugResponseFile","response","choices","content","trim","substring","safeJsonParse","stack","isTokenError","createCompletionWithRetry","retryCallback","maxRetries","attempt","messagesToSend","warn","backoffMs","Math","min","pow","resolve","transcribeAudio","filePath","audioStream","streamClosed","closeAudioStream","destroy","destroyed","streamErr","file","readStream","on","streamError","transcription","audio","transcriptions","apiError","outputDir","outputDirectory","archiveAudio","text","archiveError"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"openai.js","sources":["../../src/util/openai.ts"],"sourcesContent":["import { OpenAI } from 'openai';\nimport { ChatCompletionMessageParam } from 'openai/resources';\nimport * as Storage from './storage';\nimport { getLogger } from '../logging';\nimport { archiveAudio } from './general';\nimport { Config } from '../types';\nimport { safeJsonParse } from './validation';\n// eslint-disable-next-line no-restricted-imports\nimport fs from 'fs';\n\nexport interface Transcription {\n text: string;\n}\n\n/**\n * Get the appropriate model to use based on command-specific configuration\n * Command-specific model overrides the global model setting\n */\nexport function getModelForCommand(config: Config, commandName: string): string {\n let commandModel: string | undefined;\n\n switch (commandName) {\n case 'commit':\n case 'audio-commit':\n commandModel = config.commit?.model;\n break;\n case 'release':\n commandModel = config.release?.model;\n break;\n case 'review':\n case 'audio-review':\n commandModel = config.review?.model;\n break;\n default:\n // For other commands, just use global model\n break;\n }\n\n // Return command-specific model if available, otherwise global model\n return commandModel || config.model || 'gpt-4o-mini';\n}\n\n/**\n * Get the appropriate OpenAI reasoning level based on command-specific configuration\n * Command-specific reasoning overrides the global reasoning setting\n */\nexport function getOpenAIReasoningForCommand(config: Config, commandName: string): 'low' | 'medium' | 'high' {\n let commandReasoning: 'low' | 'medium' | 'high' | undefined;\n\n switch (commandName) {\n case 'commit':\n case 'audio-commit':\n commandReasoning = config.commit?.openaiReasoning;\n break;\n case 'release':\n commandReasoning = config.release?.openaiReasoning;\n break;\n case 'review':\n case 'audio-review':\n commandReasoning = config.review?.openaiReasoning;\n break;\n default:\n // For other commands, just use global reasoning\n break;\n }\n\n // Return command-specific reasoning if available, otherwise global reasoning\n return commandReasoning || config.openaiReasoning || 'low';\n}\n\n/**\n * Get the appropriate OpenAI max output tokens based on command-specific configuration\n * Command-specific max output tokens overrides the global setting\n */\nexport function getOpenAIMaxOutputTokensForCommand(config: Config, commandName: string): number {\n let commandMaxOutputTokens: number | undefined;\n\n switch (commandName) {\n case 'commit':\n case 'audio-commit':\n commandMaxOutputTokens = config.commit?.openaiMaxOutputTokens;\n break;\n case 'release':\n commandMaxOutputTokens = config.release?.openaiMaxOutputTokens;\n break;\n case 'review':\n case 'audio-review':\n commandMaxOutputTokens = config.review?.openaiMaxOutputTokens;\n break;\n default:\n // For other commands, just use global max output tokens\n break;\n }\n\n // Return command-specific max output tokens if available, otherwise global setting\n return commandMaxOutputTokens || config.openaiMaxOutputTokens || 10000;\n}\n\nexport class OpenAIError extends Error {\n constructor(message: string, public readonly isTokenLimitError: boolean = false) {\n super(message);\n this.name = 'OpenAIError';\n }\n}\n\n// Check if an error is a token limit exceeded error\nexport function isTokenLimitError(error: any): boolean {\n if (!error?.message) return false;\n\n const message = error.message.toLowerCase();\n return message.includes('maximum context length') ||\n message.includes('context_length_exceeded') ||\n message.includes('token limit') ||\n message.includes('too many tokens') ||\n message.includes('reduce the length');\n}\n\n// Check if an error is a rate limit error\nexport function isRateLimitError(error: any): boolean {\n if (!error?.message && !error?.code && !error?.status) return false;\n\n // Check for OpenAI specific rate limit indicators\n if (error.status === 429 || error.code === 'rate_limit_exceeded') {\n return true;\n }\n\n // Only check message if it exists\n if (error.message) {\n const message = error.message.toLowerCase();\n return message.includes('rate limit exceeded') ||\n message.includes('too many requests') ||\n message.includes('quota exceeded') ||\n (message.includes('rate') && message.includes('limit'));\n }\n\n return false;\n}\n\nexport async function createCompletion(messages: ChatCompletionMessageParam[], options: { responseFormat?: any, model?: string, debug?: boolean, debugFile?: string, debugRequestFile?: string, debugResponseFile?: string, maxTokens?: number, openaiReasoning?: 'low' | 'medium' | 'high', openaiMaxOutputTokens?: number } = { model: \"gpt-4o-mini\" }): Promise<string | any> {\n const logger = getLogger();\n const storage = Storage.create({ log: logger.debug });\n let openai: OpenAI | null = null;\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 // Create the client which we'll close in the finally block.\n const timeoutMs = parseInt(process.env.OPENAI_TIMEOUT_MS || '300000'); // Default to 5 minutes\n openai = new OpenAI({\n apiKey: apiKey,\n timeout: timeoutMs,\n });\n\n const modelToUse = options.model || \"gpt-4o-mini\";\n logger.info('š¤ Making request to OpenAI using model: %s', modelToUse);\n logger.debug('Sending prompt to OpenAI: %j', messages);\n\n // Use openaiMaxOutputTokens if specified (highest priority), otherwise fall back to maxTokens, or default to 10000\n const maxCompletionTokens = options.openaiMaxOutputTokens ?? options.maxTokens ?? 10000;\n\n // Save request debug file if enabled\n if (options.debug && (options.debugRequestFile || options.debugFile)) {\n const requestData = {\n model: modelToUse,\n messages,\n max_completion_tokens: maxCompletionTokens,\n response_format: options.responseFormat,\n reasoning_effort: options.openaiReasoning,\n };\n const debugFile = options.debugRequestFile || options.debugFile;\n await storage.writeFile(debugFile!, JSON.stringify(requestData, null, 2), 'utf8');\n logger.debug('Wrote request debug file to %s', debugFile);\n }\n\n // Prepare the API call options\n const apiOptions: any = {\n model: modelToUse,\n messages,\n max_completion_tokens: maxCompletionTokens,\n response_format: options.responseFormat,\n };\n\n // Add reasoning parameter if specified and model supports it\n if (options.openaiReasoning && (modelToUse.includes('gpt-5') || modelToUse.includes('o3'))) {\n apiOptions.reasoning_effort = options.openaiReasoning;\n logger.debug('Using OpenAI reasoning level: %s', options.openaiReasoning);\n }\n\n // Add timeout wrapper to the OpenAI API call\n const completionPromise = openai.chat.completions.create(apiOptions);\n\n // Create timeout promise with proper cleanup to prevent memory leaks\n let timeoutId: NodeJS.Timeout | null = null;\n const timeoutPromise = new Promise<never>((_, reject) => {\n const timeoutMs = parseInt(process.env.OPENAI_TIMEOUT_MS || '300000'); // Default to 5 minutes\n timeoutId = setTimeout(() => reject(new OpenAIError(`OpenAI API call timed out after ${timeoutMs/1000} seconds`)), timeoutMs);\n });\n\n let completion;\n try {\n completion = await Promise.race([completionPromise, timeoutPromise]);\n } finally {\n // Clear the timeout to prevent memory leaks\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n }\n\n // Save response debug file if enabled\n if (options.debug && (options.debugResponseFile || options.debugFile)) {\n const debugFile = options.debugResponseFile || options.debugFile;\n await storage.writeFile(debugFile!, JSON.stringify(completion, null, 2), 'utf8');\n logger.debug('Wrote response debug file to %s', debugFile);\n }\n\n const response = completion.choices[0]?.message?.content?.trim();\n if (!response) {\n throw new OpenAIError('No response received from OpenAI');\n }\n\n logger.debug('Received response from OpenAI: %s...', response.substring(0, 30));\n if (options.responseFormat) {\n return safeJsonParse(response, 'OpenAI API 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 const isTokenError = isTokenLimitError(error);\n throw new OpenAIError(`Failed to create completion: ${error.message}`, isTokenError);\n } finally {\n // OpenAI client cleanup is handled automatically by the library\n // No manual cleanup needed for newer versions\n }\n}\n\n// Create completion with automatic retry on token limit errors\nexport async function createCompletionWithRetry(\n messages: ChatCompletionMessageParam[],\n options: { responseFormat?: any, model?: string, debug?: boolean, debugFile?: string, debugRequestFile?: string, debugResponseFile?: string, maxTokens?: number, openaiReasoning?: 'low' | 'medium' | 'high', openaiMaxOutputTokens?: number } = { model: \"gpt-4o-mini\" },\n retryCallback?: (attempt: number) => Promise<ChatCompletionMessageParam[]>\n): Promise<string | any> {\n const logger = getLogger();\n const maxRetries = 3;\n\n for (let attempt = 1; attempt <= maxRetries; attempt++) {\n try {\n const messagesToSend = attempt === 1 ? messages : (retryCallback ? await retryCallback(attempt) : messages);\n return await createCompletion(messagesToSend, options);\n } catch (error: any) {\n if (error instanceof OpenAIError && error.isTokenLimitError && attempt < maxRetries && retryCallback) {\n logger.warn('Token limit exceeded on attempt %d/%d, retrying with reduced content...', attempt, maxRetries);\n // Add exponential backoff for token limit errors\n const backoffMs = Math.min(1000 * Math.pow(2, attempt - 1), 10000);\n await new Promise(resolve => setTimeout(resolve, backoffMs));\n continue;\n } else if (isRateLimitError(error) && attempt < maxRetries) {\n // Handle rate limiting with exponential backoff\n const backoffMs = Math.min(2000 * Math.pow(2, attempt - 1), 15000); // More reasonable backoff: 2s, 4s, 8s, max 15s\n logger.warn(`Rate limit hit on attempt ${attempt}/${maxRetries}, waiting ${backoffMs}ms before retry...`);\n await new Promise(resolve => setTimeout(resolve, backoffMs));\n continue;\n }\n throw error;\n }\n }\n\n // This should never be reached, but TypeScript requires it\n throw new OpenAIError('Max retries exceeded');\n}\n\nexport async function transcribeAudio(filePath: string, options: { model?: string, debug?: boolean, debugFile?: string, debugRequestFile?: string, debugResponseFile?: string, outputDirectory?: string } = { model: \"whisper-1\" }): Promise<Transcription> {\n const logger = getLogger();\n const storage = Storage.create({ log: logger.debug });\n let openai: OpenAI | null = null;\n let audioStream: fs.ReadStream | null = null;\n let streamClosed = false;\n\n // Helper function to safely close the stream\n const closeAudioStream = () => {\n if (audioStream && !streamClosed) {\n try {\n // Only call destroy if it exists and the stream isn't already destroyed\n if (typeof audioStream.destroy === 'function' && !audioStream.destroyed) {\n audioStream.destroy();\n }\n streamClosed = true;\n logger.debug('Audio stream closed successfully');\n } catch (streamErr) {\n logger.debug('Failed to destroy audio read stream: %s', (streamErr as Error).message);\n streamClosed = true; // Mark as closed even if destroy failed\n }\n }\n };\n\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 openai = new OpenAI({\n apiKey: apiKey,\n });\n\n logger.debug('Transcribing audio file: %s', filePath);\n\n // Save request debug file if enabled\n if (options.debug && (options.debugRequestFile || options.debugFile)) {\n const requestData = {\n model: options.model || \"whisper-1\",\n file: filePath, // Can't serialize the stream, so just save the file path\n response_format: \"json\",\n };\n const debugFile = options.debugRequestFile || options.debugFile;\n await storage.writeFile(debugFile!, JSON.stringify(requestData, null, 2), 'utf8');\n logger.debug('Wrote request debug file to %s', debugFile);\n }\n\n audioStream = await storage.readStream(filePath);\n\n // Set up error handler for the stream to ensure cleanup on stream errors\n // Only add handler if the stream has the 'on' method (real streams)\n if (audioStream && typeof audioStream.on === 'function') {\n audioStream.on('error', (streamError) => {\n logger.error('Audio stream error: %s', streamError.message);\n closeAudioStream();\n });\n }\n\n let transcription;\n try {\n transcription = await openai.audio.transcriptions.create({\n model: options.model || \"whisper-1\",\n file: audioStream,\n response_format: \"json\",\n });\n // Close the stream immediately after successful API call to prevent race conditions\n closeAudioStream();\n } catch (apiError) {\n // Close the stream immediately if the API call fails\n closeAudioStream();\n throw apiError;\n }\n\n // Save response debug file if enabled\n if (options.debug && (options.debugResponseFile || options.debugFile)) {\n const debugFile = options.debugResponseFile || options.debugFile;\n await storage.writeFile(debugFile!, JSON.stringify(transcription, null, 2), 'utf8');\n logger.debug('Wrote response debug file to %s', debugFile);\n }\n\n const response = transcription;\n if (!response) {\n throw new OpenAIError('No transcription received from OpenAI');\n }\n\n logger.debug('Received transcription from OpenAI: %s', response);\n\n // Archive the audio file and transcription\n try {\n const outputDir = options.outputDirectory || 'output';\n await archiveAudio(filePath, response.text, outputDir);\n } catch (archiveError: any) {\n // Don't fail the transcription if archiving fails, just log the error\n logger.warn('Failed to archive audio file: %s', archiveError.message);\n }\n\n return response;\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 } finally {\n // Ensure the audio stream is properly closed to release file handles\n closeAudioStream();\n // OpenAI client cleanup is handled automatically by the library\n // No manual cleanup needed for newer versions\n }\n}\n"],"names":["getModelForCommand","config","commandName","commandModel","commit","model","release","review","getOpenAIReasoningForCommand","commandReasoning","openaiReasoning","getOpenAIMaxOutputTokensForCommand","commandMaxOutputTokens","openaiMaxOutputTokens","OpenAIError","Error","message","isTokenLimitError","name","error","toLowerCase","includes","isRateLimitError","code","status","createCompletion","messages","options","logger","getLogger","storage","Storage","log","debug","openai","completion","apiKey","process","env","OPENAI_API_KEY","timeoutMs","parseInt","OPENAI_TIMEOUT_MS","OpenAI","timeout","modelToUse","info","maxCompletionTokens","maxTokens","debugRequestFile","debugFile","requestData","max_completion_tokens","response_format","responseFormat","reasoning_effort","writeFile","JSON","stringify","apiOptions","completionPromise","chat","completions","create","timeoutId","timeoutPromise","Promise","_","reject","setTimeout","race","clearTimeout","debugResponseFile","response","choices","content","trim","substring","safeJsonParse","stack","isTokenError","createCompletionWithRetry","retryCallback","maxRetries","attempt","messagesToSend","warn","backoffMs","Math","min","pow","resolve","transcribeAudio","filePath","audioStream","streamClosed","closeAudioStream","destroy","destroyed","streamErr","file","readStream","on","streamError","transcription","audio","transcriptions","apiError","outputDir","outputDirectory","archiveAudio","text","archiveError"],"mappings":"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"}
|
package/dist/util/validation.js
CHANGED
|
@@ -39,6 +39,29 @@
|
|
|
39
39
|
}
|
|
40
40
|
return value;
|
|
41
41
|
};
|
|
42
|
+
/**
|
|
43
|
+
* Sanitizes and truncates direction parameter for safe use in prompts
|
|
44
|
+
* @param direction The direction string to sanitize
|
|
45
|
+
* @param maxLength Maximum length before truncation (default: 2000)
|
|
46
|
+
* @returns Sanitized and truncated direction string
|
|
47
|
+
*/ const sanitizeDirection = (direction, maxLength = 2000)=>{
|
|
48
|
+
if (!direction) {
|
|
49
|
+
return undefined;
|
|
50
|
+
}
|
|
51
|
+
// Remove newlines and excessive whitespace to prevent template breakage
|
|
52
|
+
const sanitized = direction.replace(/\r?\n/g, ' ') // Replace newlines with spaces
|
|
53
|
+
.replace(/\s+/g, ' ') // Replace multiple whitespace with single space
|
|
54
|
+
.trim();
|
|
55
|
+
// Truncate if too long
|
|
56
|
+
if (sanitized.length > maxLength) {
|
|
57
|
+
const truncated = sanitized.substring(0, maxLength - 3) + '...';
|
|
58
|
+
// Log truncation for debugging
|
|
59
|
+
// eslint-disable-next-line no-console
|
|
60
|
+
console.warn(`Direction truncated from ${sanitized.length} to ${truncated.length} characters`);
|
|
61
|
+
return truncated;
|
|
62
|
+
}
|
|
63
|
+
return sanitized;
|
|
64
|
+
};
|
|
42
65
|
/**
|
|
43
66
|
* Validates package.json structure has basic required fields
|
|
44
67
|
*/ const validatePackageJson = (data, context, requireName = true)=>{
|
|
@@ -53,5 +76,5 @@
|
|
|
53
76
|
return data;
|
|
54
77
|
};
|
|
55
78
|
|
|
56
|
-
export { safeJsonParse, validatePackageJson, validateReleaseSummary, validateString };
|
|
79
|
+
export { safeJsonParse, sanitizeDirection, validatePackageJson, validateReleaseSummary, validateString };
|
|
57
80
|
//# sourceMappingURL=validation.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"validation.js","sources":["../../src/util/validation.ts"],"sourcesContent":["/**\n * Runtime validation utilities for safe type handling\n */\n\nexport interface ReleaseSummary {\n title: string;\n body: string;\n}\n\n\n\nexport interface TranscriptionResult {\n text: string;\n [key: string]: any;\n}\n\n/**\n * Validates and safely casts data to ReleaseSummary type\n */\nexport const validateReleaseSummary = (data: any): ReleaseSummary => {\n if (!data || typeof data !== 'object') {\n throw new Error('Invalid release summary: not an object');\n }\n if (typeof data.title !== 'string') {\n throw new Error('Invalid release summary: title must be a string');\n }\n if (typeof data.body !== 'string') {\n throw new Error('Invalid release summary: body must be a string');\n }\n return data as ReleaseSummary;\n};\n\n\n\n/**\n * Validates transcription result has required text property\n */\nexport const validateTranscriptionResult = (data: any): TranscriptionResult => {\n if (!data || typeof data !== 'object') {\n throw new Error('Invalid transcription result: not an object');\n }\n if (typeof data.text !== 'string') {\n throw new Error('Invalid transcription result: text property must be a string');\n }\n return data as TranscriptionResult;\n};\n\n/**\n * Safely parses JSON with error handling\n */\nexport const safeJsonParse = <T = any>(jsonString: string, context?: string): T => {\n try {\n const parsed = JSON.parse(jsonString);\n if (parsed === null || parsed === undefined) {\n throw new Error('Parsed JSON is null or undefined');\n }\n return parsed;\n } catch (error) {\n const contextStr = context ? ` (${context})` : '';\n throw new Error(`Failed to parse JSON${contextStr}: ${error instanceof Error ? error.message : 'Unknown error'}`);\n }\n};\n\n/**\n * Validates that a value is a non-empty string\n */\nexport const validateString = (value: any, fieldName: string): string => {\n if (typeof value !== 'string') {\n throw new Error(`${fieldName} must be a string, got ${typeof value}`);\n }\n if (value.trim() === '') {\n throw new Error(`${fieldName} cannot be empty`);\n }\n return value;\n};\n\n/**\n * Validates that a value exists and has a specific property\n */\nexport const validateHasProperty = (obj: any, property: string, context?: string): void => {\n if (!obj || typeof obj !== 'object') {\n const contextStr = context ? ` in ${context}` : '';\n throw new Error(`Object is null or not an object${contextStr}`);\n }\n if (!(property in obj)) {\n const contextStr = context ? ` in ${context}` : '';\n throw new Error(`Missing required property '${property}'${contextStr}`);\n }\n};\n\n/**\n * Validates package.json structure has basic required fields\n */\nexport const validatePackageJson = (data: any, context?: string, requireName: boolean = true): any => {\n if (!data || typeof data !== 'object') {\n const contextStr = context ? ` (${context})` : '';\n throw new Error(`Invalid package.json${contextStr}: not an object`);\n }\n if (requireName && typeof data.name !== 'string') {\n const contextStr = context ? ` (${context})` : '';\n throw new Error(`Invalid package.json${contextStr}: name must be a string`);\n }\n return data;\n};\n"],"names":["validateReleaseSummary","data","Error","title","body","safeJsonParse","jsonString","context","parsed","JSON","parse","undefined","error","contextStr","message","validateString","value","fieldName","trim","validatePackageJson","requireName","name"],"mappings":"AAAA;;;;IAmBO,MAAMA,sBAAAA,GAAyB,CAACC,IAAAA,GAAAA;AACnC,IAAA,IAAI,CAACA,IAAAA,IAAQ,OAAOA,IAAAA,KAAS,QAAA,EAAU;AACnC,QAAA,MAAM,IAAIC,KAAAA,CAAM,wCAAA,CAAA;AACpB,IAAA;AACA,IAAA,IAAI,OAAOD,IAAAA,CAAKE,KAAK,KAAK,QAAA,EAAU;AAChC,QAAA,MAAM,IAAID,KAAAA,CAAM,iDAAA,CAAA;AACpB,IAAA;AACA,IAAA,IAAI,OAAOD,IAAAA,CAAKG,IAAI,KAAK,QAAA,EAAU;AAC/B,QAAA,MAAM,IAAIF,KAAAA,CAAM,gDAAA,CAAA;AACpB,IAAA;IACA,OAAOD,IAAAA;AACX;AAiBA;;AAEC,IACM,MAAMI,aAAAA,GAAgB,CAAUC,UAAAA,EAAoBC,OAAAA,GAAAA;IACvD,IAAI;QACA,MAAMC,MAAAA,GAASC,IAAAA,CAAKC,KAAK,CAACJ,UAAAA,CAAAA;QAC1B,IAAIE,MAAAA,KAAW,IAAA,IAAQA,MAAAA,KAAWG,SAAAA,EAAW;AACzC,YAAA,MAAM,IAAIT,KAAAA,CAAM,kCAAA,CAAA;AACpB,QAAA;QACA,OAAOM,MAAAA;AACX,IAAA,CAAA,CAAE,OAAOI,KAAAA,EAAO;QACZ,MAAMC,UAAAA,GAAaN,UAAU,CAAC,EAAE,EAAEA,OAAAA,CAAQ,CAAC,CAAC,GAAG,EAAA;AAC/C,QAAA,MAAM,IAAIL,KAAAA,CAAM,CAAC,oBAAoB,EAAEW,UAAAA,CAAW,EAAE,EAAED,KAAAA,YAAiBV,KAAAA,GAAQU,KAAAA,CAAME,OAAO,GAAG,eAAA,CAAA,CAAiB,CAAA;AACpH,IAAA;AACJ;AAEA;;AAEC,IACM,MAAMC,cAAAA,GAAiB,CAACC,KAAAA,EAAYC,SAAAA,GAAAA;IACvC,IAAI,OAAOD,UAAU,QAAA,EAAU;AAC3B,QAAA,MAAM,IAAId,KAAAA,CAAM,CAAA,EAAGe,UAAU,uBAAuB,EAAE,OAAOD,KAAAA,CAAAA,CAAO,CAAA;AACxE,IAAA;IACA,IAAIA,KAAAA,CAAME,IAAI,EAAA,KAAO,EAAA,EAAI;AACrB,QAAA,MAAM,IAAIhB,KAAAA,CAAM,CAAA,EAAGe,SAAAA,CAAU,gBAAgB,CAAC,CAAA;AAClD,IAAA;IACA,OAAOD,KAAAA;AACX;AAgBA;;AAEC,IACM,
|
|
1
|
+
{"version":3,"file":"validation.js","sources":["../../src/util/validation.ts"],"sourcesContent":["/**\n * Runtime validation utilities for safe type handling\n */\n\nexport interface ReleaseSummary {\n title: string;\n body: string;\n}\n\n\n\nexport interface TranscriptionResult {\n text: string;\n [key: string]: any;\n}\n\n/**\n * Validates and safely casts data to ReleaseSummary type\n */\nexport const validateReleaseSummary = (data: any): ReleaseSummary => {\n if (!data || typeof data !== 'object') {\n throw new Error('Invalid release summary: not an object');\n }\n if (typeof data.title !== 'string') {\n throw new Error('Invalid release summary: title must be a string');\n }\n if (typeof data.body !== 'string') {\n throw new Error('Invalid release summary: body must be a string');\n }\n return data as ReleaseSummary;\n};\n\n\n\n/**\n * Validates transcription result has required text property\n */\nexport const validateTranscriptionResult = (data: any): TranscriptionResult => {\n if (!data || typeof data !== 'object') {\n throw new Error('Invalid transcription result: not an object');\n }\n if (typeof data.text !== 'string') {\n throw new Error('Invalid transcription result: text property must be a string');\n }\n return data as TranscriptionResult;\n};\n\n/**\n * Safely parses JSON with error handling\n */\nexport const safeJsonParse = <T = any>(jsonString: string, context?: string): T => {\n try {\n const parsed = JSON.parse(jsonString);\n if (parsed === null || parsed === undefined) {\n throw new Error('Parsed JSON is null or undefined');\n }\n return parsed;\n } catch (error) {\n const contextStr = context ? ` (${context})` : '';\n throw new Error(`Failed to parse JSON${contextStr}: ${error instanceof Error ? error.message : 'Unknown error'}`);\n }\n};\n\n/**\n * Validates that a value is a non-empty string\n */\nexport const validateString = (value: any, fieldName: string): string => {\n if (typeof value !== 'string') {\n throw new Error(`${fieldName} must be a string, got ${typeof value}`);\n }\n if (value.trim() === '') {\n throw new Error(`${fieldName} cannot be empty`);\n }\n return value;\n};\n\n/**\n * Sanitizes and truncates direction parameter for safe use in prompts\n * @param direction The direction string to sanitize\n * @param maxLength Maximum length before truncation (default: 2000)\n * @returns Sanitized and truncated direction string\n */\nexport const sanitizeDirection = (direction: string | undefined, maxLength: number = 2000): string | undefined => {\n if (!direction) {\n return undefined;\n }\n\n // Remove newlines and excessive whitespace to prevent template breakage\n const sanitized = direction\n .replace(/\\r?\\n/g, ' ') // Replace newlines with spaces\n .replace(/\\s+/g, ' ') // Replace multiple whitespace with single space\n .trim();\n\n // Truncate if too long\n if (sanitized.length > maxLength) {\n const truncated = sanitized.substring(0, maxLength - 3) + '...';\n // Log truncation for debugging\n // eslint-disable-next-line no-console\n console.warn(`Direction truncated from ${sanitized.length} to ${truncated.length} characters`);\n return truncated;\n }\n\n return sanitized;\n};\n\n/**\n * Validates that a value exists and has a specific property\n */\nexport const validateHasProperty = (obj: any, property: string, context?: string): void => {\n if (!obj || typeof obj !== 'object') {\n const contextStr = context ? ` in ${context}` : '';\n throw new Error(`Object is null or not an object${contextStr}`);\n }\n if (!(property in obj)) {\n const contextStr = context ? ` in ${context}` : '';\n throw new Error(`Missing required property '${property}'${contextStr}`);\n }\n};\n\n/**\n * Validates package.json structure has basic required fields\n */\nexport const validatePackageJson = (data: any, context?: string, requireName: boolean = true): any => {\n if (!data || typeof data !== 'object') {\n const contextStr = context ? ` (${context})` : '';\n throw new Error(`Invalid package.json${contextStr}: not an object`);\n }\n if (requireName && typeof data.name !== 'string') {\n const contextStr = context ? ` (${context})` : '';\n throw new Error(`Invalid package.json${contextStr}: name must be a string`);\n }\n return data;\n};\n"],"names":["validateReleaseSummary","data","Error","title","body","safeJsonParse","jsonString","context","parsed","JSON","parse","undefined","error","contextStr","message","validateString","value","fieldName","trim","sanitizeDirection","direction","maxLength","sanitized","replace","length","truncated","substring","console","warn","validatePackageJson","requireName","name"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@eldrforge/kodrdriv",
|
|
3
|
-
"version": "1.2.
|
|
3
|
+
"version": "1.2.2",
|
|
4
4
|
"description": "Create Intelligent Release Notes or Change Logs from Git",
|
|
5
5
|
"main": "dist/main.js",
|
|
6
6
|
"type": "module",
|
|
@@ -48,7 +48,7 @@
|
|
|
48
48
|
"glob": "^11.0.3",
|
|
49
49
|
"js-yaml": "^4.1.0",
|
|
50
50
|
"moment-timezone": "^0.6.0",
|
|
51
|
-
"openai": "^5.
|
|
51
|
+
"openai": "^5.12.2",
|
|
52
52
|
"semver": "^7.7.2",
|
|
53
53
|
"shell-escape": "^0.2.0",
|
|
54
54
|
"winston": "^3.17.0",
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
#!/bin/bash
|
|
2
|
+
|
|
3
|
+
echo "Testing the new two-phase review flow..."
|
|
4
|
+
echo "This will test the file selection phase and then the analysis phase"
|
|
5
|
+
|
|
6
|
+
# Test with a single file in the test-project directory
|
|
7
|
+
echo "Test review note content" > test-project/test-review.md
|
|
8
|
+
|
|
9
|
+
echo "Running review command in test-project directory..."
|
|
10
|
+
echo "You should see:"
|
|
11
|
+
echo "1. File selection phase with c/s/a options"
|
|
12
|
+
echo "2. Analysis phase after files are selected"
|
|
13
|
+
|
|
14
|
+
cd test-project
|
|
15
|
+
../dist/main.js review --directory .
|