@eldrforge/kodrdriv 1.2.23 → 1.2.24
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/PARALLEL-EXECUTION-FIXES.md +132 -0
- package/PARALLEL_EXECUTION_FIX.md +146 -0
- package/RECOVERY-FIXES.md +72 -0
- package/dist/arguments.js +26 -3
- package/dist/arguments.js.map +1 -1
- package/dist/commands/audio-commit.js +3 -3
- package/dist/commands/audio-commit.js.map +1 -1
- package/dist/commands/audio-review.js +13 -13
- package/dist/commands/audio-review.js.map +1 -1
- package/dist/commands/link.js +13 -13
- package/dist/commands/link.js.map +1 -1
- package/dist/commands/publish.js +200 -146
- package/dist/commands/publish.js.map +1 -1
- package/dist/commands/review.js +6 -6
- package/dist/commands/review.js.map +1 -1
- package/dist/commands/select-audio.js +4 -4
- package/dist/commands/select-audio.js.map +1 -1
- package/dist/commands/tree.js +242 -318
- package/dist/commands/tree.js.map +1 -1
- package/dist/commands/unlink.js +8 -8
- package/dist/commands/unlink.js.map +1 -1
- package/dist/commands/versions.js +3 -3
- package/dist/commands/versions.js.map +1 -1
- package/dist/constants.js +4 -4
- package/dist/constants.js.map +1 -1
- package/dist/content/diff.js +5 -2
- package/dist/content/diff.js.map +1 -1
- package/dist/content/files.js +4 -4
- package/dist/content/files.js.map +1 -1
- package/dist/execution/CommandValidator.js +160 -0
- package/dist/execution/CommandValidator.js.map +1 -0
- package/dist/execution/DependencyChecker.js +102 -0
- package/dist/execution/DependencyChecker.js.map +1 -0
- package/dist/execution/DynamicTaskPool.js +455 -0
- package/dist/execution/DynamicTaskPool.js.map +1 -0
- package/dist/execution/RecoveryManager.js +502 -0
- package/dist/execution/RecoveryManager.js.map +1 -0
- package/dist/execution/ResourceMonitor.js +125 -0
- package/dist/execution/ResourceMonitor.js.map +1 -0
- package/dist/execution/Scheduler.js +98 -0
- package/dist/execution/Scheduler.js.map +1 -0
- package/dist/execution/TreeExecutionAdapter.js +170 -0
- package/dist/execution/TreeExecutionAdapter.js.map +1 -0
- package/dist/logging.js +3 -3
- package/dist/logging.js.map +1 -1
- package/dist/ui/ProgressFormatter.js +230 -0
- package/dist/ui/ProgressFormatter.js.map +1 -0
- package/dist/util/checkpointManager.js +168 -0
- package/dist/util/checkpointManager.js.map +1 -0
- package/dist/util/dependencyGraph.js +224 -0
- package/dist/util/dependencyGraph.js.map +1 -0
- package/dist/util/fileLock.js +204 -0
- package/dist/util/fileLock.js.map +1 -0
- package/dist/util/general.js +5 -5
- package/dist/util/general.js.map +1 -1
- package/dist/util/gitMutex.js +116 -0
- package/dist/util/gitMutex.js.map +1 -0
- package/dist/util/mutex.js +96 -0
- package/dist/util/mutex.js.map +1 -0
- package/dist/util/performance.js +4 -4
- package/dist/util/performance.js.map +1 -1
- package/dist/util/safety.js +4 -4
- package/dist/util/safety.js.map +1 -1
- package/dist/util/storage.js +2 -2
- package/dist/util/storage.js.map +1 -1
- package/package.json +6 -6
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 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\n/**\n * Increment prerelease version with a specific tag\n * Examples:\n * - incrementPrereleaseVersion(\"1.2.3-dev.0\", \"dev\") => \"1.2.3-dev.1\"\n * - incrementPrereleaseVersion(\"1.2.3\", \"dev\") => \"1.2.3-dev.0\"\n * - incrementPrereleaseVersion(\"1.2.3-dev.5\", \"test\") => \"1.2.3-test.0\"\n */\nexport const incrementPrereleaseVersion = (version: string, tag: string): string => {\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split on dots but only use first 3 parts for major.minor.patch\n // This handles cases like \"1.2.3-dev.5\" correctly\n const dotParts = cleanVersion.split('.');\n if (dotParts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n const major = dotParts[0];\n const minor = dotParts[1];\n\n // Reconstruct the patch part - everything after the second dot\n const patchAndPrerelease = dotParts.slice(2).join('.');\n const patchComponents = patchAndPrerelease.split('-');\n const patchNumber = patchComponents[0];\n\n if (patchComponents.length > 1) {\n // Already has prerelease (e.g., \"3-dev.0\" or \"3-test.2\")\n const prereleaseString = patchComponents.slice(1).join('-'); // Handle multiple dashes\n const prereleaseComponents = prereleaseString.split('.');\n const existingTag = prereleaseComponents[0];\n const existingPrereleaseVersion = prereleaseComponents[1];\n\n if (existingTag === tag) {\n // Same tag, increment the prerelease version\n const prereleaseNumber = parseInt(existingPrereleaseVersion) || 0;\n return `${major}.${minor}.${patchNumber}-${tag}.${prereleaseNumber + 1}`;\n } else {\n // Different tag, start at 0\n return `${major}.${minor}.${patchNumber}-${tag}.0`;\n }\n } else {\n // No prerelease yet, add it\n return `${major}.${minor}.${patchNumber}-${tag}.0`;\n }\n};\n\n/**\n * Convert prerelease version to release version\n * Examples:\n * - convertToReleaseVersion(\"1.2.3-dev.5\") => \"1.2.3\"\n * - convertToReleaseVersion(\"1.2.3-test.2\") => \"1.2.3\"\n * - convertToReleaseVersion(\"1.2.3\") => \"1.2.3\"\n */\nexport const convertToReleaseVersion = (version: string): string => {\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split on dots but only use first 3 parts for major.minor.patch\n const dotParts = cleanVersion.split('.');\n if (dotParts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n const major = dotParts[0];\n const minor = dotParts[1];\n\n // Reconstruct the patch part - everything after the second dot\n const patchAndPrerelease = dotParts.slice(2).join('.');\n const patchComponents = patchAndPrerelease.split('-');\n const patchNumber = patchComponents[0];\n\n return `${major}.${minor}.${patchNumber}`;\n};\n\n/**\n * Get version from a specific branch's package.json\n */\nexport const getVersionFromBranch = async (branchName: string): Promise<string | null> => {\n const { runSecure, validateGitRef, safeJsonParse, validatePackageJson } = await import('@eldrforge/git-tools');\n\n try {\n // Validate branch name to prevent injection\n if (!validateGitRef(branchName)) {\n throw new Error(`Invalid branch name: ${branchName}`);\n }\n const { stdout } = await runSecure('git', ['show', `${branchName}:package.json`]);\n const packageJson = safeJsonParse(stdout, 'package.json');\n const validated = validatePackageJson(packageJson, 'package.json');\n return validated.version;\n } catch {\n // Return null if we can't get the version (branch may not exist or no package.json)\n return null;\n }\n};\n\n/**\n * Calculate target version based on branch configuration\n * SEMANTICS: The version config specifies what version should be ON the target branch\n */\nexport const calculateBranchDependentVersion = async (\n currentVersion: string,\n currentBranch: string,\n branchesConfig: any,\n targetBranch?: string\n): Promise<{ version: string; targetBranch: string }> => {\n const { getLogger } = await import('../logging');\n const logger = getLogger();\n\n // Look up the source branch to find the target branch\n if (!branchesConfig || !branchesConfig[currentBranch]) {\n // Use default configuration from constants\n const { KODRDRIV_DEFAULTS } = await import('../constants');\n const defaultConfig = KODRDRIV_DEFAULTS.branches as any;\n\n if (defaultConfig && defaultConfig[currentBranch]) {\n const sourceConfig = defaultConfig[currentBranch];\n const finalTargetBranch = sourceConfig.targetBranch || targetBranch || 'main';\n\n // Look at target branch's version config to determine what version it should have\n const targetConfig = defaultConfig[finalTargetBranch];\n\n logger.info(`🎯 Using default branch configuration: ${currentBranch} → ${finalTargetBranch}`);\n\n if (!targetConfig?.version) {\n const defaultVersion = incrementPatchVersion(currentVersion);\n logger.debug(`No version config for target branch '${finalTargetBranch}', using default increment`);\n return { version: defaultVersion, targetBranch: finalTargetBranch };\n }\n\n return calculateVersionFromTargetConfig(currentVersion, finalTargetBranch, targetConfig.version, logger);\n }\n\n // No config at all, use traditional defaults\n const defaultTargetBranch = targetBranch || 'main';\n const defaultVersion = incrementPatchVersion(currentVersion);\n logger.debug(`No branch-specific config found for '${currentBranch}', using defaults`);\n return { version: defaultVersion, targetBranch: defaultTargetBranch };\n }\n\n const sourceConfig = branchesConfig[currentBranch];\n const finalTargetBranch = sourceConfig.targetBranch || targetBranch || 'main';\n\n // Look at target branch's version config to determine what version it should have\n const targetConfig = branchesConfig[finalTargetBranch];\n\n logger.info(`🎯 Using branch-dependent targeting: ${currentBranch} → ${finalTargetBranch}`);\n\n if (!targetConfig?.version) {\n // No version config for target, use default increment\n const defaultVersion = incrementPatchVersion(currentVersion);\n logger.debug(`No version config for target branch '${finalTargetBranch}', using default increment`);\n return { version: defaultVersion, targetBranch: finalTargetBranch };\n }\n\n return calculateVersionFromTargetConfig(currentVersion, finalTargetBranch, targetConfig.version, logger);\n};\n\n/**\n * Calculate version based on target branch configuration\n */\nconst calculateVersionFromTargetConfig = async (\n currentVersion: string,\n targetBranch: string,\n versionConfig: any,\n logger: any\n): Promise<{ version: string; targetBranch: string }> => {\n if (versionConfig.type === 'release') {\n // Convert to release version (remove prerelease tags)\n const releaseVersion = convertToReleaseVersion(currentVersion);\n logger.info(`📦 Converting to release version: ${currentVersion} → ${releaseVersion}`);\n return { version: releaseVersion, targetBranch };\n } else if (versionConfig.type === 'prerelease') {\n if (!versionConfig.tag) {\n throw new Error(`Prerelease version type requires a tag in branch configuration`);\n }\n\n const tag = versionConfig.tag;\n\n if (versionConfig.increment) {\n // Check if there's already a version with this tag in the target branch\n const targetBranchVersion = await getVersionFromBranch(targetBranch);\n\n if (targetBranchVersion) {\n // Use the target branch version as the base and increment\n const newVersion = incrementPrereleaseVersion(targetBranchVersion, tag);\n logger.info(`📦 Incrementing prerelease in target branch: ${targetBranchVersion} → ${newVersion}`);\n return { version: newVersion, targetBranch };\n } else {\n // No version in target branch, use current version as base\n const newVersion = incrementPrereleaseVersion(currentVersion, tag);\n logger.info(`📦 Creating new prerelease version: ${currentVersion} → ${newVersion}`);\n return { version: newVersion, targetBranch };\n }\n } else {\n // Just add/change the prerelease tag without incrementing\n const baseVersion = convertToReleaseVersion(currentVersion);\n const newVersion = `${baseVersion}-${tag}.0`;\n logger.info(`📦 Setting prerelease tag: ${currentVersion} → ${newVersion}`);\n return { version: newVersion, targetBranch };\n }\n }\n\n throw new Error(`Invalid version type: ${versionConfig.type}`);\n};\n\n\n/**\n * Find the development branch from branches configuration\n * Returns the branch marked with developmentBranch: true\n */\nexport const findDevelopmentBranch = (branchesConfig: any): string | null => {\n if (!branchesConfig || typeof branchesConfig !== 'object') {\n return null;\n }\n\n for (const [branchName, branchConfig] of Object.entries(branchesConfig)) {\n if (branchConfig && typeof branchConfig === 'object' && (branchConfig as any).developmentBranch === true) {\n return branchName;\n }\n }\n\n return null;\n};\n\n/**\n * Check if two prerelease versions have the same tag\n * Examples:\n * - haveSamePrereleaseTag(\"1.2.3-dev.0\", \"1.2.3-dev.5\") => true\n * - haveSamePrereleaseTag(\"1.2.3-dev.0\", \"1.2.3-test.0\") => false\n * - haveSamePrereleaseTag(\"1.2.3\", \"1.2.3-dev.0\") => false\n */\nexport const haveSamePrereleaseTag = (version1: string, version2: string): boolean => {\n const extractTag = (version: string): string | null => {\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n const parts = cleanVersion.split('.');\n if (parts.length < 3) return null;\n\n const patchAndPrerelease = parts.slice(2).join('.');\n const patchComponents = patchAndPrerelease.split('-');\n\n if (patchComponents.length > 1) {\n const prereleaseString = patchComponents.slice(1).join('-');\n const prereleaseComponents = prereleaseString.split('.');\n return prereleaseComponents[0] || null;\n }\n\n return null;\n };\n\n const tag1 = extractTag(version1);\n const tag2 = extractTag(version2);\n\n return tag1 !== null && tag2 !== null && tag1 === tag2;\n};\n\nexport const checkIfTagExists = async (tagName: string): Promise<boolean> => {\n const { runSecure, validateGitRef } = await import('@eldrforge/git-tools');\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","incrementPrereleaseVersion","tag","dotParts","major","minor","patchAndPrerelease","join","prereleaseString","prereleaseComponents","existingTag","existingPrereleaseVersion","prereleaseNumber","convertToReleaseVersion","getVersionFromBranch","branchName","runSecure","validateGitRef","safeJsonParse","validatePackageJson","stdout","packageJson","validated","calculateBranchDependentVersion","currentBranch","branchesConfig","targetBranch","getLogger","logger","KODRDRIV_DEFAULTS","defaultConfig","branches","sourceConfig","finalTargetBranch","targetConfig","info","defaultVersion","debug","calculateVersionFromTargetConfig","defaultTargetBranch","versionConfig","type","releaseVersion","increment","targetBranchVersion","baseVersion","findDevelopmentBranch","branchConfig","entries","developmentBranch","checkIfTagExists","tagName","trim","confirmVersionInteractively","proposedVersion","targetVersionInput","getUserChoice","getUserTextInput","requireTTY","choices","label","choice","customVersion","cleanCustomVersion","getOutputPath","outputDirectory","filename","path","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","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\n/**\n * Increment prerelease version with a specific tag\n * Examples:\n * - incrementPrereleaseVersion(\"1.2.3-dev.0\", \"dev\") => \"1.2.3-dev.1\"\n * - incrementPrereleaseVersion(\"1.2.3\", \"dev\") => \"1.2.3-dev.0\"\n * - incrementPrereleaseVersion(\"1.2.3-dev.5\", \"test\") => \"1.2.3-test.0\"\n */\nexport const incrementPrereleaseVersion = (version: string, tag: string): string => {\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split on dots but only use first 3 parts for major.minor.patch\n // This handles cases like \"1.2.3-dev.5\" correctly\n const dotParts = cleanVersion.split('.');\n if (dotParts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n const major = dotParts[0];\n const minor = dotParts[1];\n\n // Reconstruct the patch part - everything after the second dot\n const patchAndPrerelease = dotParts.slice(2).join('.');\n const patchComponents = patchAndPrerelease.split('-');\n const patchNumber = patchComponents[0];\n\n if (patchComponents.length > 1) {\n // Already has prerelease (e.g., \"3-dev.0\" or \"3-test.2\")\n const prereleaseString = patchComponents.slice(1).join('-'); // Handle multiple dashes\n const prereleaseComponents = prereleaseString.split('.');\n const existingTag = prereleaseComponents[0];\n const existingPrereleaseVersion = prereleaseComponents[1];\n\n if (existingTag === tag) {\n // Same tag, increment the prerelease version\n const prereleaseNumber = parseInt(existingPrereleaseVersion) || 0;\n return `${major}.${minor}.${patchNumber}-${tag}.${prereleaseNumber + 1}`;\n } else {\n // Different tag, start at 0\n return `${major}.${minor}.${patchNumber}-${tag}.0`;\n }\n } else {\n // No prerelease yet, add it\n return `${major}.${minor}.${patchNumber}-${tag}.0`;\n }\n};\n\n/**\n * Convert prerelease version to release version\n * Examples:\n * - convertToReleaseVersion(\"1.2.3-dev.5\") => \"1.2.3\"\n * - convertToReleaseVersion(\"1.2.3-test.2\") => \"1.2.3\"\n * - convertToReleaseVersion(\"1.2.3\") => \"1.2.3\"\n */\nexport const convertToReleaseVersion = (version: string): string => {\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n\n // Split on dots but only use first 3 parts for major.minor.patch\n const dotParts = cleanVersion.split('.');\n if (dotParts.length < 3) {\n throw new Error(`Invalid version string: ${version}`);\n }\n\n const major = dotParts[0];\n const minor = dotParts[1];\n\n // Reconstruct the patch part - everything after the second dot\n const patchAndPrerelease = dotParts.slice(2).join('.');\n const patchComponents = patchAndPrerelease.split('-');\n const patchNumber = patchComponents[0];\n\n return `${major}.${minor}.${patchNumber}`;\n};\n\n/**\n * Get version from a specific branch's package.json\n */\nexport const getVersionFromBranch = async (branchName: string): Promise<string | null> => {\n const { runSecure, validateGitRef, safeJsonParse, validatePackageJson } = await import('@eldrforge/git-tools');\n\n try {\n // Validate branch name to prevent injection\n if (!validateGitRef(branchName)) {\n throw new Error(`Invalid branch name: ${branchName}`);\n }\n const { stdout } = await runSecure('git', ['show', `${branchName}:package.json`]);\n const packageJson = safeJsonParse(stdout, 'package.json');\n const validated = validatePackageJson(packageJson, 'package.json');\n return validated.version;\n } catch {\n // Return null if we can't get the version (branch may not exist or no package.json)\n return null;\n }\n};\n\n/**\n * Calculate target version based on branch configuration\n * SEMANTICS: The version config specifies what version should be ON the target branch\n */\nexport const calculateBranchDependentVersion = async (\n currentVersion: string,\n currentBranch: string,\n branchesConfig: any,\n targetBranch?: string\n): Promise<{ version: string; targetBranch: string }> => {\n const { getLogger } = await import('../logging');\n const logger = getLogger();\n\n // Look up the source branch to find the target branch\n if (!branchesConfig || !branchesConfig[currentBranch]) {\n // Use default configuration from constants\n const { KODRDRIV_DEFAULTS } = await import('../constants');\n const defaultConfig = KODRDRIV_DEFAULTS.branches as any;\n\n if (defaultConfig && defaultConfig[currentBranch]) {\n const sourceConfig = defaultConfig[currentBranch];\n const finalTargetBranch = sourceConfig.targetBranch || targetBranch || 'main';\n\n // Look at target branch's version config to determine what version it should have\n const targetConfig = defaultConfig[finalTargetBranch];\n\n logger.info(`🎯 Using default branch configuration: ${currentBranch} → ${finalTargetBranch}`);\n\n if (!targetConfig?.version) {\n const defaultVersion = incrementPatchVersion(currentVersion);\n logger.debug(`No version config for target branch '${finalTargetBranch}', using default increment`);\n return { version: defaultVersion, targetBranch: finalTargetBranch };\n }\n\n return calculateVersionFromTargetConfig(currentVersion, finalTargetBranch, targetConfig.version, logger);\n }\n\n // No config at all, use traditional defaults\n const defaultTargetBranch = targetBranch || 'main';\n const defaultVersion = incrementPatchVersion(currentVersion);\n logger.debug(`No branch-specific config found for '${currentBranch}', using defaults`);\n return { version: defaultVersion, targetBranch: defaultTargetBranch };\n }\n\n const sourceConfig = branchesConfig[currentBranch];\n const finalTargetBranch = sourceConfig.targetBranch || targetBranch || 'main';\n\n // Look at target branch's version config to determine what version it should have\n const targetConfig = branchesConfig[finalTargetBranch];\n\n logger.info(`🎯 Using branch-dependent targeting: ${currentBranch} → ${finalTargetBranch}`);\n\n if (!targetConfig?.version) {\n // No version config for target, use default increment\n const defaultVersion = incrementPatchVersion(currentVersion);\n logger.debug(`No version config for target branch '${finalTargetBranch}', using default increment`);\n return { version: defaultVersion, targetBranch: finalTargetBranch };\n }\n\n return calculateVersionFromTargetConfig(currentVersion, finalTargetBranch, targetConfig.version, logger);\n};\n\n/**\n * Calculate version based on target branch configuration\n */\nconst calculateVersionFromTargetConfig = async (\n currentVersion: string,\n targetBranch: string,\n versionConfig: any,\n logger: any\n): Promise<{ version: string; targetBranch: string }> => {\n if (versionConfig.type === 'release') {\n // Convert to release version (remove prerelease tags)\n const releaseVersion = convertToReleaseVersion(currentVersion);\n logger.info(`📦 Converting to release version: ${currentVersion} → ${releaseVersion}`);\n return { version: releaseVersion, targetBranch };\n } else if (versionConfig.type === 'prerelease') {\n if (!versionConfig.tag) {\n throw new Error(`Prerelease version type requires a tag in branch configuration`);\n }\n\n const tag = versionConfig.tag;\n\n if (versionConfig.increment) {\n // Check if there's already a version with this tag in the target branch\n const targetBranchVersion = await getVersionFromBranch(targetBranch);\n\n if (targetBranchVersion) {\n // Use the target branch version as the base and increment\n const newVersion = incrementPrereleaseVersion(targetBranchVersion, tag);\n logger.info(`📦 Incrementing prerelease in target branch: ${targetBranchVersion} → ${newVersion}`);\n return { version: newVersion, targetBranch };\n } else {\n // No version in target branch, use current version as base\n const newVersion = incrementPrereleaseVersion(currentVersion, tag);\n logger.info(`📦 Creating new prerelease version: ${currentVersion} → ${newVersion}`);\n return { version: newVersion, targetBranch };\n }\n } else {\n // Just add/change the prerelease tag without incrementing\n const baseVersion = convertToReleaseVersion(currentVersion);\n const newVersion = `${baseVersion}-${tag}.0`;\n logger.info(`📦 Setting prerelease tag: ${currentVersion} → ${newVersion}`);\n return { version: newVersion, targetBranch };\n }\n }\n\n throw new Error(`Invalid version type: ${versionConfig.type}`);\n};\n\n\n/**\n * Find the development branch from branches configuration\n * Returns the branch marked with developmentBranch: true\n */\nexport const findDevelopmentBranch = (branchesConfig: any): string | null => {\n if (!branchesConfig || typeof branchesConfig !== 'object') {\n return null;\n }\n\n for (const [branchName, branchConfig] of Object.entries(branchesConfig)) {\n if (branchConfig && typeof branchConfig === 'object' && (branchConfig as any).developmentBranch === true) {\n return branchName;\n }\n }\n\n return null;\n};\n\n/**\n * Check if two prerelease versions have the same tag\n * Examples:\n * - haveSamePrereleaseTag(\"1.2.3-dev.0\", \"1.2.3-dev.5\") => true\n * - haveSamePrereleaseTag(\"1.2.3-dev.0\", \"1.2.3-test.0\") => false\n * - haveSamePrereleaseTag(\"1.2.3\", \"1.2.3-dev.0\") => false\n */\nexport const haveSamePrereleaseTag = (version1: string, version2: string): boolean => {\n const extractTag = (version: string): string | null => {\n const cleanVersion = version.startsWith('v') ? version.slice(1) : version;\n const parts = cleanVersion.split('.');\n if (parts.length < 3) return null;\n\n const patchAndPrerelease = parts.slice(2).join('.');\n const patchComponents = patchAndPrerelease.split('-');\n\n if (patchComponents.length > 1) {\n const prereleaseString = patchComponents.slice(1).join('-');\n const prereleaseComponents = prereleaseString.split('.');\n return prereleaseComponents[0] || null;\n }\n\n return null;\n };\n\n const tag1 = extractTag(version1);\n const tag2 = extractTag(version2);\n\n return tag1 !== null && tag2 !== null && tag1 === tag2;\n};\n\nexport const checkIfTagExists = async (tagName: string): Promise<boolean> => {\n const { runSecure, validateGitRef } = await import('@eldrforge/git-tools');\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","incrementPrereleaseVersion","tag","dotParts","major","minor","patchAndPrerelease","join","prereleaseString","prereleaseComponents","existingTag","existingPrereleaseVersion","prereleaseNumber","convertToReleaseVersion","getVersionFromBranch","branchName","runSecure","validateGitRef","safeJsonParse","validatePackageJson","stdout","packageJson","validated","calculateBranchDependentVersion","currentBranch","branchesConfig","targetBranch","getLogger","logger","KODRDRIV_DEFAULTS","defaultConfig","branches","sourceConfig","finalTargetBranch","targetConfig","info","defaultVersion","debug","calculateVersionFromTargetConfig","defaultTargetBranch","versionConfig","type","releaseVersion","increment","targetBranchVersion","baseVersion","findDevelopmentBranch","branchConfig","entries","developmentBranch","checkIfTagExists","tagName","trim","confirmVersionInteractively","proposedVersion","targetVersionInput","getUserChoice","getUserTextInput","requireTTY","choices","label","choice","customVersion","cleanCustomVersion","getOutputPath","outputDirectory","filename","path","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","ensureDirectory","extname","archivedAudioFilename","archivedTranscriptFilename","archivedAudioPath","archivedTranscriptPath","isFileReadable","audioBuffer","fs","promises","readFile","writeFile","warn","transcriptContent","toISOString","audioPath","transcriptPath","error","message"],"mappings":"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"}
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
import * as path from 'path';
|
|
2
|
+
import { statSync } from 'fs';
|
|
3
|
+
import { RepositoryFileLockManager } from './fileLock.js';
|
|
4
|
+
import { getLogger } from '../logging.js';
|
|
5
|
+
|
|
6
|
+
function _define_property(obj, key, value) {
|
|
7
|
+
if (key in obj) {
|
|
8
|
+
Object.defineProperty(obj, key, {
|
|
9
|
+
value: value,
|
|
10
|
+
enumerable: true,
|
|
11
|
+
configurable: true,
|
|
12
|
+
writable: true
|
|
13
|
+
});
|
|
14
|
+
} else {
|
|
15
|
+
obj[key] = value;
|
|
16
|
+
}
|
|
17
|
+
return obj;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Manages per-repository locks for git operations (cross-process safe)
|
|
21
|
+
* Prevents concurrent git operations in the same repository (which cause .git/index.lock conflicts)
|
|
22
|
+
* while still allowing parallel operations across different repositories
|
|
23
|
+
*
|
|
24
|
+
* Uses file-based locks to coordinate across multiple processes (e.g., parallel tree execution)
|
|
25
|
+
*/ class RepositoryMutexManager {
|
|
26
|
+
/**
|
|
27
|
+
* Execute a git operation with repository-level locking
|
|
28
|
+
* @param packagePath Path to the package (will find its git repo root)
|
|
29
|
+
* @param operation The async operation to execute under lock
|
|
30
|
+
* @param operationName Optional name for logging
|
|
31
|
+
* @returns Result of the operation
|
|
32
|
+
*/ async withGitLock(packagePath, operation, operationName) {
|
|
33
|
+
const repoPath = getGitRepositoryRoot(packagePath);
|
|
34
|
+
if (!repoPath) {
|
|
35
|
+
// Not in a git repository, execute without lock
|
|
36
|
+
this.logger.debug(`No git repository found for ${packagePath}, executing without lock`);
|
|
37
|
+
return await operation();
|
|
38
|
+
}
|
|
39
|
+
return await this.lockManager.withGitLock(repoPath, operation, operationName);
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Destroy all locks and clean up resources
|
|
43
|
+
*/ destroy() {
|
|
44
|
+
this.lockManager.destroy();
|
|
45
|
+
}
|
|
46
|
+
constructor(){
|
|
47
|
+
_define_property(this, "lockManager", void 0);
|
|
48
|
+
_define_property(this, "logger", getLogger());
|
|
49
|
+
this.lockManager = new RepositoryFileLockManager();
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* Find the git repository root for a given path
|
|
54
|
+
* Walks up the directory tree until it finds a .git directory
|
|
55
|
+
* @param startPath Starting path (can be a file or directory)
|
|
56
|
+
* @returns Absolute path to git repository root, or null if not in a git repo
|
|
57
|
+
*/ function getGitRepositoryRoot(startPath) {
|
|
58
|
+
let currentPath = path.resolve(startPath);
|
|
59
|
+
// If startPath is a file, start from its directory
|
|
60
|
+
try {
|
|
61
|
+
const stats = statSync(currentPath);
|
|
62
|
+
if (stats.isFile()) {
|
|
63
|
+
currentPath = path.dirname(currentPath);
|
|
64
|
+
}
|
|
65
|
+
} catch {
|
|
66
|
+
// If stat fails, assume it's a directory and continue
|
|
67
|
+
}
|
|
68
|
+
// Walk up until we find .git or reach root
|
|
69
|
+
const root = path.parse(currentPath).root;
|
|
70
|
+
while(currentPath !== root){
|
|
71
|
+
const gitPath = path.join(currentPath, '.git');
|
|
72
|
+
try {
|
|
73
|
+
const stats = statSync(gitPath);
|
|
74
|
+
if (stats.isDirectory() || stats.isFile()) {
|
|
75
|
+
// Found .git (can be directory or file for submodules)
|
|
76
|
+
return currentPath;
|
|
77
|
+
}
|
|
78
|
+
} catch {
|
|
79
|
+
// .git doesn't exist at this level, continue up
|
|
80
|
+
}
|
|
81
|
+
// Move up one directory
|
|
82
|
+
const parentPath = path.dirname(currentPath);
|
|
83
|
+
if (parentPath === currentPath) {
|
|
84
|
+
break;
|
|
85
|
+
}
|
|
86
|
+
currentPath = parentPath;
|
|
87
|
+
}
|
|
88
|
+
return null;
|
|
89
|
+
}
|
|
90
|
+
// Global singleton instance
|
|
91
|
+
let globalGitMutexManager = null;
|
|
92
|
+
/**
|
|
93
|
+
* Get the global git mutex manager instance
|
|
94
|
+
* Creates one if it doesn't exist
|
|
95
|
+
*/ function getGitMutexManager() {
|
|
96
|
+
if (!globalGitMutexManager) {
|
|
97
|
+
globalGitMutexManager = new RepositoryMutexManager();
|
|
98
|
+
}
|
|
99
|
+
return globalGitMutexManager;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Helper function to wrap git operations with automatic locking
|
|
103
|
+
* Uses the global git mutex manager
|
|
104
|
+
*
|
|
105
|
+
* @example
|
|
106
|
+
* await runGitWithLock(packagePath, async () => {
|
|
107
|
+
* await run('git add package.json');
|
|
108
|
+
* await run('git commit -m "Update version"');
|
|
109
|
+
* }, 'version bump commit');
|
|
110
|
+
*/ async function runGitWithLock(packagePath, operation, operationName) {
|
|
111
|
+
const manager = getGitMutexManager();
|
|
112
|
+
return await manager.withGitLock(packagePath, operation, operationName);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
export { RepositoryMutexManager, getGitMutexManager, getGitRepositoryRoot, runGitWithLock };
|
|
116
|
+
//# sourceMappingURL=gitMutex.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gitMutex.js","sources":["../../src/util/gitMutex.ts"],"sourcesContent":["import * as path from 'path';\n// eslint-disable-next-line no-restricted-imports\nimport { statSync } from 'fs';\nimport { RepositoryFileLockManager } from './fileLock';\nimport { getLogger } from '../logging';\n\n/**\n * Manages per-repository locks for git operations (cross-process safe)\n * Prevents concurrent git operations in the same repository (which cause .git/index.lock conflicts)\n * while still allowing parallel operations across different repositories\n *\n * Uses file-based locks to coordinate across multiple processes (e.g., parallel tree execution)\n */\nexport class RepositoryMutexManager {\n private lockManager: RepositoryFileLockManager;\n private logger = getLogger();\n\n constructor() {\n this.lockManager = new RepositoryFileLockManager();\n }\n\n /**\n * Execute a git operation with repository-level locking\n * @param packagePath Path to the package (will find its git repo root)\n * @param operation The async operation to execute under lock\n * @param operationName Optional name for logging\n * @returns Result of the operation\n */\n async withGitLock<T>(\n packagePath: string,\n operation: () => Promise<T>,\n operationName?: string\n ): Promise<T> {\n const repoPath = getGitRepositoryRoot(packagePath);\n\n if (!repoPath) {\n // Not in a git repository, execute without lock\n this.logger.debug(`No git repository found for ${packagePath}, executing without lock`);\n return await operation();\n }\n\n return await this.lockManager.withGitLock(repoPath, operation, operationName);\n }\n\n /**\n * Destroy all locks and clean up resources\n */\n destroy(): void {\n this.lockManager.destroy();\n }\n}\n\n/**\n * Find the git repository root for a given path\n * Walks up the directory tree until it finds a .git directory\n * @param startPath Starting path (can be a file or directory)\n * @returns Absolute path to git repository root, or null if not in a git repo\n */\nexport function getGitRepositoryRoot(startPath: string): string | null {\n let currentPath = path.resolve(startPath);\n\n // If startPath is a file, start from its directory\n try {\n const stats = statSync(currentPath);\n if (stats.isFile()) {\n currentPath = path.dirname(currentPath);\n }\n } catch {\n // If stat fails, assume it's a directory and continue\n }\n\n // Walk up until we find .git or reach root\n const root = path.parse(currentPath).root;\n\n while (currentPath !== root) {\n const gitPath = path.join(currentPath, '.git');\n\n try {\n const stats = statSync(gitPath);\n if (stats.isDirectory() || stats.isFile()) {\n // Found .git (can be directory or file for submodules)\n return currentPath;\n }\n } catch {\n // .git doesn't exist at this level, continue up\n }\n\n // Move up one directory\n const parentPath = path.dirname(currentPath);\n if (parentPath === currentPath) {\n // Reached root without finding .git\n break;\n }\n currentPath = parentPath;\n }\n\n return null;\n}\n\n/**\n * Check if a path is within a git repository\n * @param checkPath Path to check\n * @returns true if path is in a git repository\n */\nexport function isInGitRepository(checkPath: string): boolean {\n return getGitRepositoryRoot(checkPath) !== null;\n}\n\n/**\n * Check if two paths are in the same git repository\n * @param path1 First path\n * @param path2 Second path\n * @returns true if both paths are in the same git repository\n */\nexport function areInSameRepository(path1: string, path2: string): boolean {\n const repo1 = getGitRepositoryRoot(path1);\n const repo2 = getGitRepositoryRoot(path2);\n\n if (!repo1 || !repo2) {\n return false;\n }\n\n return repo1 === repo2;\n}\n\n// Global singleton instance\nlet globalGitMutexManager: RepositoryMutexManager | null = null;\n\n/**\n * Get the global git mutex manager instance\n * Creates one if it doesn't exist\n */\nexport function getGitMutexManager(): RepositoryMutexManager {\n if (!globalGitMutexManager) {\n globalGitMutexManager = new RepositoryMutexManager();\n }\n return globalGitMutexManager;\n}\n\n/**\n * Destroy the global git mutex manager\n * Should be called when shutting down or during cleanup\n */\nexport function destroyGitMutexManager(): void {\n if (globalGitMutexManager) {\n globalGitMutexManager.destroy();\n globalGitMutexManager = null;\n }\n}\n\n/**\n * Helper function to wrap git operations with automatic locking\n * Uses the global git mutex manager\n *\n * @example\n * await runGitWithLock(packagePath, async () => {\n * await run('git add package.json');\n * await run('git commit -m \"Update version\"');\n * }, 'version bump commit');\n */\nexport async function runGitWithLock<T>(\n packagePath: string,\n operation: () => Promise<T>,\n operationName?: string\n): Promise<T> {\n const manager = getGitMutexManager();\n return await manager.withGitLock(packagePath, operation, operationName);\n}\n"],"names":["RepositoryMutexManager","withGitLock","packagePath","operation","operationName","repoPath","getGitRepositoryRoot","logger","debug","lockManager","destroy","getLogger","RepositoryFileLockManager","startPath","currentPath","path","resolve","stats","statSync","isFile","dirname","root","parse","gitPath","join","isDirectory","parentPath","globalGitMutexManager","getGitMutexManager","runGitWithLock","manager"],"mappings":";;;;;;;;;;;;;;;;;;AAMA;;;;;;AAMC,IACM,MAAMA,sBAAAA,CAAAA;AAQT;;;;;;AAMC,QACD,MAAMC,WAAAA,CACFC,WAAmB,EACnBC,SAA2B,EAC3BC,aAAsB,EACZ;AACV,QAAA,MAAMC,WAAWC,oBAAAA,CAAqBJ,WAAAA,CAAAA;AAEtC,QAAA,IAAI,CAACG,QAAAA,EAAU;;YAEX,IAAI,CAACE,MAAM,CAACC,KAAK,CAAC,CAAC,4BAA4B,EAAEN,WAAAA,CAAY,wBAAwB,CAAC,CAAA;AACtF,YAAA,OAAO,MAAMC,SAAAA,EAAAA;AACjB,QAAA;QAEA,OAAO,MAAM,IAAI,CAACM,WAAW,CAACR,WAAW,CAACI,UAAUF,SAAAA,EAAWC,aAAAA,CAAAA;AACnE,IAAA;AAEA;;AAEC,QACDM,OAAAA,GAAgB;QACZ,IAAI,CAACD,WAAW,CAACC,OAAO,EAAA;AAC5B,IAAA;IAhCA,WAAA,EAAc;AAHd,QAAA,gBAAA,CAAA,IAAA,EAAQD,eAAR,MAAA,CAAA;AACA,QAAA,gBAAA,CAAA,IAAA,EAAQF,QAAAA,EAASI,SAAAA,EAAAA,CAAAA;QAGb,IAAI,CAACF,WAAW,GAAG,IAAIG,yBAAAA,EAAAA;AAC3B,IAAA;AA+BJ;AAEA;;;;;IAMO,SAASN,oBAAAA,CAAqBO,SAAiB,EAAA;IAClD,IAAIC,WAAAA,GAAcC,IAAAA,CAAKC,OAAO,CAACH,SAAAA,CAAAA;;IAG/B,IAAI;AACA,QAAA,MAAMI,QAAQC,QAAAA,CAASJ,WAAAA,CAAAA;QACvB,IAAIG,KAAAA,CAAME,MAAM,EAAA,EAAI;YAChBL,WAAAA,GAAcC,IAAAA,CAAKK,OAAO,CAACN,WAAAA,CAAAA;AAC/B,QAAA;AACJ,IAAA,CAAA,CAAE,OAAM;;AAER,IAAA;;AAGA,IAAA,MAAMO,IAAAA,GAAON,IAAAA,CAAKO,KAAK,CAACR,aAAaO,IAAI;AAEzC,IAAA,MAAOP,gBAAgBO,IAAAA,CAAM;AACzB,QAAA,MAAME,OAAAA,GAAUR,IAAAA,CAAKS,IAAI,CAACV,WAAAA,EAAa,MAAA,CAAA;QAEvC,IAAI;AACA,YAAA,MAAMG,QAAQC,QAAAA,CAASK,OAAAA,CAAAA;AACvB,YAAA,IAAIN,KAAAA,CAAMQ,WAAW,EAAA,IAAMR,KAAAA,CAAME,MAAM,EAAA,EAAI;;gBAEvC,OAAOL,WAAAA;AACX,YAAA;AACJ,QAAA,CAAA,CAAE,OAAM;;AAER,QAAA;;QAGA,MAAMY,UAAAA,GAAaX,IAAAA,CAAKK,OAAO,CAACN,WAAAA,CAAAA;AAChC,QAAA,IAAIY,eAAeZ,WAAAA,EAAa;AAE5B,YAAA;AACJ,QAAA;QACAA,WAAAA,GAAcY,UAAAA;AAClB,IAAA;IAEA,OAAO,IAAA;AACX;AA4BA;AACA,IAAIC,qBAAAA,GAAuD,IAAA;AAE3D;;;AAGC,IACM,SAASC,kBAAAA,GAAAA;AACZ,IAAA,IAAI,CAACD,qBAAAA,EAAuB;AACxBA,QAAAA,qBAAAA,GAAwB,IAAI3B,sBAAAA,EAAAA;AAChC,IAAA;IACA,OAAO2B,qBAAAA;AACX;AAaA;;;;;;;;;AASC,IACM,eAAeE,cAAAA,CAClB3B,WAAmB,EACnBC,SAA2B,EAC3BC,aAAsB,EAAA;AAEtB,IAAA,MAAM0B,OAAAA,GAAUF,kBAAAA,EAAAA;AAChB,IAAA,OAAO,MAAME,OAAAA,CAAQ7B,WAAW,CAACC,aAAaC,SAAAA,EAAWC,aAAAA,CAAAA;AAC7D;;;;"}
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Simple mutex implementation for serializing async operations
|
|
3
|
+
* Prevents race conditions when multiple async operations need exclusive access
|
|
4
|
+
*/ function _define_property(obj, key, value) {
|
|
5
|
+
if (key in obj) {
|
|
6
|
+
Object.defineProperty(obj, key, {
|
|
7
|
+
value: value,
|
|
8
|
+
enumerable: true,
|
|
9
|
+
configurable: true,
|
|
10
|
+
writable: true
|
|
11
|
+
});
|
|
12
|
+
} else {
|
|
13
|
+
obj[key] = value;
|
|
14
|
+
}
|
|
15
|
+
return obj;
|
|
16
|
+
}
|
|
17
|
+
class SimpleMutex {
|
|
18
|
+
/**
|
|
19
|
+
* Acquire the mutex lock
|
|
20
|
+
* If already locked, waits in queue until released
|
|
21
|
+
*/ async lock() {
|
|
22
|
+
return new Promise((resolve, reject)=>{
|
|
23
|
+
if (this.destroyed) {
|
|
24
|
+
reject(new Error('Mutex has been destroyed'));
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
if (!this.locked) {
|
|
28
|
+
this.locked = true;
|
|
29
|
+
resolve();
|
|
30
|
+
} else {
|
|
31
|
+
this.queue.push(resolve);
|
|
32
|
+
}
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Release the mutex lock
|
|
37
|
+
* Allows next waiting operation in queue to proceed
|
|
38
|
+
*/ unlock() {
|
|
39
|
+
if (this.destroyed) {
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
this.locked = false;
|
|
43
|
+
const next = this.queue.shift();
|
|
44
|
+
if (next) {
|
|
45
|
+
this.locked = true;
|
|
46
|
+
try {
|
|
47
|
+
next();
|
|
48
|
+
} catch {
|
|
49
|
+
// If resolver throws, unlock and continue with next in queue
|
|
50
|
+
this.locked = false;
|
|
51
|
+
const nextInQueue = this.queue.shift();
|
|
52
|
+
if (nextInQueue) {
|
|
53
|
+
this.locked = true;
|
|
54
|
+
nextInQueue();
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Destroy the mutex and reject all waiting operations
|
|
61
|
+
* Prevents memory leaks when mutex is no longer needed
|
|
62
|
+
*/ destroy() {
|
|
63
|
+
this.destroyed = true;
|
|
64
|
+
this.locked = false;
|
|
65
|
+
// Reject all queued promises to prevent memory leaks
|
|
66
|
+
while(this.queue.length > 0){
|
|
67
|
+
const resolve = this.queue.shift();
|
|
68
|
+
if (resolve) {
|
|
69
|
+
try {
|
|
70
|
+
// Resolve with error state
|
|
71
|
+
resolve();
|
|
72
|
+
} catch {
|
|
73
|
+
// Ignore errors during cleanup
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Check if mutex is currently locked
|
|
80
|
+
*/ isLocked() {
|
|
81
|
+
return this.locked;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Get number of operations waiting in queue
|
|
85
|
+
*/ getQueueLength() {
|
|
86
|
+
return this.queue.length;
|
|
87
|
+
}
|
|
88
|
+
constructor(){
|
|
89
|
+
_define_property(this, "locked", false);
|
|
90
|
+
_define_property(this, "queue", []);
|
|
91
|
+
_define_property(this, "destroyed", false);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
export { SimpleMutex };
|
|
96
|
+
//# sourceMappingURL=mutex.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mutex.js","sources":["../../src/util/mutex.ts"],"sourcesContent":["/**\n * Simple mutex implementation for serializing async operations\n * Prevents race conditions when multiple async operations need exclusive access\n */\nexport class SimpleMutex {\n private locked = false;\n private queue: Array<() => void> = [];\n private destroyed = false;\n\n /**\n * Acquire the mutex lock\n * If already locked, waits in queue until released\n */\n async lock(): Promise<void> {\n return new Promise<void>((resolve, reject) => {\n if (this.destroyed) {\n reject(new Error('Mutex has been destroyed'));\n return;\n }\n\n if (!this.locked) {\n this.locked = true;\n resolve();\n } else {\n this.queue.push(resolve);\n }\n });\n }\n\n /**\n * Release the mutex lock\n * Allows next waiting operation in queue to proceed\n */\n unlock(): void {\n if (this.destroyed) {\n return;\n }\n\n this.locked = false;\n const next = this.queue.shift();\n if (next) {\n this.locked = true;\n try {\n next();\n } catch {\n // If resolver throws, unlock and continue with next in queue\n this.locked = false;\n const nextInQueue = this.queue.shift();\n if (nextInQueue) {\n this.locked = true;\n nextInQueue();\n }\n }\n }\n }\n\n /**\n * Destroy the mutex and reject all waiting operations\n * Prevents memory leaks when mutex is no longer needed\n */\n destroy(): void {\n this.destroyed = true;\n this.locked = false;\n\n // Reject all queued promises to prevent memory leaks\n while (this.queue.length > 0) {\n const resolve = this.queue.shift();\n if (resolve) {\n try {\n // Resolve with error state\n resolve();\n } catch {\n // Ignore errors during cleanup\n }\n }\n }\n }\n\n /**\n * Check if mutex is currently locked\n */\n isLocked(): boolean {\n return this.locked;\n }\n\n /**\n * Get number of operations waiting in queue\n */\n getQueueLength(): number {\n return this.queue.length;\n }\n}\n"],"names":["SimpleMutex","lock","Promise","resolve","reject","destroyed","Error","locked","queue","push","unlock","next","shift","nextInQueue","destroy","length","isLocked","getQueueLength"],"mappings":"AAAA;;;AAGC,IAAA,SAAA,gBAAA,CAAA,GAAA,EAAA,GAAA,EAAA,KAAA,EAAA;;;;;;;;;;;;;AACM,MAAMA,WAAAA,CAAAA;AAKT;;;AAGC,QACD,MAAMC,IAAAA,GAAsB;QACxB,OAAO,IAAIC,OAAAA,CAAc,CAACC,OAAAA,EAASC,MAAAA,GAAAA;YAC/B,IAAI,IAAI,CAACC,SAAS,EAAE;AAChBD,gBAAAA,MAAAA,CAAO,IAAIE,KAAAA,CAAM,0BAAA,CAAA,CAAA;AACjB,gBAAA;AACJ,YAAA;AAEA,YAAA,IAAI,CAAC,IAAI,CAACC,MAAM,EAAE;gBACd,IAAI,CAACA,MAAM,GAAG,IAAA;AACdJ,gBAAAA,OAAAA,EAAAA;YACJ,CAAA,MAAO;AACH,gBAAA,IAAI,CAACK,KAAK,CAACC,IAAI,CAACN,OAAAA,CAAAA;AACpB,YAAA;AACJ,QAAA,CAAA,CAAA;AACJ,IAAA;AAEA;;;AAGC,QACDO,MAAAA,GAAe;QACX,IAAI,IAAI,CAACL,SAAS,EAAE;AAChB,YAAA;AACJ,QAAA;QAEA,IAAI,CAACE,MAAM,GAAG,KAAA;AACd,QAAA,MAAMI,IAAAA,GAAO,IAAI,CAACH,KAAK,CAACI,KAAK,EAAA;AAC7B,QAAA,IAAID,IAAAA,EAAM;YACN,IAAI,CAACJ,MAAM,GAAG,IAAA;YACd,IAAI;AACAI,gBAAAA,IAAAA,EAAAA;AACJ,YAAA,CAAA,CAAE,OAAM;;gBAEJ,IAAI,CAACJ,MAAM,GAAG,KAAA;AACd,gBAAA,MAAMM,WAAAA,GAAc,IAAI,CAACL,KAAK,CAACI,KAAK,EAAA;AACpC,gBAAA,IAAIC,WAAAA,EAAa;oBACb,IAAI,CAACN,MAAM,GAAG,IAAA;AACdM,oBAAAA,WAAAA,EAAAA;AACJ,gBAAA;AACJ,YAAA;AACJ,QAAA;AACJ,IAAA;AAEA;;;AAGC,QACDC,OAAAA,GAAgB;QACZ,IAAI,CAACT,SAAS,GAAG,IAAA;QACjB,IAAI,CAACE,MAAM,GAAG,KAAA;;AAGd,QAAA,MAAO,IAAI,CAACC,KAAK,CAACO,MAAM,GAAG,CAAA,CAAG;AAC1B,YAAA,MAAMZ,OAAAA,GAAU,IAAI,CAACK,KAAK,CAACI,KAAK,EAAA;AAChC,YAAA,IAAIT,OAAAA,EAAS;gBACT,IAAI;;AAEAA,oBAAAA,OAAAA,EAAAA;AACJ,gBAAA,CAAA,CAAE,OAAM;;AAER,gBAAA;AACJ,YAAA;AACJ,QAAA;AACJ,IAAA;AAEA;;AAEC,QACDa,QAAAA,GAAoB;QAChB,OAAO,IAAI,CAACT,MAAM;AACtB,IAAA;AAEA;;AAEC,QACDU,cAAAA,GAAyB;AACrB,QAAA,OAAO,IAAI,CAACT,KAAK,CAACO,MAAM;AAC5B,IAAA;;AArFA,QAAA,gBAAA,CAAA,IAAA,EAAQR,QAAAA,EAAS,KAAA,CAAA;AACjB,QAAA,gBAAA,CAAA,IAAA,EAAQC,SAA2B,EAAE,CAAA;AACrC,QAAA,gBAAA,CAAA,IAAA,EAAQH,WAAAA,EAAY,KAAA,CAAA;;AAoFxB;;;;"}
|
package/dist/util/performance.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import
|
|
1
|
+
import path__default from 'path';
|
|
2
2
|
import { getLogger } from '../logging.js';
|
|
3
3
|
import { safeJsonParse, validatePackageJson } from '@eldrforge/git-tools';
|
|
4
4
|
|
|
@@ -55,7 +55,7 @@ const batchReadPackageJsonFiles = async (packageJsonPaths, storage, rootDir)=>{
|
|
|
55
55
|
const packageJsonContent = await storage.readFile(packageJsonPath, 'utf-8');
|
|
56
56
|
const parsed = safeJsonParse(packageJsonContent, packageJsonPath);
|
|
57
57
|
const packageJson = validatePackageJson(parsed, packageJsonPath, false);
|
|
58
|
-
const relativePath =
|
|
58
|
+
const relativePath = path__default.relative(rootDir, path__default.dirname(packageJsonPath));
|
|
59
59
|
return {
|
|
60
60
|
path: packageJsonPath,
|
|
61
61
|
packageJson,
|
|
@@ -88,7 +88,7 @@ const findAllPackageJsonFiles = async (rootDir, storage)=>{
|
|
|
88
88
|
const foundPaths = [];
|
|
89
89
|
// Check for package.json in current directory
|
|
90
90
|
if (items.includes('package.json')) {
|
|
91
|
-
const packageJsonPath =
|
|
91
|
+
const packageJsonPath = path__default.join(currentDir, 'package.json');
|
|
92
92
|
foundPaths.push(packageJsonPath);
|
|
93
93
|
}
|
|
94
94
|
// Process subdirectories in parallel
|
|
@@ -97,7 +97,7 @@ const findAllPackageJsonFiles = async (rootDir, storage)=>{
|
|
|
97
97
|
if (EXCLUDED_DIRECTORIES.includes(item)) {
|
|
98
98
|
continue;
|
|
99
99
|
}
|
|
100
|
-
const itemPath =
|
|
100
|
+
const itemPath = path__default.join(currentDir, item);
|
|
101
101
|
subdirPromises.push((async ()=>{
|
|
102
102
|
try {
|
|
103
103
|
if (await storage.isDirectory(itemPath)) {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"performance.js","sources":["../../src/util/performance.ts"],"sourcesContent":["/* eslint-disable @typescript-eslint/no-unused-vars */\nimport path from 'path';\nimport { getLogger } from '../logging';\nimport { safeJsonParse, validatePackageJson } from '@eldrforge/git-tools';\n\n// Performance timing helper\nexport class PerformanceTimer {\n private startTime: number;\n private logger: any;\n\n constructor(logger: any) {\n this.logger = logger;\n this.startTime = Date.now();\n }\n\n static start(logger: any, operation: string): PerformanceTimer {\n logger.verbose(`⏱️ Starting: ${operation}`);\n return new PerformanceTimer(logger);\n }\n\n end(operation: string): number {\n const duration = Date.now() - this.startTime;\n this.logger.verbose(`⏱️ Completed: ${operation} (${duration}ms)`);\n return duration;\n }\n}\n\nexport interface PackageJson {\n name?: string;\n dependencies?: Record<string, string>;\n devDependencies?: Record<string, string>;\n peerDependencies?: Record<string, string>;\n}\n\nexport interface PackageJsonLocation {\n path: string;\n packageJson: PackageJson;\n relativePath: string;\n}\n\nconst EXCLUDED_DIRECTORIES = [\n 'node_modules',\n 'dist',\n 'build',\n 'coverage',\n '.git',\n '.next',\n '.nuxt',\n 'out',\n 'public',\n 'static',\n 'assets'\n];\n\n// Batch read multiple package.json files in parallel\nexport const batchReadPackageJsonFiles = async (\n packageJsonPaths: string[],\n storage: any,\n rootDir: string\n): Promise<PackageJsonLocation[]> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, `Batch reading ${packageJsonPaths.length} package.json files`);\n\n const readPromises = packageJsonPaths.map(async (packageJsonPath): Promise<PackageJsonLocation | null> => {\n try {\n const packageJsonContent = await storage.readFile(packageJsonPath, 'utf-8');\n const parsed = safeJsonParse(packageJsonContent, packageJsonPath);\n const packageJson = validatePackageJson(parsed, packageJsonPath, false);\n const relativePath = path.relative(rootDir, path.dirname(packageJsonPath));\n\n return {\n path: packageJsonPath,\n packageJson,\n relativePath: relativePath || '.'\n };\n } catch (error: any) {\n logger.debug(`Skipped invalid package.json at ${packageJsonPath}: ${error.message}`);\n return null;\n }\n });\n\n const results = await Promise.all(readPromises);\n const validResults = results.filter((result): result is PackageJsonLocation => result !== null);\n\n timer.end(`Successfully read ${validResults.length}/${packageJsonPaths.length} package.json files`);\n return validResults;\n};\n\n// Optimized recursive package.json finder with parallel processing\nexport const findAllPackageJsonFiles = async (rootDir: string, storage: any): Promise<PackageJsonLocation[]> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, 'Optimized scanning for package.json files');\n\n // Phase 1: Find all package.json file paths in parallel\n const packageJsonPaths: string[] = [];\n\n const scanForPaths = async (currentDir: string, depth: number = 0): Promise<string[]> => {\n // Prevent infinite recursion and overly deep scanning\n if (depth > 5) {\n return [];\n }\n\n try {\n if (!await storage.exists(currentDir) || !await storage.isDirectory(currentDir)) {\n return [];\n }\n\n const items = await storage.listFiles(currentDir);\n const foundPaths: string[] = [];\n\n // Check for package.json in current directory\n if (items.includes('package.json')) {\n const packageJsonPath = path.join(currentDir, 'package.json');\n foundPaths.push(packageJsonPath);\n }\n\n // Process subdirectories in parallel\n const subdirPromises: Promise<string[]>[] = [];\n for (const item of items) {\n if (EXCLUDED_DIRECTORIES.includes(item)) {\n continue;\n }\n\n const itemPath = path.join(currentDir, item);\n subdirPromises.push(\n (async () => {\n try {\n if (await storage.isDirectory(itemPath)) {\n return await scanForPaths(itemPath, depth + 1);\n }\n } catch (error: any) {\n logger.debug(`Skipped directory ${itemPath}: ${error.message}`);\n }\n return [];\n })()\n );\n }\n\n if (subdirPromises.length > 0) {\n const subdirResults = await Promise.all(subdirPromises);\n for (const subdirPaths of subdirResults) {\n foundPaths.push(...subdirPaths);\n }\n }\n\n return foundPaths;\n } catch (error: any) {\n logger.debug(`Failed to scan directory ${currentDir}: ${error.message}`);\n return [];\n }\n };\n\n const pathsTimer = PerformanceTimer.start(logger, 'Finding all package.json paths');\n const allPaths = await scanForPaths(rootDir);\n pathsTimer.end(`Found ${allPaths.length} package.json file paths`);\n\n // Phase 2: Batch read all package.json files in parallel\n const packageJsonFiles = await batchReadPackageJsonFiles(allPaths, storage, rootDir);\n\n timer.end(`Found ${packageJsonFiles.length} valid package.json files`);\n return packageJsonFiles;\n};\n\n// Optimized package scanning with parallel processing\nexport const scanDirectoryForPackages = async (rootDir: string, storage: any): Promise<Map<string, string>> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, `Optimized package scanning: ${rootDir}`);\n const packageMap = new Map<string, string>(); // packageName -> relativePath\n\n const absoluteRootDir = path.resolve(process.cwd(), rootDir);\n logger.verbose(`Scanning directory for packages: ${absoluteRootDir}`);\n\n try {\n // Quick existence and directory check\n const existsTimer = PerformanceTimer.start(logger, `Checking directory: ${absoluteRootDir}`);\n if (!await storage.exists(absoluteRootDir) || !await storage.isDirectory(absoluteRootDir)) {\n existsTimer.end(`Directory not found or not a directory: ${absoluteRootDir}`);\n timer.end(`Directory invalid: ${rootDir}`);\n return packageMap;\n }\n existsTimer.end(`Directory verified: ${absoluteRootDir}`);\n\n // Get all items and process in parallel\n const listTimer = PerformanceTimer.start(logger, `Listing contents: ${absoluteRootDir}`);\n const items = await storage.listFiles(absoluteRootDir);\n listTimer.end(`Listed ${items.length} items`);\n\n // Create batched promises for better performance\n const BATCH_SIZE = 10; // Process directories in batches to avoid overwhelming filesystem\n const batches = [];\n\n for (let i = 0; i < items.length; i += BATCH_SIZE) {\n const batch = items.slice(i, i + BATCH_SIZE);\n batches.push(batch);\n }\n\n const processTimer = PerformanceTimer.start(logger, `Processing ${batches.length} batches of directories`);\n\n for (const batch of batches) {\n const batchPromises = batch.map(async (item: string) => {\n const itemPath = path.join(absoluteRootDir, item);\n try {\n if (await storage.isDirectory(itemPath)) {\n const packageJsonPath = path.join(itemPath, 'package.json');\n\n if (await storage.exists(packageJsonPath)) {\n const packageJsonContent = await storage.readFile(packageJsonPath, 'utf-8');\n const parsed = safeJsonParse(packageJsonContent, packageJsonPath);\n const packageJson = validatePackageJson(parsed, packageJsonPath);\n\n if (packageJson.name) {\n const relativePath = path.relative(process.cwd(), itemPath);\n return { name: packageJson.name, path: relativePath };\n }\n }\n }\n } catch (error: any) {\n logger.debug(`Skipped ${itemPath}: ${error.message || error}`);\n }\n return null;\n });\n\n const batchResults = await Promise.all(batchPromises);\n\n for (const result of batchResults) {\n if (result) {\n packageMap.set(result.name, result.path);\n logger.debug(`Found package: ${result.name} at ${result.path}`);\n }\n }\n }\n\n processTimer.end(`Processed ${items.length} directories in ${batches.length} batches`);\n logger.verbose(`Found ${packageMap.size} packages in ${items.length} subdirectories`);\n } catch (error) {\n logger.warn(`Failed to read directory ${absoluteRootDir}: ${error}`);\n }\n\n timer.end(`Found ${packageMap.size} packages in: ${rootDir}`);\n return packageMap;\n};\n\n// Parallel scope processing for better performance\nexport const findPackagesByScope = async (\n dependencies: Record<string, string>,\n scopeRoots: Record<string, string>,\n storage: any\n): Promise<Map<string, string>> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, 'Finding packages by scope (optimized)');\n const workspacePackages = new Map<string, string>();\n\n logger.silly(`Checking dependencies against scope roots: ${JSON.stringify(scopeRoots)}`);\n\n // Process all scopes in parallel for maximum performance\n const scopeTimer = PerformanceTimer.start(logger, 'Parallel scope scanning');\n const scopePromises = Object.entries(scopeRoots).map(async ([scope, rootDir]) => {\n logger.verbose(`Scanning scope ${scope} at root directory: ${rootDir}`);\n const scopePackages = await scanDirectoryForPackages(rootDir, storage);\n\n // Filter packages that match the scope\n const matchingPackages: Array<[string, string]> = [];\n for (const [packageName, packagePath] of scopePackages) {\n if (packageName.startsWith(scope)) {\n matchingPackages.push([packageName, packagePath]);\n logger.debug(`Registered package: ${packageName} -> ${packagePath}`);\n }\n }\n return { scope, packages: matchingPackages };\n });\n\n const allScopeResults = await Promise.all(scopePromises);\n\n // Aggregate all packages from all scopes\n const allPackages = new Map<string, string>();\n for (const { scope, packages } of allScopeResults) {\n for (const [packageName, packagePath] of packages) {\n allPackages.set(packageName, packagePath);\n }\n }\n\n scopeTimer.end(`Scanned ${Object.keys(scopeRoots).length} scope roots, found ${allPackages.size} packages`);\n\n // Match dependencies to available packages\n const matchTimer = PerformanceTimer.start(logger, 'Matching dependencies to packages');\n for (const [depName, depVersion] of Object.entries(dependencies)) {\n logger.debug(`Processing dependency: ${depName}@${depVersion}`);\n\n if (allPackages.has(depName)) {\n const packagePath = allPackages.get(depName)!;\n workspacePackages.set(depName, packagePath);\n logger.verbose(`Found sibling package: ${depName} at ${packagePath}`);\n }\n }\n matchTimer.end(`Matched ${workspacePackages.size} dependencies to workspace packages`);\n\n timer.end(`Found ${workspacePackages.size} packages to link`);\n return workspacePackages;\n};\n\n// Utility to collect all dependencies from package.json files efficiently\nexport const collectAllDependencies = (packageJsonFiles: PackageJsonLocation[]): Record<string, string> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, 'Collecting all dependencies');\n\n const allDependencies: Record<string, string> = {};\n for (const { packageJson } of packageJsonFiles) {\n Object.assign(allDependencies, packageJson.dependencies);\n Object.assign(allDependencies, packageJson.devDependencies);\n Object.assign(allDependencies, packageJson.peerDependencies);\n }\n\n timer.end(`Collected ${Object.keys(allDependencies).length} unique dependencies`);\n return allDependencies;\n};\n\n// Utility to check for file: dependencies\nexport const checkForFileDependencies = (packageJsonFiles: PackageJsonLocation[]): void => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, 'Checking for file: dependencies');\n const filesWithFileDepedencies: Array<{path: string, dependencies: string[]}> = [];\n\n for (const { path: packagePath, packageJson, relativePath } of packageJsonFiles) {\n const fileDeps: string[] = [];\n\n // Check all dependency types for file: paths\n const allDeps = {\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n ...packageJson.peerDependencies\n };\n\n for (const [name, version] of Object.entries(allDeps)) {\n if (version.startsWith('file:')) {\n fileDeps.push(`${name}: ${version}`);\n }\n }\n\n if (fileDeps.length > 0) {\n filesWithFileDepedencies.push({\n path: relativePath,\n dependencies: fileDeps\n });\n }\n }\n\n if (filesWithFileDepedencies.length > 0) {\n logger.warn('⚠️ WARNING: Found file: dependencies that should not be committed:');\n for (const file of filesWithFileDepedencies) {\n logger.warn(` 📄 ${file.path}:`);\n for (const dep of file.dependencies) {\n logger.warn(` - ${dep}`);\n }\n }\n logger.warn('');\n logger.warn('💡 Remember to run \"kodrdriv unlink\" before committing to restore registry versions!');\n logger.warn(' Or add a pre-commit hook to prevent accidental commits of linked dependencies.');\n }\n\n timer.end(`Checked ${packageJsonFiles.length} files, found ${filesWithFileDepedencies.length} with file: dependencies`);\n};\n"],"names":["PerformanceTimer","start","logger","operation","verbose","end","duration","Date","now","startTime","EXCLUDED_DIRECTORIES","batchReadPackageJsonFiles","packageJsonPaths","storage","rootDir","getLogger","timer","length","readPromises","map","packageJsonPath","packageJsonContent","readFile","parsed","safeJsonParse","packageJson","validatePackageJson","relativePath","path","relative","dirname","error","debug","message","results","Promise","all","validResults","filter","result","findAllPackageJsonFiles","scanForPaths","currentDir","depth","exists","isDirectory","items","listFiles","foundPaths","includes","join","push","subdirPromises","item","itemPath","subdirResults","subdirPaths","pathsTimer","allPaths","packageJsonFiles"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"performance.js","sources":["../../src/util/performance.ts"],"sourcesContent":["/* eslint-disable @typescript-eslint/no-unused-vars */\nimport path from 'path';\nimport { getLogger } from '../logging';\nimport { safeJsonParse, validatePackageJson } from '@eldrforge/git-tools';\n\n// Performance timing helper\nexport class PerformanceTimer {\n private startTime: number;\n private logger: any;\n\n constructor(logger: any) {\n this.logger = logger;\n this.startTime = Date.now();\n }\n\n static start(logger: any, operation: string): PerformanceTimer {\n logger.verbose(`⏱️ Starting: ${operation}`);\n return new PerformanceTimer(logger);\n }\n\n end(operation: string): number {\n const duration = Date.now() - this.startTime;\n this.logger.verbose(`⏱️ Completed: ${operation} (${duration}ms)`);\n return duration;\n }\n}\n\nexport interface PackageJson {\n name?: string;\n dependencies?: Record<string, string>;\n devDependencies?: Record<string, string>;\n peerDependencies?: Record<string, string>;\n}\n\nexport interface PackageJsonLocation {\n path: string;\n packageJson: PackageJson;\n relativePath: string;\n}\n\nconst EXCLUDED_DIRECTORIES = [\n 'node_modules',\n 'dist',\n 'build',\n 'coverage',\n '.git',\n '.next',\n '.nuxt',\n 'out',\n 'public',\n 'static',\n 'assets'\n];\n\n// Batch read multiple package.json files in parallel\nexport const batchReadPackageJsonFiles = async (\n packageJsonPaths: string[],\n storage: any,\n rootDir: string\n): Promise<PackageJsonLocation[]> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, `Batch reading ${packageJsonPaths.length} package.json files`);\n\n const readPromises = packageJsonPaths.map(async (packageJsonPath): Promise<PackageJsonLocation | null> => {\n try {\n const packageJsonContent = await storage.readFile(packageJsonPath, 'utf-8');\n const parsed = safeJsonParse(packageJsonContent, packageJsonPath);\n const packageJson = validatePackageJson(parsed, packageJsonPath, false);\n const relativePath = path.relative(rootDir, path.dirname(packageJsonPath));\n\n return {\n path: packageJsonPath,\n packageJson,\n relativePath: relativePath || '.'\n };\n } catch (error: any) {\n logger.debug(`Skipped invalid package.json at ${packageJsonPath}: ${error.message}`);\n return null;\n }\n });\n\n const results = await Promise.all(readPromises);\n const validResults = results.filter((result): result is PackageJsonLocation => result !== null);\n\n timer.end(`Successfully read ${validResults.length}/${packageJsonPaths.length} package.json files`);\n return validResults;\n};\n\n// Optimized recursive package.json finder with parallel processing\nexport const findAllPackageJsonFiles = async (rootDir: string, storage: any): Promise<PackageJsonLocation[]> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, 'Optimized scanning for package.json files');\n\n // Phase 1: Find all package.json file paths in parallel\n const packageJsonPaths: string[] = [];\n\n const scanForPaths = async (currentDir: string, depth: number = 0): Promise<string[]> => {\n // Prevent infinite recursion and overly deep scanning\n if (depth > 5) {\n return [];\n }\n\n try {\n if (!await storage.exists(currentDir) || !await storage.isDirectory(currentDir)) {\n return [];\n }\n\n const items = await storage.listFiles(currentDir);\n const foundPaths: string[] = [];\n\n // Check for package.json in current directory\n if (items.includes('package.json')) {\n const packageJsonPath = path.join(currentDir, 'package.json');\n foundPaths.push(packageJsonPath);\n }\n\n // Process subdirectories in parallel\n const subdirPromises: Promise<string[]>[] = [];\n for (const item of items) {\n if (EXCLUDED_DIRECTORIES.includes(item)) {\n continue;\n }\n\n const itemPath = path.join(currentDir, item);\n subdirPromises.push(\n (async () => {\n try {\n if (await storage.isDirectory(itemPath)) {\n return await scanForPaths(itemPath, depth + 1);\n }\n } catch (error: any) {\n logger.debug(`Skipped directory ${itemPath}: ${error.message}`);\n }\n return [];\n })()\n );\n }\n\n if (subdirPromises.length > 0) {\n const subdirResults = await Promise.all(subdirPromises);\n for (const subdirPaths of subdirResults) {\n foundPaths.push(...subdirPaths);\n }\n }\n\n return foundPaths;\n } catch (error: any) {\n logger.debug(`Failed to scan directory ${currentDir}: ${error.message}`);\n return [];\n }\n };\n\n const pathsTimer = PerformanceTimer.start(logger, 'Finding all package.json paths');\n const allPaths = await scanForPaths(rootDir);\n pathsTimer.end(`Found ${allPaths.length} package.json file paths`);\n\n // Phase 2: Batch read all package.json files in parallel\n const packageJsonFiles = await batchReadPackageJsonFiles(allPaths, storage, rootDir);\n\n timer.end(`Found ${packageJsonFiles.length} valid package.json files`);\n return packageJsonFiles;\n};\n\n// Optimized package scanning with parallel processing\nexport const scanDirectoryForPackages = async (rootDir: string, storage: any): Promise<Map<string, string>> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, `Optimized package scanning: ${rootDir}`);\n const packageMap = new Map<string, string>(); // packageName -> relativePath\n\n const absoluteRootDir = path.resolve(process.cwd(), rootDir);\n logger.verbose(`Scanning directory for packages: ${absoluteRootDir}`);\n\n try {\n // Quick existence and directory check\n const existsTimer = PerformanceTimer.start(logger, `Checking directory: ${absoluteRootDir}`);\n if (!await storage.exists(absoluteRootDir) || !await storage.isDirectory(absoluteRootDir)) {\n existsTimer.end(`Directory not found or not a directory: ${absoluteRootDir}`);\n timer.end(`Directory invalid: ${rootDir}`);\n return packageMap;\n }\n existsTimer.end(`Directory verified: ${absoluteRootDir}`);\n\n // Get all items and process in parallel\n const listTimer = PerformanceTimer.start(logger, `Listing contents: ${absoluteRootDir}`);\n const items = await storage.listFiles(absoluteRootDir);\n listTimer.end(`Listed ${items.length} items`);\n\n // Create batched promises for better performance\n const BATCH_SIZE = 10; // Process directories in batches to avoid overwhelming filesystem\n const batches = [];\n\n for (let i = 0; i < items.length; i += BATCH_SIZE) {\n const batch = items.slice(i, i + BATCH_SIZE);\n batches.push(batch);\n }\n\n const processTimer = PerformanceTimer.start(logger, `Processing ${batches.length} batches of directories`);\n\n for (const batch of batches) {\n const batchPromises = batch.map(async (item: string) => {\n const itemPath = path.join(absoluteRootDir, item);\n try {\n if (await storage.isDirectory(itemPath)) {\n const packageJsonPath = path.join(itemPath, 'package.json');\n\n if (await storage.exists(packageJsonPath)) {\n const packageJsonContent = await storage.readFile(packageJsonPath, 'utf-8');\n const parsed = safeJsonParse(packageJsonContent, packageJsonPath);\n const packageJson = validatePackageJson(parsed, packageJsonPath);\n\n if (packageJson.name) {\n const relativePath = path.relative(process.cwd(), itemPath);\n return { name: packageJson.name, path: relativePath };\n }\n }\n }\n } catch (error: any) {\n logger.debug(`Skipped ${itemPath}: ${error.message || error}`);\n }\n return null;\n });\n\n const batchResults = await Promise.all(batchPromises);\n\n for (const result of batchResults) {\n if (result) {\n packageMap.set(result.name, result.path);\n logger.debug(`Found package: ${result.name} at ${result.path}`);\n }\n }\n }\n\n processTimer.end(`Processed ${items.length} directories in ${batches.length} batches`);\n logger.verbose(`Found ${packageMap.size} packages in ${items.length} subdirectories`);\n } catch (error) {\n logger.warn(`Failed to read directory ${absoluteRootDir}: ${error}`);\n }\n\n timer.end(`Found ${packageMap.size} packages in: ${rootDir}`);\n return packageMap;\n};\n\n// Parallel scope processing for better performance\nexport const findPackagesByScope = async (\n dependencies: Record<string, string>,\n scopeRoots: Record<string, string>,\n storage: any\n): Promise<Map<string, string>> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, 'Finding packages by scope (optimized)');\n const workspacePackages = new Map<string, string>();\n\n logger.silly(`Checking dependencies against scope roots: ${JSON.stringify(scopeRoots)}`);\n\n // Process all scopes in parallel for maximum performance\n const scopeTimer = PerformanceTimer.start(logger, 'Parallel scope scanning');\n const scopePromises = Object.entries(scopeRoots).map(async ([scope, rootDir]) => {\n logger.verbose(`Scanning scope ${scope} at root directory: ${rootDir}`);\n const scopePackages = await scanDirectoryForPackages(rootDir, storage);\n\n // Filter packages that match the scope\n const matchingPackages: Array<[string, string]> = [];\n for (const [packageName, packagePath] of scopePackages) {\n if (packageName.startsWith(scope)) {\n matchingPackages.push([packageName, packagePath]);\n logger.debug(`Registered package: ${packageName} -> ${packagePath}`);\n }\n }\n return { scope, packages: matchingPackages };\n });\n\n const allScopeResults = await Promise.all(scopePromises);\n\n // Aggregate all packages from all scopes\n const allPackages = new Map<string, string>();\n for (const { scope, packages } of allScopeResults) {\n for (const [packageName, packagePath] of packages) {\n allPackages.set(packageName, packagePath);\n }\n }\n\n scopeTimer.end(`Scanned ${Object.keys(scopeRoots).length} scope roots, found ${allPackages.size} packages`);\n\n // Match dependencies to available packages\n const matchTimer = PerformanceTimer.start(logger, 'Matching dependencies to packages');\n for (const [depName, depVersion] of Object.entries(dependencies)) {\n logger.debug(`Processing dependency: ${depName}@${depVersion}`);\n\n if (allPackages.has(depName)) {\n const packagePath = allPackages.get(depName)!;\n workspacePackages.set(depName, packagePath);\n logger.verbose(`Found sibling package: ${depName} at ${packagePath}`);\n }\n }\n matchTimer.end(`Matched ${workspacePackages.size} dependencies to workspace packages`);\n\n timer.end(`Found ${workspacePackages.size} packages to link`);\n return workspacePackages;\n};\n\n// Utility to collect all dependencies from package.json files efficiently\nexport const collectAllDependencies = (packageJsonFiles: PackageJsonLocation[]): Record<string, string> => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, 'Collecting all dependencies');\n\n const allDependencies: Record<string, string> = {};\n for (const { packageJson } of packageJsonFiles) {\n Object.assign(allDependencies, packageJson.dependencies);\n Object.assign(allDependencies, packageJson.devDependencies);\n Object.assign(allDependencies, packageJson.peerDependencies);\n }\n\n timer.end(`Collected ${Object.keys(allDependencies).length} unique dependencies`);\n return allDependencies;\n};\n\n// Utility to check for file: dependencies\nexport const checkForFileDependencies = (packageJsonFiles: PackageJsonLocation[]): void => {\n const logger = getLogger();\n const timer = PerformanceTimer.start(logger, 'Checking for file: dependencies');\n const filesWithFileDepedencies: Array<{path: string, dependencies: string[]}> = [];\n\n for (const { path: packagePath, packageJson, relativePath } of packageJsonFiles) {\n const fileDeps: string[] = [];\n\n // Check all dependency types for file: paths\n const allDeps = {\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n ...packageJson.peerDependencies\n };\n\n for (const [name, version] of Object.entries(allDeps)) {\n if (version.startsWith('file:')) {\n fileDeps.push(`${name}: ${version}`);\n }\n }\n\n if (fileDeps.length > 0) {\n filesWithFileDepedencies.push({\n path: relativePath,\n dependencies: fileDeps\n });\n }\n }\n\n if (filesWithFileDepedencies.length > 0) {\n logger.warn('⚠️ WARNING: Found file: dependencies that should not be committed:');\n for (const file of filesWithFileDepedencies) {\n logger.warn(` 📄 ${file.path}:`);\n for (const dep of file.dependencies) {\n logger.warn(` - ${dep}`);\n }\n }\n logger.warn('');\n logger.warn('💡 Remember to run \"kodrdriv unlink\" before committing to restore registry versions!');\n logger.warn(' Or add a pre-commit hook to prevent accidental commits of linked dependencies.');\n }\n\n timer.end(`Checked ${packageJsonFiles.length} files, found ${filesWithFileDepedencies.length} with file: dependencies`);\n};\n"],"names":["PerformanceTimer","start","logger","operation","verbose","end","duration","Date","now","startTime","EXCLUDED_DIRECTORIES","batchReadPackageJsonFiles","packageJsonPaths","storage","rootDir","getLogger","timer","length","readPromises","map","packageJsonPath","packageJsonContent","readFile","parsed","safeJsonParse","packageJson","validatePackageJson","relativePath","path","relative","dirname","error","debug","message","results","Promise","all","validResults","filter","result","findAllPackageJsonFiles","scanForPaths","currentDir","depth","exists","isDirectory","items","listFiles","foundPaths","includes","join","push","subdirPromises","item","itemPath","subdirResults","subdirPaths","pathsTimer","allPaths","packageJsonFiles"],"mappings":"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"}
|