@crossplatformai/dependency-graph 0.10.0 ā 0.11.0-next.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli/pr-preview.d.ts +2 -0
- package/dist/cli/pr-preview.d.ts.map +1 -1
- package/dist/index-cli.js +167 -45
- package/dist/index-cli.js.map +1 -1
- package/dist/index.d.ts +2 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +114 -17
- package/dist/index.js.map +1 -1
- package/dist/workflow/impact.d.ts +5 -0
- package/dist/workflow/impact.d.ts.map +1 -0
- package/dist/workflow/types.d.ts +8 -1
- package/dist/workflow/types.d.ts.map +1 -1
- package/dist/workflow/validator.d.ts.map +1 -1
- package/package.json +5 -7
package/dist/index-cli.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/cli/pr-preview.ts","../src/graph/builder.ts","../src/graph/traversal.ts","../src/workspace/discovery.ts","../src/workspace/package-map.ts","../src/workflow/policy.ts","../src/workflow/validator.ts","../src/workflow/discovery.ts","../src/workflow/expected-paths.ts","../src/workflow/parser.ts","../src/cli/validate-workflows.ts","../src/cli/index.ts"],"sourcesContent":["#!/usr/bin/env node\n\nimport { readFileSync } from 'node:fs';\nimport { resolve } from 'node:path';\nimport { glob } from 'glob';\nimport { parse as parseYaml } from 'yaml';\n\n// Import dependency graph functions from parent module\nimport {\n discoverWorkspaces,\n buildDependencyGraph,\n findAffectedPackages,\n type WorkspacePackage,\n} from '../index';\n\ninterface ChangedFile {\n path: string;\n status: 'modified' | 'added' | 'deleted';\n}\n\ninterface DeploymentIndicator {\n file?: string;\n field?: string;\n dependency?: string;\n}\n\ninterface PlatformDetection {\n platform: string;\n indicators: DeploymentIndicator[];\n}\n\nconst WORKFLOW_MAPPING: Record<string, string> = {\n 'deploy-web.yml': 'web',\n 'deploy-api-origin.yml': 'api-origin',\n 'deploy-api-edge.yml': 'api-edge',\n 'release-mobile.yml': 'mobile',\n 'release-desktop.yml': 'desktop',\n 'release-cli.yml': 'cli',\n};\n\n// Convention-based deployment detection (order matters - most specific first)\nconst PLATFORM_INDICATORS: PlatformDetection[] = [\n {\n platform: 'cloudflare-workers',\n indicators: [{ file: 'wrangler.toml' }],\n },\n {\n platform: 'expo',\n indicators: [{ file: 'app.json' }, { file: 'eas.json' }],\n },\n {\n platform: 'npm-package',\n indicators: [{ field: 'bin' }],\n },\n {\n platform: 'electron',\n indicators: [\n { dependency: 'electron' },\n { dependency: 'electron-builder' },\n ],\n },\n {\n platform: 'next.js',\n indicators: [\n { file: 'next.config.js' },\n { file: 'next.config.mjs' },\n { file: 'next.config.ts' },\n { dependency: 'next' },\n ],\n },\n {\n platform: 'node.js',\n indicators: [\n { field: 'start' }, // has start script\n ],\n },\n];\n\nconst DEFAULT_APP_DIRECTORIES = ['apps'];\n\nfunction isDeployableApp(pkg: WorkspacePackage): {\n deployable: boolean;\n platform?: string;\n} {\n // Check if package is in apps directory (configurable in future)\n const isInAppDirectory = DEFAULT_APP_DIRECTORIES.some(\n (dir) => pkg.path.includes(`/${dir}/`) || pkg.path.endsWith(`/${dir}`),\n );\n\n if (!isInAppDirectory) {\n return { deployable: false };\n }\n\n // Detect platform based on indicators\n for (const platformDetection of PLATFORM_INDICATORS) {\n const hasIndicators = platformDetection.indicators.some((indicator) => {\n if (indicator.file) {\n try {\n const filePath = resolve(pkg.path, indicator.file);\n readFileSync(filePath, 'utf-8');\n return true;\n } catch {\n return false;\n }\n }\n\n if (indicator.field) {\n const scripts = pkg.packageJson.scripts as\n | Record<string, string>\n | undefined;\n if (indicator.field === 'start' && scripts?.start) {\n return true;\n }\n const packageJsonField =\n pkg.packageJson[indicator.field as keyof typeof pkg.packageJson];\n if (indicator.field !== 'start' && packageJsonField) {\n return true;\n }\n }\n\n if (indicator.dependency) {\n return !!(\n pkg.dependencies[indicator.dependency] ||\n pkg.devDependencies[indicator.dependency]\n );\n }\n\n return false;\n });\n\n if (hasIndicators) {\n return { deployable: true, platform: platformDetection.platform };\n }\n }\n\n // If in apps directory but no specific platform detected, not deployable\n return { deployable: false };\n}\n\nasync function getChangedFiles(): Promise<ChangedFile[]> {\n const { execSync } = await import('node:child_process');\n\n try {\n // Try local production first, then fall back to origin/production\n let baseBranch = 'production';\n try {\n execSync('git rev-parse production', {\n encoding: 'utf-8',\n stdio: 'pipe',\n });\n } catch {\n baseBranch = 'origin/production';\n }\n\n const output = execSync(`git diff --name-status ${baseBranch}...HEAD`, {\n encoding: 'utf-8',\n });\n\n return output\n .trim()\n .split('\\n')\n .filter((line) => line)\n .map((line) => {\n const [status, path] = line.split('\\t');\n\n return {\n path: path ?? '',\n status:\n status === 'D' ? 'deleted' : status === 'A' ? 'added' : 'modified',\n };\n })\n .filter((file): file is ChangedFile => file.path !== '');\n } catch {\n console.error(\n 'Failed to get changed files. Ensure you are on a branch with commits compared to production.',\n );\n process.exit(1);\n }\n}\n\nexport async function runPrPreview(): Promise<void> {\n try {\n const rootDir = resolve(process.cwd());\n const changedFiles = await getChangedFiles();\n\n console.log(\n `\\nš¦ Release Preview - Changed Files: ${changedFiles.length}\\n`,\n );\n\n changedFiles.forEach((file) => {\n console.log(` ${file.status === 'deleted' ? 'ā' : 'š'} ${file.path}`);\n });\n\n const packages = await discoverWorkspaces(rootDir, {\n fs: {\n readFile: (path) => {\n try {\n return Promise.resolve(readFileSync(path, 'utf-8'));\n } catch {\n return Promise.reject(new Error(`Failed to read file: ${path}`));\n }\n },\n exists: (path) => {\n try {\n readFileSync(path, 'utf-8');\n return Promise.resolve(true);\n } catch {\n return Promise.resolve(false);\n }\n },\n },\n glob: {\n glob: async (pattern, options) => glob(pattern, options),\n },\n yaml: {\n parse: (content) => parseYaml(content) as Record<string, unknown>,\n },\n });\n\n const graph = buildDependencyGraph(packages);\n const changedPackages = new Set<string>();\n\n for (const file of changedFiles) {\n if (file.status === 'deleted') continue;\n\n // Convert absolute package paths to relative for comparison\n const pkg = packages.find((p) => {\n const relativePkgPath = p.path.replace(rootDir + '/', '');\n return file.path.startsWith(relativePkgPath);\n });\n\n if (pkg) {\n changedPackages.add(pkg.name);\n }\n }\n\n const changedWorkflows = changedFiles.filter((f) =>\n f.path.startsWith('.github/workflows/'),\n );\n\n const workflowAffectedApps = new Set<string>();\n for (const workflow of changedWorkflows) {\n const workflowName = workflow.path.split('/').pop();\n if (workflowName && WORKFLOW_MAPPING[workflowName]) {\n workflowAffectedApps.add(WORKFLOW_MAPPING[workflowName]);\n }\n }\n\n console.log(\n `\\nš¦ Changed packages: ${Array.from(changedPackages).join(', ') || 'none'}\\n`,\n );\n\n if (changedPackages.size === 0) {\n console.log('\\n⨠No packages changed - no deployments needed\\n');\n return;\n }\n\n const affected = findAffectedPackages(changedPackages, graph, {\n direction: 'upstream',\n respectAffectsUpstream: true,\n });\n\n // Find all deployable apps (affected and unaffected)\n const allDeployableApps = packages\n .map((pkg) => {\n const detection = isDeployableApp(pkg);\n return detection.deployable\n ? { name: pkg.name, pkg, platform: detection.platform }\n : null;\n })\n .filter(\n (\n item,\n ): item is {\n name: string;\n pkg: WorkspacePackage;\n platform: string | undefined;\n } => item !== null,\n );\n\n // Categorize apps by deployment type\n const affectedApps = allDeployableApps.filter(\n (app): app is NonNullable<typeof app> =>\n affected.has(app.name) || workflowAffectedApps.has(app.name),\n );\n const unaffectedApps = allDeployableApps.filter(\n (app): app is NonNullable<typeof app> =>\n !affected.has(app.name) && !workflowAffectedApps.has(app.name),\n );\n\n // Separate deploys (web-based) vs releases (installed)\n const getAppCategory = (platform?: string) => {\n switch (platform) {\n case 'next.js':\n case 'cloudflare-workers':\n case 'node.js':\n return 'deploy';\n case 'expo':\n case 'electron':\n case 'npm-package':\n return 'release';\n default:\n return 'deploy'; // default to deploy for generic\n }\n };\n\n const getAppIcon = (platform?: string) => {\n switch (platform) {\n case 'next.js':\n case 'cloudflare-workers':\n case 'node.js':\n return 'š';\n case 'expo':\n return 'š±';\n case 'electron':\n return 'š„ļø';\n case 'npm-package':\n return 'ā”';\n default:\n return 'š';\n }\n };\n\n const affectedDeploys = affectedApps.filter(\n (app): app is NonNullable<typeof app> =>\n getAppCategory(app.platform) === 'deploy',\n );\n const affectedReleases = affectedApps.filter(\n (app): app is NonNullable<typeof app> =>\n getAppCategory(app.platform) === 'release',\n );\n\n console.log(`\\nš PR Preview\\n`);\n\n if (affectedApps.length === 0) {\n console.log('No apps affected by changes.\\n');\n } else {\n if (affectedDeploys.length > 0) {\n console.log('š Apps that will be DEPLOYED:');\n for (const item of affectedDeploys) {\n const platformDisplay =\n item.platform === 'generic' ? '' : ` (${item.platform})`;\n const icon = getAppIcon(item.platform);\n console.log(`${icon} ${item.name}${platformDisplay}`);\n }\n console.log('');\n }\n\n if (affectedReleases.length > 0) {\n console.log('š Apps that will be RELEASED:');\n for (const item of affectedReleases) {\n const platformDisplay =\n item.platform === 'generic' ? '' : ` (${item.platform})`;\n const icon = getAppIcon(item.platform);\n console.log(`${icon} ${item.name}${platformDisplay}`);\n }\n console.log('');\n }\n }\n\n if (unaffectedApps.length > 0) {\n console.log(`š Apps that will NOT be affected:`);\n for (const item of unaffectedApps) {\n const platformDisplay =\n item.platform === 'generic' ? '' : ` (${item.platform})`;\n console.log(`āļø ${item.name}${platformDisplay}`);\n }\n console.log('');\n }\n\n console.log(`Legend:`);\n console.log(`š = Deploy (web-based, instant updates)`);\n console.log(`š± = Release (mobile app, user installs)`);\n console.log(`š„ļø = Release (desktop app, user installs)`);\n console.log(`ā” = Release (CLI tool, user installs)`);\n console.log(`āļø = Unaffected (no changes needed)`);\n console.log(`š = File changed`);\n console.log(`ā = File deleted\\n`);\n } catch (error) {\n console.error(\n 'Error:',\n error instanceof Error ? error.message : String(error),\n );\n process.exit(1);\n }\n}\n\nif (import.meta.url === new URL(process.argv[1] ?? '', 'file:').href) {\n void runPrPreview();\n}\n","import type { WorkspacePackage, DependencyGraph } from './types';\n\nexport function buildDependencyGraph(\n packages: WorkspacePackage[],\n): DependencyGraph {\n const graph: DependencyGraph = {\n packages: new Map(),\n dependsOn: new Map(),\n dependedBy: new Map(),\n };\n\n for (const pkg of packages) {\n graph.packages.set(pkg.name, pkg);\n graph.dependsOn.set(pkg.name, new Set());\n graph.dependedBy.set(pkg.name, new Set());\n }\n\n for (const pkg of packages) {\n const allDeps = {\n ...pkg.dependencies,\n ...pkg.devDependencies,\n };\n\n for (const depName of Object.keys(allDeps)) {\n const matchedPkg = findMatchingPackage(depName, packages);\n\n if (matchedPkg) {\n graph.dependsOn.get(pkg.name)!.add(matchedPkg.name);\n graph.dependedBy.get(matchedPkg.name)!.add(pkg.name);\n }\n }\n }\n\n return graph;\n}\n\nfunction findMatchingPackage(\n depName: string,\n packages: WorkspacePackage[],\n): WorkspacePackage | undefined {\n let match = packages.find((p) => p.name === depName);\n if (match) return match;\n\n match = packages.find((p) => p.name === `@repo/${depName}`);\n if (match) return match;\n\n const nameWithoutRepo = depName.replace(/^@repo\\//, '');\n match = packages.find((p) => p.name === nameWithoutRepo);\n if (match) return match;\n\n return undefined;\n}\n","import type { DependencyGraph, TraversalOptions } from './types';\n\nexport function findAffectedPackages(\n startingPackages: Set<string>,\n graph: DependencyGraph,\n options: TraversalOptions = {},\n): Set<string> {\n const {\n direction = 'upstream',\n maxDepth = Infinity,\n filter,\n respectAffectsUpstream = false,\n } = options;\n\n const affected = new Set<string>(startingPackages);\n const queue: Array<{ name: string; depth: number }> = Array.from(\n startingPackages,\n ).map((name) => ({ name, depth: 0 }));\n const visited = new Set<string>();\n\n while (queue.length > 0) {\n const current = queue.shift()!;\n\n if (visited.has(current.name)) continue;\n visited.add(current.name);\n\n if (current.depth >= maxDepth) {\n continue;\n }\n\n const pkg = graph.packages.get(current.name);\n if (!pkg) continue;\n\n if (respectAffectsUpstream && direction === 'upstream') {\n const release = pkg.packageJson.release;\n if (release && release.affectsUpstream === false) {\n continue;\n }\n }\n\n const nextPackages = new Set<string>();\n\n if (direction === 'upstream' || direction === 'both') {\n const upstream = graph.dependedBy.get(current.name) || new Set();\n upstream.forEach((p) => nextPackages.add(p));\n }\n\n if (direction === 'downstream' || direction === 'both') {\n const downstream = graph.dependsOn.get(current.name) || new Set();\n downstream.forEach((p) => nextPackages.add(p));\n }\n\n for (const pkgName of nextPackages) {\n const nextPkg = graph.packages.get(pkgName);\n\n if (filter && nextPkg && !filter(nextPkg)) {\n continue;\n }\n\n if (!affected.has(pkgName)) {\n affected.add(pkgName);\n queue.push({ name: pkgName, depth: current.depth + 1 });\n }\n }\n }\n\n return affected;\n}\n\nexport function findDependencyPath(\n from: string,\n to: string,\n graph: DependencyGraph,\n): string[] | null {\n const queue: string[][] = [[from]];\n const visited = new Set<string>([from]);\n\n while (queue.length > 0) {\n const path = queue.shift()!;\n const current = path[path.length - 1];\n\n if (!current) {\n continue;\n }\n\n if (current === to) {\n return path;\n }\n\n const dependents = graph.dependedBy.get(current) || new Set();\n for (const dependent of dependents) {\n if (!visited.has(dependent)) {\n visited.add(dependent);\n queue.push([...path, dependent]);\n }\n }\n }\n\n return null;\n}\n\nexport function findAllPaths(\n from: string,\n to: string,\n graph: DependencyGraph,\n maxPaths = 10,\n): string[][] {\n const paths: string[][] = [];\n const visited = new Set<string>();\n\n function dfs(current: string, path: string[]): void {\n if (paths.length >= maxPaths) return;\n\n if (current === to) {\n paths.push([...path]);\n return;\n }\n\n if (visited.has(current)) return;\n visited.add(current);\n\n const dependents = graph.dependedBy.get(current) || new Set();\n for (const dependent of dependents) {\n dfs(dependent, [...path, dependent]);\n }\n\n visited.delete(current);\n }\n\n dfs(from, [from]);\n return paths;\n}\n","import { join, resolve } from 'node:path';\nimport type { WorkspacePackage } from '../graph/types';\nimport type {\n FileSystemClient,\n GlobClient,\n YamlClient,\n} from '../types/clients';\n\nexport interface WorkspaceConfig {\n packages: string[];\n}\n\nexport interface WorkspaceDiscoveryConfig {\n fs: FileSystemClient;\n glob: GlobClient;\n yaml: YamlClient;\n}\n\nexport async function discoverWorkspaces(\n rootDir: string,\n config: WorkspaceDiscoveryConfig,\n): Promise<WorkspacePackage[]> {\n const workspaceConfig = await loadWorkspaceConfig(rootDir, config);\n const packages: WorkspacePackage[] = [];\n\n for (const pattern of workspaceConfig.packages) {\n if (pattern.startsWith('!')) continue;\n\n const pkgDirs = await findPackageDirectories(rootDir, pattern, config);\n\n for (const pkgDir of pkgDirs) {\n const pkgJsonPath = join(pkgDir, 'package.json');\n\n try {\n const pkgJsonContent: string = await config.fs.readFile(\n pkgJsonPath,\n 'utf-8',\n );\n const pkgJson = JSON.parse(pkgJsonContent) as {\n name: string;\n version?: string;\n dependencies?: Record<string, string>;\n devDependencies?: Record<string, string>;\n [key: string]: unknown;\n };\n\n packages.push({\n name: pkgJson.name,\n version: pkgJson.version || '0.0.0',\n path: pkgDir,\n packageJson: pkgJson,\n dependencies: pkgJson.dependencies || {},\n devDependencies: pkgJson.devDependencies || {},\n });\n } catch {\n continue;\n }\n }\n }\n\n return packages;\n}\n\nasync function loadWorkspaceConfig(\n rootDir: string,\n config: WorkspaceDiscoveryConfig,\n): Promise<WorkspaceConfig> {\n const workspaceFilePath = join(rootDir, 'pnpm-workspace.yaml');\n\n try {\n const content: string = await config.fs.readFile(\n workspaceFilePath,\n 'utf-8',\n );\n const parsed = config.yaml.parse(content) as WorkspaceConfig;\n return parsed;\n } catch {\n return { packages: [] };\n }\n}\n\nasync function findPackageDirectories(\n rootDir: string,\n pattern: string,\n config: WorkspaceDiscoveryConfig,\n): Promise<string[]> {\n const matches: string[] = await config.glob.glob(pattern, {\n cwd: rootDir,\n absolute: false,\n ignore: ['**/node_modules/**', '**/.next/**', '**/dist/**'],\n });\n\n return matches.map((match: string) => resolve(rootDir, match));\n}\n","import { existsSync } from 'node:fs';\nimport { join, relative, sep } from 'node:path';\nimport type { WorkspacePackage } from '../graph/types';\n\nexport interface MappedWorkspacePath {\n filesystemPath: string;\n workflowPath: string | null;\n}\n\nexport interface WorkspacePackageMap {\n packageMap: Map<string, MappedWorkspacePath>;\n workspaceRoots: Set<string>;\n}\n\nexport function normalizeRelativePath(path: string): string {\n return path.split(sep).join('/');\n}\n\nfunction resolveWorkflowPath(\n relativePath: string,\n rootDir: string,\n): string | null {\n if (!relativePath.startsWith('../')) {\n return relativePath;\n }\n\n const segments = relativePath.split('/');\n\n for (let index = 0; index < segments.length; index += 1) {\n const candidateSegments = segments.slice(index);\n if (candidateSegments.length === 0 || candidateSegments[0] === '..') {\n continue;\n }\n\n const candidatePath = candidateSegments.join('/');\n if (existsSync(join(rootDir, candidatePath))) {\n return candidatePath;\n }\n }\n\n return null;\n}\n\nexport function buildPackageMap(\n packages: WorkspacePackage[],\n rootDir: string,\n): WorkspacePackageMap {\n const packageMap = new Map<string, MappedWorkspacePath>();\n const workspaceRoots = new Set<string>();\n\n for (const pkg of packages) {\n const relativePath = normalizeRelativePath(relative(rootDir, pkg.path));\n const workflowPath = resolveWorkflowPath(relativePath, rootDir);\n\n packageMap.set(pkg.name, {\n filesystemPath: relativePath,\n workflowPath,\n });\n\n if (!workflowPath) {\n continue;\n }\n\n const [workspaceRoot] = workflowPath.split('/');\n if (workspaceRoot) {\n workspaceRoots.add(workspaceRoot);\n }\n }\n\n return {\n packageMap,\n workspaceRoots,\n };\n}\n","import type { WorkflowValidationPolicy } from './types';\n\nexport const defaultWorkflowValidationPolicy: WorkflowValidationPolicy = {\n workflowFilePatterns: ['deploy-*.yml', 'release-*.yml'],\n allowedRootPaths: [\n 'package.json',\n 'pnpm-lock.yaml',\n 'pnpm-workspace.yaml',\n 'turbo.json',\n ],\n includeDevDependenciesForRootPackage: true,\n includeDevDependenciesTransitively: false,\n};\n","import { existsSync } from 'node:fs';\nimport { readFile } from 'node:fs/promises';\nimport { glob } from 'glob';\nimport { parse as parseYaml } from 'yaml';\nimport { discoverWorkspaces } from '../workspace/discovery';\nimport { buildPackageMap } from '../workspace/package-map';\nimport type {\n WorkflowValidationIssue,\n WorkflowValidationPolicy,\n WorkflowValidationResult,\n} from './types';\nimport { discoverWorkflowTargets } from './discovery';\nimport { getExpectedWorkflowPaths } from './expected-paths';\nimport { parseWorkflowFile } from './parser';\nimport { defaultWorkflowValidationPolicy } from './policy';\n\nfunction uniqueSorted(values: string[]): string[] {\n return Array.from(new Set(values)).sort();\n}\n\nfunction buildBroadWildcards(workspaceRoots: Set<string>): Set<string> {\n const wildcards = new Set<string>();\n\n for (const root of workspaceRoots) {\n wildcards.add(`${root}/*`);\n wildcards.add(`${root}/**`);\n }\n\n return wildcards;\n}\n\nfunction splitActualPaths(\n paths: string[],\n workspaceRoots: Set<string>,\n allowedRootPaths: string[],\n workflowPath: string,\n): { workspacePaths: string[]; ignoredPaths: string[] } {\n const workspacePaths: string[] = [];\n const ignoredPaths: string[] = [];\n\n for (const path of paths) {\n if (path === workflowPath || allowedRootPaths.includes(path)) {\n ignoredPaths.push(path);\n continue;\n }\n\n const [rootSegment] = path.split('/');\n if (rootSegment && workspaceRoots.has(rootSegment)) {\n workspacePaths.push(path);\n continue;\n }\n\n ignoredPaths.push(path);\n }\n\n return {\n workspacePaths: uniqueSorted(workspacePaths),\n ignoredPaths: uniqueSorted(ignoredPaths),\n };\n}\n\nfunction isCoveredByExpectedPath(\n actualPath: string,\n expectedPaths: string[],\n): boolean {\n return expectedPaths.some((expectedPath) => {\n if (expectedPath === actualPath) {\n return true;\n }\n\n if (!expectedPath.endsWith('/**')) {\n return false;\n }\n\n const expectedPrefix = expectedPath.slice(0, -3);\n return actualPath.startsWith(`${expectedPrefix}/`);\n });\n}\n\nfunction validateWorkflowResult(\n workflow: string,\n targetPackage: string,\n expectedPaths: string[],\n actualPaths: string[],\n broadWildcards: Set<string>,\n): WorkflowValidationResult {\n const issues: WorkflowValidationIssue[] = [];\n const missing = expectedPaths.filter((path) => !actualPaths.includes(path));\n const unnecessary = actualPaths.filter(\n (path) => !isCoveredByExpectedPath(path, expectedPaths),\n );\n\n for (const path of missing) {\n issues.push({\n kind: 'missing',\n path,\n message: `Missing path '${path}'`,\n });\n }\n\n for (const path of unnecessary) {\n issues.push({\n kind: 'unnecessary',\n path,\n message: `Unnecessary path '${path}'`,\n });\n }\n\n for (const path of actualPaths) {\n if (broadWildcards.has(path)) {\n issues.push({\n kind: 'broad-wildcard',\n path,\n message: `Uses broad wildcard '${path}' which triggers on all workspace changes under that root`,\n });\n }\n }\n\n return {\n workflow,\n targetPackage,\n valid: issues.length === 0,\n expectedPaths,\n actualPaths,\n missing,\n unnecessary,\n issues,\n };\n}\n\nexport async function validateWorkflows(\n rootDir: string,\n policyOverrides?: Partial<WorkflowValidationPolicy>,\n): Promise<WorkflowValidationResult[]> {\n const discoveredPackages = await discoverWorkspaces(rootDir, {\n fs: {\n readFile: (path, encoding) => readFile(path, encoding),\n exists: (path) => Promise.resolve(existsSync(path)),\n },\n glob: {\n glob: (pattern, options) => glob(pattern, options),\n },\n yaml: {\n parse: (content): unknown => parseYaml(content),\n },\n });\n\n const policy: WorkflowValidationPolicy = {\n ...defaultWorkflowValidationPolicy,\n ...policyOverrides,\n allowedRootPaths: uniqueSorted([\n ...defaultWorkflowValidationPolicy.allowedRootPaths,\n ...(policyOverrides?.allowedRootPaths ?? []),\n ]),\n };\n\n const { packageMap, workspaceRoots } = buildPackageMap(\n discoveredPackages,\n rootDir,\n );\n const workflowTargets = discoverWorkflowTargets(\n rootDir,\n discoveredPackages,\n policy,\n );\n const broadWildcards = buildBroadWildcards(workspaceRoots);\n\n return workflowTargets.map((workflowTarget) => {\n if (!workflowTarget.targetPackage) {\n return {\n workflow: workflowTarget.workflowFile,\n targetPackage: workflowTarget.targetSlug,\n valid: false,\n expectedPaths: [],\n actualPaths: [],\n missing: [],\n unnecessary: [],\n issues: [\n {\n kind: 'config-error',\n message: `Could not resolve workflow target package for '${workflowTarget.workflowFile}'`,\n },\n ],\n };\n }\n\n try {\n const parsedWorkflow = parseWorkflowFile(\n workflowTarget.workflowFile,\n rootDir,\n );\n\n if (parsedWorkflow.paths.length === 0) {\n return {\n workflow: workflowTarget.workflowFile,\n targetPackage: workflowTarget.targetPackage,\n valid: true,\n expectedPaths: [],\n actualPaths: [],\n missing: [],\n unnecessary: [],\n issues: [],\n };\n }\n\n const { workspacePaths: actualPaths } = splitActualPaths(\n parsedWorkflow.paths,\n workspaceRoots,\n policy.allowedRootPaths,\n workflowTarget.workflowPath,\n );\n const expectedPaths = getExpectedWorkflowPaths({\n workflowTarget,\n packages: discoveredPackages,\n packageMap,\n policy,\n });\n\n return validateWorkflowResult(\n workflowTarget.workflowFile,\n workflowTarget.targetPackage,\n expectedPaths,\n actualPaths,\n broadWildcards,\n );\n } catch (error) {\n return {\n workflow: workflowTarget.workflowFile,\n targetPackage: workflowTarget.targetPackage,\n valid: false,\n expectedPaths: [],\n actualPaths: [],\n missing: [],\n unnecessary: [],\n issues: [\n {\n kind: 'parse-error',\n message: error instanceof Error ? error.message : String(error),\n },\n ],\n };\n }\n });\n}\n","import { existsSync, readdirSync } from 'node:fs';\nimport { join } from 'node:path';\nimport type { WorkspacePackage } from '../graph/types';\nimport { buildPackageMap } from '../workspace/package-map';\nimport type { WorkflowTarget, WorkflowValidationPolicy } from './types';\n\nfunction patternToRegExp(pattern: string): RegExp {\n const escaped = pattern.replace(/[.+?^${}()|[\\]\\\\]/g, '\\\\$&');\n return new RegExp(`^${escaped.replace(/\\*/g, '.*')}$`);\n}\n\nexport function discoverWorkflowTargets(\n rootDir: string,\n packages: WorkspacePackage[],\n policy: WorkflowValidationPolicy,\n): WorkflowTarget[] {\n const workflowsDir = join(rootDir, '.github/workflows');\n if (!existsSync(workflowsDir)) {\n return [];\n }\n\n const { packageMap } = buildPackageMap(packages, rootDir);\n const appPackages = packages.filter((pkg) =>\n packageMap.get(pkg.name)?.workflowPath?.startsWith('apps/'),\n );\n const bySlug = new Map<string, string>();\n\n for (const pkg of appPackages) {\n const relativePath = packageMap.get(pkg.name)?.workflowPath;\n if (!relativePath) {\n continue;\n }\n\n const slug = relativePath.split('/').at(-1);\n if (slug) {\n bySlug.set(slug, pkg.name);\n }\n }\n\n const allowedPatterns = policy.workflowFilePatterns.map(patternToRegExp);\n const files = readdirSync(workflowsDir)\n .filter((file) => file.endsWith('.yml'))\n .filter((file) => allowedPatterns.some((pattern) => pattern.test(file)));\n\n return files\n .map((workflowFile) => {\n const match = /^(?:deploy|release)-(.+)\\.yml$/.exec(workflowFile);\n const targetSlug = match?.[1] ?? workflowFile.replace(/\\.yml$/, '');\n\n return {\n workflowFile,\n workflowPath: `.github/workflows/${workflowFile}`,\n targetSlug,\n targetPackage: bySlug.get(targetSlug) ?? null,\n };\n })\n .sort((a, b) => a.workflowFile.localeCompare(b.workflowFile));\n}\n","import type { WorkspacePackage } from '../graph/types';\nimport type { WorkflowTarget, WorkflowValidationPolicy } from './types';\n\ninterface ExpectedPathsOptions {\n workflowTarget: WorkflowTarget;\n packages: WorkspacePackage[];\n packageMap: Map<string, { workflowPath: string | null }>;\n policy: WorkflowValidationPolicy;\n}\n\nfunction uniqueSorted(values: string[]): string[] {\n return Array.from(new Set(values)).sort();\n}\n\nfunction resolveWorkspaceDependency(\n dependencyName: string,\n packages: WorkspacePackage[],\n): WorkspacePackage | undefined {\n let match = packages.find((pkg) => pkg.name === dependencyName);\n if (match) {\n return match;\n }\n\n match = packages.find((pkg) => pkg.name === `@repo/${dependencyName}`);\n if (match) {\n return match;\n }\n\n const nameWithoutRepo = dependencyName.replace(/^@repo\\//, '');\n return packages.find((pkg) => pkg.name === nameWithoutRepo);\n}\n\nfunction collectWorkspaceDependencyNames(\n pkg: WorkspacePackage,\n packages: WorkspacePackage[],\n visited: Set<string>,\n includeDevDependencies: boolean,\n includeDevDependenciesTransitively: boolean,\n): Set<string> {\n const collected = new Set<string>();\n const dependencyEntries = Object.entries(pkg.dependencies);\n const devDependencyEntries = includeDevDependencies\n ? Object.entries(pkg.devDependencies)\n : [];\n\n for (const [dependencyName] of [\n ...dependencyEntries,\n ...devDependencyEntries,\n ]) {\n const dependency = resolveWorkspaceDependency(dependencyName, packages);\n if (!dependency || visited.has(dependency.name)) {\n continue;\n }\n\n visited.add(dependency.name);\n collected.add(dependency.name);\n\n const nestedDependencies = collectWorkspaceDependencyNames(\n dependency,\n packages,\n visited,\n includeDevDependenciesTransitively,\n includeDevDependenciesTransitively,\n );\n\n for (const nestedDependency of nestedDependencies) {\n collected.add(nestedDependency);\n }\n }\n\n return collected;\n}\n\nexport function getExpectedWorkflowPaths({\n workflowTarget,\n packages,\n packageMap,\n policy,\n}: ExpectedPathsOptions): string[] {\n const targetPackageName = workflowTarget.targetPackage;\n if (!targetPackageName) {\n return [];\n }\n\n const targetPackage = packages.find((pkg) => pkg.name === targetPackageName);\n if (!targetPackage) {\n return [];\n }\n\n const dependencyNames = collectWorkspaceDependencyNames(\n targetPackage,\n packages,\n new Set<string>(),\n policy.includeDevDependenciesForRootPackage,\n policy.includeDevDependenciesTransitively,\n );\n\n const expectedPaths: string[] = [];\n const targetPackagePath = packageMap.get(targetPackageName)?.workflowPath;\n if (targetPackagePath) {\n expectedPaths.push(`${targetPackagePath}/**`);\n }\n\n for (const dependencyName of dependencyNames) {\n const dependencyPath = packageMap.get(dependencyName)?.workflowPath;\n if (dependencyPath) {\n expectedPaths.push(`${dependencyPath}/**`);\n }\n }\n\n return uniqueSorted(expectedPaths);\n}\n","import { existsSync, readFileSync } from 'node:fs';\nimport { join } from 'node:path';\nimport { parse as parseYaml } from 'yaml';\nimport type { ParsedWorkflow } from './types';\n\ninterface WorkflowConfig {\n name?: string;\n on?: {\n push?: {\n paths?: string[];\n };\n pull_request?: {\n paths?: string[];\n };\n };\n}\n\nfunction uniqueSorted(values: string[]): string[] {\n return Array.from(new Set(values)).sort();\n}\n\nexport function parseWorkflowFile(\n workflowFile: string,\n rootDir: string,\n): ParsedWorkflow {\n const workflowPath = join(rootDir, '.github/workflows', workflowFile);\n\n if (!existsSync(workflowPath)) {\n throw new Error(`Workflow file not found: ${workflowFile}`);\n }\n\n const content = readFileSync(workflowPath, 'utf-8');\n const workflow = parseYaml(content) as WorkflowConfig;\n const pushPaths = workflow.on?.push?.paths ?? [];\n const pullRequestPaths = workflow.on?.pull_request?.paths ?? [];\n\n const parsedWorkflow: ParsedWorkflow = {\n pushPaths: uniqueSorted(pushPaths),\n pullRequestPaths: uniqueSorted(pullRequestPaths),\n paths: uniqueSorted([...pushPaths, ...pullRequestPaths]),\n };\n\n if (workflow.name) {\n parsedWorkflow.name = workflow.name;\n }\n\n return parsedWorkflow;\n}\n","#!/usr/bin/env node\n\nimport { existsSync, readFileSync, readdirSync } from 'node:fs';\nimport { join, resolve } from 'node:path';\nimport { parse as parseYaml } from 'yaml';\nimport { validateWorkflows } from '../workflow/validator';\nimport type { WorkflowValidationResult } from '../workflow/types';\n\ninterface PackageJson {\n packageManager?: string;\n}\n\ninterface PnpmValidationResult {\n valid: boolean;\n workflowIssues: { workflow: string; issue: string }[];\n dockerfileIssues: { dockerfile: string; issue: string }[];\n}\n\nfunction discoverPnpmWorkflows(rootDir: string): string[] {\n const workflowsDir = join(rootDir, '.github/workflows');\n if (!existsSync(workflowsDir)) {\n return [];\n }\n\n const workflows: string[] = [];\n const files = readdirSync(workflowsDir).filter((file) =>\n file.endsWith('.yml'),\n );\n\n for (const file of files) {\n const content = readFileSync(join(workflowsDir, file), 'utf-8');\n if (content.includes('pnpm/action-setup')) {\n workflows.push(file);\n }\n }\n\n return workflows;\n}\n\nfunction discoverPnpmDockerfiles(rootDir: string): string[] {\n const dockerfiles: string[] = [];\n\n for (const entry of readdirSync(rootDir)) {\n if (!entry.startsWith('Dockerfile')) {\n continue;\n }\n\n const content = readFileSync(join(rootDir, entry), 'utf-8');\n if (content.includes('pnpm@')) {\n dockerfiles.push(entry);\n }\n }\n\n const appsDir = join(rootDir, 'apps');\n if (!existsSync(appsDir)) {\n return dockerfiles;\n }\n\n for (const entry of readdirSync(appsDir, { withFileTypes: true })) {\n if (!entry.isDirectory()) {\n continue;\n }\n\n const dockerfilePath = join(appsDir, entry.name, 'Dockerfile');\n if (!existsSync(dockerfilePath)) {\n continue;\n }\n\n const content = readFileSync(dockerfilePath, 'utf-8');\n if (content.includes('pnpm@')) {\n dockerfiles.push(`apps/${entry.name}/Dockerfile`);\n }\n }\n\n return dockerfiles;\n}\n\nfunction getExpectedPnpmVersion(rootDir: string): string {\n const packageJson = JSON.parse(\n readFileSync(join(rootDir, 'package.json'), 'utf-8'),\n ) as PackageJson;\n const packageManager = packageJson.packageManager;\n\n if (!packageManager?.startsWith('pnpm@')) {\n throw new Error('packageManager field must specify pnpm version');\n }\n\n return packageManager.replace('pnpm@', '');\n}\n\nfunction checkWorkflowPnpmVersion(\n workflowFile: string,\n rootDir: string,\n): { valid: boolean; issue?: string } {\n const workflowPath = join(rootDir, '.github/workflows', workflowFile);\n if (!existsSync(workflowPath)) {\n return { valid: true };\n }\n\n const content = readFileSync(workflowPath, 'utf-8');\n const workflow = parseYaml(content) as {\n jobs?: Record<\n string,\n { steps?: Array<{ uses?: string; with?: { version?: string | number } }> }\n >;\n };\n\n for (const job of Object.values(workflow.jobs ?? {})) {\n for (const step of job.steps ?? []) {\n if (!step.uses?.startsWith('pnpm/action-setup')) {\n continue;\n }\n\n const version = step.with?.version;\n if (version !== undefined && !/^\\d+$/.test(String(version))) {\n return {\n valid: false,\n issue: `Hardcoded pnpm version '${version}' - remove 'version' key to auto-detect from packageManager`,\n };\n }\n }\n }\n\n return { valid: true };\n}\n\nfunction checkDockerfilePnpmVersion(\n dockerfile: string,\n expectedVersion: string,\n rootDir: string,\n): { valid: boolean; issue?: string } {\n const dockerfilePath = join(rootDir, dockerfile);\n if (!existsSync(dockerfilePath)) {\n return { valid: true };\n }\n\n const content = readFileSync(dockerfilePath, 'utf-8');\n const matches = content.matchAll(/npm install -g pnpm@([\\d.]+)/g);\n\n for (const match of matches) {\n if (match[1] !== expectedVersion) {\n return {\n valid: false,\n issue: `${dockerfile} uses pnpm@${match[1]} but package.json specifies pnpm@${expectedVersion}`,\n };\n }\n }\n\n return { valid: true };\n}\n\nfunction validatePnpmVersions(rootDir: string): PnpmValidationResult {\n const result: PnpmValidationResult = {\n valid: true,\n workflowIssues: [],\n dockerfileIssues: [],\n };\n\n for (const workflowFile of discoverPnpmWorkflows(rootDir)) {\n const check = checkWorkflowPnpmVersion(workflowFile, rootDir);\n if (!check.valid && check.issue) {\n result.valid = false;\n result.workflowIssues.push({\n workflow: workflowFile,\n issue: check.issue,\n });\n }\n }\n\n const expectedVersion = getExpectedPnpmVersion(rootDir);\n for (const dockerfile of discoverPnpmDockerfiles(rootDir)) {\n const check = checkDockerfilePnpmVersion(\n dockerfile,\n expectedVersion,\n rootDir,\n );\n if (!check.valid && check.issue) {\n result.valid = false;\n result.dockerfileIssues.push({ dockerfile, issue: check.issue });\n }\n }\n\n return result;\n}\n\nfunction printResult(result: WorkflowValidationResult): void {\n const icon = result.valid ? 'ā
' : 'ā';\n console.log(`\\n${icon} ${result.workflow} (${result.targetPackage})`);\n\n if (result.valid) {\n console.log(' All paths match dependencies');\n return;\n }\n\n const extraIssues = result.issues.filter(\n (issue) => issue.kind !== 'missing' && issue.kind !== 'unnecessary',\n );\n\n if (extraIssues.length > 0) {\n console.log(' Issues:');\n for (const issue of extraIssues) {\n console.log(` ā ļø ${issue.message}`);\n }\n }\n\n if (result.missing.length > 0) {\n console.log(' Missing paths:');\n for (const path of result.missing) {\n console.log(` - ${path}`);\n }\n }\n\n if (result.unnecessary.length > 0) {\n console.log(' Unnecessary paths:');\n for (const path of result.unnecessary) {\n console.log(` - ${path}`);\n }\n }\n}\n\nexport async function runValidateWorkflows(): Promise<void> {\n const rootDir = resolve(process.cwd());\n let hasErrors = false;\n\n console.log(\n 'š Validating GitHub Actions workflows against dependencies...\\n',\n );\n const results = await validateWorkflows(rootDir);\n console.log(`Found ${results.length} workflow(s) to validate\\n`);\n\n if (results.length === 0) {\n console.log('No deploy-*.yml or release-*.yml workflows found.\\n');\n }\n\n for (const result of results) {\n printResult(result);\n }\n\n const validCount = results.filter((result) => result.valid).length;\n const invalidCount = results.length - validCount;\n\n console.log('\\n' + '='.repeat(60));\n console.log(\n `\\nš Path validation: ${validCount} valid, ${invalidCount} invalid\\n`,\n );\n\n if (invalidCount > 0) {\n console.log('ā Some workflows need updates to match dependencies');\n console.log('\\nTo fix:');\n console.log('1. Update workflow path filters to match missing paths');\n console.log('2. Remove unnecessary paths');\n console.log('3. Replace broad workspace wildcards with specific paths\\n');\n hasErrors = true;\n } else if (results.length > 0) {\n console.log('ā
All workflows match their dependencies!\\n');\n }\n\n console.log('='.repeat(60));\n console.log('\\nš Validating pnpm version consistency...\\n');\n\n const pnpmResult = validatePnpmVersions(rootDir);\n if (!pnpmResult.valid) {\n console.log('ā PNPM version issues found:\\n');\n for (const { workflow, issue } of pnpmResult.workflowIssues) {\n console.log(` ā ļø ${workflow}: ${issue}`);\n }\n for (const { issue } of pnpmResult.dockerfileIssues) {\n console.log(` ā ļø ${issue}`);\n }\n console.log('\\nTo fix:');\n console.log(\n '1. Remove hardcoded pnpm versions from workflows (let pnpm/action-setup auto-detect from packageManager)',\n );\n console.log(\n '2. Update Dockerfile pnpm versions to match package.json packageManager field\\n',\n );\n hasErrors = true;\n } else {\n console.log('ā
PNPM versions are consistent!\\n');\n }\n\n if (hasErrors) {\n process.exit(1);\n }\n}\n\nif (import.meta.url === new URL(process.argv[1] ?? '', 'file:').href) {\n void runValidateWorkflows();\n}\n","#!/usr/bin/env node\n\nimport { runPrPreview } from './pr-preview';\nimport { runValidateWorkflows } from './validate-workflows';\n\nfunction printHelp(): void {\n console.log(`dependency-graph <command>\n\nCommands:\n pr-preview Show affected deploys and releases for the current branch\n validate-workflows Validate workflow path filters and pnpm version consistency`);\n}\n\nasync function main(): Promise<void> {\n const command = process.argv[2];\n\n switch (command) {\n case 'pr-preview':\n await runPrPreview();\n return;\n case 'validate-workflows':\n await runValidateWorkflows();\n return;\n case '--help':\n case '-h':\n case undefined:\n printHelp();\n return;\n default:\n console.error(`Unknown dependency-graph command: ${command}`);\n printHelp();\n process.exit(1);\n }\n}\n\nvoid main();\n"],"mappings":"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","names":["readFileSync","resolve","glob","parseYaml","join","existsSync","parseYaml","existsSync","join","join","existsSync","existsSync","join","uniqueSorted","uniqueSorted","existsSync","parseYaml","resolve","readFileSync","glob","parseYaml","existsSync","readFileSync","readdirSync","join","resolve","parseYaml","join","existsSync","readdirSync","readFileSync","parseYaml","resolve"]}
|
|
1
|
+
{"version":3,"sources":["../src/cli/pr-preview.ts","../src/graph/builder.ts","../src/graph/traversal.ts","../src/workspace/discovery.ts","../src/workspace/package-map.ts","../src/workflow/policy.ts","../src/workflow/discovery.ts","../src/workflow/expected-paths.ts","../src/workflow/impact.ts","../src/workflow/validator.ts","../src/workflow/parser.ts","../src/cli/validate-workflows.ts","../src/cli/index.ts"],"sourcesContent":["#!/usr/bin/env node\n\nimport { readFileSync } from 'node:fs';\nimport { resolve } from 'node:path';\nimport { glob } from 'glob';\nimport { parse as parseYaml } from 'yaml';\n\n// Import dependency graph functions from parent module\nimport {\n discoverWorkspaces,\n buildDependencyGraph,\n findAffectedPackages,\n getWorkflowImpacts,\n validateWorkflows,\n type WorkspacePackage,\n type WorkflowValidationResult,\n} from '../index';\n\ninterface ChangedFile {\n path: string;\n status: 'modified' | 'added' | 'deleted';\n}\n\ninterface DeploymentIndicator {\n file?: string;\n field?: string;\n dependency?: string;\n}\n\ninterface PlatformDetection {\n platform: string;\n indicators: DeploymentIndicator[];\n}\n\n// Convention-based deployment detection (order matters - most specific first)\nconst PLATFORM_INDICATORS: PlatformDetection[] = [\n {\n platform: 'cloudflare-workers',\n indicators: [{ file: 'wrangler.toml' }],\n },\n {\n platform: 'expo',\n indicators: [{ file: 'app.json' }, { file: 'eas.json' }],\n },\n {\n platform: 'npm-package',\n indicators: [{ field: 'bin' }],\n },\n {\n platform: 'electron',\n indicators: [\n { dependency: 'electron' },\n { dependency: 'electron-builder' },\n ],\n },\n {\n platform: 'next.js',\n indicators: [\n { file: 'next.config.js' },\n { file: 'next.config.mjs' },\n { file: 'next.config.ts' },\n { dependency: 'next' },\n ],\n },\n {\n platform: 'node.js',\n indicators: [\n { field: 'start' }, // has start script\n ],\n },\n];\n\nconst DEFAULT_APP_DIRECTORIES = ['apps'];\n\nexport function formatWorkflowPathDrift(\n validationResults: WorkflowValidationResult[],\n): string[] {\n const invalidResults = validationResults.filter((result) => !result.valid);\n if (invalidResults.length === 0) {\n return [];\n }\n\n const lines = [\n 'ā ļø Workflow path drift (advisory - does not affect impact above):',\n ];\n\n for (const result of invalidResults) {\n lines.push(` ${result.workflow} (${result.targetPackage}):`);\n\n for (const issue of result.issues) {\n if (issue.kind === 'missing') {\n lines.push(` + ${issue.message}`);\n } else if (issue.kind === 'unnecessary') {\n lines.push(` - ${issue.message}`);\n } else {\n lines.push(` ! ${issue.message}`);\n }\n }\n }\n\n return lines;\n}\n\nfunction isDeployableApp(pkg: WorkspacePackage): {\n deployable: boolean;\n platform?: string;\n} {\n // Check if package is in apps directory (configurable in future)\n const isInAppDirectory = DEFAULT_APP_DIRECTORIES.some(\n (dir) => pkg.path.includes(`/${dir}/`) || pkg.path.endsWith(`/${dir}`),\n );\n\n if (!isInAppDirectory) {\n return { deployable: false };\n }\n\n // Detect platform based on indicators\n for (const platformDetection of PLATFORM_INDICATORS) {\n const hasIndicators = platformDetection.indicators.some((indicator) => {\n if (indicator.file) {\n try {\n const filePath = resolve(pkg.path, indicator.file);\n readFileSync(filePath, 'utf-8');\n return true;\n } catch {\n return false;\n }\n }\n\n if (indicator.field) {\n const scripts = pkg.packageJson.scripts as\n | Record<string, string>\n | undefined;\n if (indicator.field === 'start' && scripts?.start) {\n return true;\n }\n const packageJsonField =\n pkg.packageJson[indicator.field as keyof typeof pkg.packageJson];\n if (indicator.field !== 'start' && packageJsonField) {\n return true;\n }\n }\n\n if (indicator.dependency) {\n return !!(\n pkg.dependencies[indicator.dependency] ||\n pkg.devDependencies[indicator.dependency]\n );\n }\n\n return false;\n });\n\n if (hasIndicators) {\n return { deployable: true, platform: platformDetection.platform };\n }\n }\n\n // If in apps directory but no specific platform detected, not deployable\n return { deployable: false };\n}\n\nasync function getChangedFiles(): Promise<ChangedFile[]> {\n const { execSync } = await import('node:child_process');\n\n try {\n // Try local production first, then fall back to origin/production\n let baseBranch = 'production';\n try {\n execSync('git rev-parse production', {\n encoding: 'utf-8',\n stdio: 'pipe',\n });\n } catch {\n baseBranch = 'origin/production';\n }\n\n const output = execSync(`git diff --name-status ${baseBranch}...HEAD`, {\n encoding: 'utf-8',\n });\n\n return output\n .trim()\n .split('\\n')\n .filter((line) => line)\n .map((line) => {\n const [status, path] = line.split('\\t');\n\n return {\n path: path ?? '',\n status:\n status === 'D' ? 'deleted' : status === 'A' ? 'added' : 'modified',\n };\n })\n .filter((file): file is ChangedFile => file.path !== '');\n } catch {\n console.error(\n 'Failed to get changed files. Ensure you are on a branch with commits compared to production.',\n );\n process.exit(1);\n }\n}\n\nexport async function runPrPreview(): Promise<void> {\n try {\n const rootDir = resolve(process.cwd());\n const changedFiles = await getChangedFiles();\n\n console.log(\n `\\nš¦ Release Preview - Changed Files: ${changedFiles.length}\\n`,\n );\n\n changedFiles.forEach((file) => {\n console.log(` ${file.status === 'deleted' ? 'ā' : 'š'} ${file.path}`);\n });\n\n const packages = await discoverWorkspaces(rootDir, {\n fs: {\n readFile: (path) => {\n try {\n return Promise.resolve(readFileSync(path, 'utf-8'));\n } catch {\n return Promise.reject(new Error(`Failed to read file: ${path}`));\n }\n },\n exists: (path) => {\n try {\n readFileSync(path, 'utf-8');\n return Promise.resolve(true);\n } catch {\n return Promise.resolve(false);\n }\n },\n },\n glob: {\n glob: async (pattern, options) => glob(pattern, options),\n },\n yaml: {\n parse: (content) => parseYaml(content) as Record<string, unknown>,\n },\n });\n\n const graph = buildDependencyGraph(packages);\n const changedPackages = new Set<string>();\n\n for (const file of changedFiles) {\n if (file.status === 'deleted') continue;\n\n // Convert absolute package paths to relative for comparison\n const pkg = packages.find((p) => {\n const relativePkgPath = p.path.replace(rootDir + '/', '');\n return file.path.startsWith(relativePkgPath);\n });\n\n if (pkg) {\n changedPackages.add(pkg.name);\n }\n }\n\n const workflowImpacts = getWorkflowImpacts(\n rootDir,\n packages,\n changedFiles.map((file) => file.path),\n );\n const workflowDriftLines = formatWorkflowPathDrift(\n await validateWorkflows(rootDir),\n );\n\n const workflowAffectedApps = new Set(\n workflowImpacts.map(\n (workflowImpact) =>\n workflowImpact.targetPackage ?? workflowImpact.targetSlug,\n ),\n );\n\n console.log(\n `\\nš¦ Changed packages: ${Array.from(changedPackages).join(', ') || 'none'}\\n`,\n );\n\n if (changedPackages.size === 0 && workflowAffectedApps.size === 0) {\n console.log(\n '\\n⨠No packages or workflow filters changed - no deployments needed\\n',\n );\n if (workflowDriftLines.length > 0) {\n console.log(workflowDriftLines.join('\\n'));\n console.log('');\n }\n return;\n }\n\n const affected = findAffectedPackages(changedPackages, graph, {\n direction: 'upstream',\n respectAffectsUpstream: true,\n });\n\n // Find all deployable apps (affected and unaffected)\n const allDeployableApps = packages\n .map((pkg) => {\n const detection = isDeployableApp(pkg);\n return detection.deployable\n ? { name: pkg.name, pkg, platform: detection.platform }\n : null;\n })\n .filter(\n (\n item,\n ): item is {\n name: string;\n pkg: WorkspacePackage;\n platform: string | undefined;\n } => item !== null,\n );\n\n // Categorize apps by deployment type\n const affectedApps = allDeployableApps.filter(\n (app): app is NonNullable<typeof app> =>\n affected.has(app.name) || workflowAffectedApps.has(app.name),\n );\n const unaffectedApps = allDeployableApps.filter(\n (app): app is NonNullable<typeof app> =>\n !affected.has(app.name) && !workflowAffectedApps.has(app.name),\n );\n\n // Separate deploys (web-based) vs releases (installed)\n const getAppCategory = (platform?: string) => {\n switch (platform) {\n case 'next.js':\n case 'cloudflare-workers':\n case 'node.js':\n return 'deploy';\n case 'expo':\n case 'electron':\n case 'npm-package':\n return 'release';\n default:\n return 'deploy'; // default to deploy for generic\n }\n };\n\n const getAppIcon = (platform?: string) => {\n switch (platform) {\n case 'next.js':\n case 'cloudflare-workers':\n case 'node.js':\n return 'š';\n case 'expo':\n return 'š±';\n case 'electron':\n return 'š„ļø';\n case 'npm-package':\n return 'ā”';\n default:\n return 'š';\n }\n };\n\n const affectedDeploys = affectedApps.filter(\n (app): app is NonNullable<typeof app> =>\n getAppCategory(app.platform) === 'deploy',\n );\n const affectedReleases = affectedApps.filter(\n (app): app is NonNullable<typeof app> =>\n getAppCategory(app.platform) === 'release',\n );\n\n console.log(`\\nš PR Preview\\n`);\n\n if (affectedApps.length === 0) {\n console.log('No apps affected by changes.\\n');\n } else {\n if (affectedDeploys.length > 0) {\n console.log('š Apps that will be DEPLOYED:');\n for (const item of affectedDeploys) {\n const platformDisplay =\n item.platform === 'generic' ? '' : ` (${item.platform})`;\n const icon = getAppIcon(item.platform);\n console.log(`${icon} ${item.name}${platformDisplay}`);\n }\n console.log('');\n }\n\n if (affectedReleases.length > 0) {\n console.log('š Apps that will be RELEASED:');\n for (const item of affectedReleases) {\n const platformDisplay =\n item.platform === 'generic' ? '' : ` (${item.platform})`;\n const icon = getAppIcon(item.platform);\n console.log(`${icon} ${item.name}${platformDisplay}`);\n }\n console.log('');\n }\n }\n\n if (unaffectedApps.length > 0) {\n console.log(`š Apps that will NOT be affected:`);\n for (const item of unaffectedApps) {\n const platformDisplay =\n item.platform === 'generic' ? '' : ` (${item.platform})`;\n console.log(`āļø ${item.name}${platformDisplay}`);\n }\n console.log('');\n }\n\n console.log(`Legend:`);\n console.log(`š = Deploy (web-based, instant updates)`);\n console.log(`š± = Release (mobile app, user installs)`);\n console.log(`š„ļø = Release (desktop app, user installs)`);\n console.log(`ā” = Release (CLI tool, user installs)`);\n console.log(`āļø = Unaffected (no changes needed)`);\n console.log(`š = File changed`);\n console.log(`ā = File deleted\\n`);\n\n if (workflowDriftLines.length > 0) {\n console.log(workflowDriftLines.join('\\n'));\n console.log('');\n }\n } catch (error) {\n console.error(\n 'Error:',\n error instanceof Error ? error.message : String(error),\n );\n process.exit(1);\n }\n}\n\nif (import.meta.url === new URL(process.argv[1] ?? '', 'file:').href) {\n void runPrPreview();\n}\n","import type { WorkspacePackage, DependencyGraph } from './types';\n\nexport function buildDependencyGraph(\n packages: WorkspacePackage[],\n): DependencyGraph {\n const graph: DependencyGraph = {\n packages: new Map(),\n dependsOn: new Map(),\n dependedBy: new Map(),\n };\n\n for (const pkg of packages) {\n graph.packages.set(pkg.name, pkg);\n graph.dependsOn.set(pkg.name, new Set());\n graph.dependedBy.set(pkg.name, new Set());\n }\n\n for (const pkg of packages) {\n const allDeps = {\n ...pkg.dependencies,\n ...pkg.devDependencies,\n };\n\n for (const depName of Object.keys(allDeps)) {\n const matchedPkg = findMatchingPackage(depName, packages);\n\n if (matchedPkg) {\n graph.dependsOn.get(pkg.name)!.add(matchedPkg.name);\n graph.dependedBy.get(matchedPkg.name)!.add(pkg.name);\n }\n }\n }\n\n return graph;\n}\n\nfunction findMatchingPackage(\n depName: string,\n packages: WorkspacePackage[],\n): WorkspacePackage | undefined {\n let match = packages.find((p) => p.name === depName);\n if (match) return match;\n\n match = packages.find((p) => p.name === `@repo/${depName}`);\n if (match) return match;\n\n const nameWithoutRepo = depName.replace(/^@repo\\//, '');\n match = packages.find((p) => p.name === nameWithoutRepo);\n if (match) return match;\n\n return undefined;\n}\n","import type { DependencyGraph, TraversalOptions } from './types';\n\nexport function findAffectedPackages(\n startingPackages: Set<string>,\n graph: DependencyGraph,\n options: TraversalOptions = {},\n): Set<string> {\n const {\n direction = 'upstream',\n maxDepth = Infinity,\n filter,\n respectAffectsUpstream = false,\n } = options;\n\n const affected = new Set<string>(startingPackages);\n const queue: Array<{ name: string; depth: number }> = Array.from(\n startingPackages,\n ).map((name) => ({ name, depth: 0 }));\n const visited = new Set<string>();\n\n while (queue.length > 0) {\n const current = queue.shift()!;\n\n if (visited.has(current.name)) continue;\n visited.add(current.name);\n\n if (current.depth >= maxDepth) {\n continue;\n }\n\n const pkg = graph.packages.get(current.name);\n if (!pkg) continue;\n\n if (respectAffectsUpstream && direction === 'upstream') {\n const release = pkg.packageJson.release;\n if (release && release.affectsUpstream === false) {\n continue;\n }\n }\n\n const nextPackages = new Set<string>();\n\n if (direction === 'upstream' || direction === 'both') {\n const upstream = graph.dependedBy.get(current.name) || new Set();\n upstream.forEach((p) => nextPackages.add(p));\n }\n\n if (direction === 'downstream' || direction === 'both') {\n const downstream = graph.dependsOn.get(current.name) || new Set();\n downstream.forEach((p) => nextPackages.add(p));\n }\n\n for (const pkgName of nextPackages) {\n const nextPkg = graph.packages.get(pkgName);\n\n if (filter && nextPkg && !filter(nextPkg)) {\n continue;\n }\n\n if (!affected.has(pkgName)) {\n affected.add(pkgName);\n queue.push({ name: pkgName, depth: current.depth + 1 });\n }\n }\n }\n\n return affected;\n}\n\nexport function findDependencyPath(\n from: string,\n to: string,\n graph: DependencyGraph,\n): string[] | null {\n const queue: string[][] = [[from]];\n const visited = new Set<string>([from]);\n\n while (queue.length > 0) {\n const path = queue.shift()!;\n const current = path[path.length - 1];\n\n if (!current) {\n continue;\n }\n\n if (current === to) {\n return path;\n }\n\n const dependents = graph.dependedBy.get(current) || new Set();\n for (const dependent of dependents) {\n if (!visited.has(dependent)) {\n visited.add(dependent);\n queue.push([...path, dependent]);\n }\n }\n }\n\n return null;\n}\n\nexport function findAllPaths(\n from: string,\n to: string,\n graph: DependencyGraph,\n maxPaths = 10,\n): string[][] {\n const paths: string[][] = [];\n const visited = new Set<string>();\n\n function dfs(current: string, path: string[]): void {\n if (paths.length >= maxPaths) return;\n\n if (current === to) {\n paths.push([...path]);\n return;\n }\n\n if (visited.has(current)) return;\n visited.add(current);\n\n const dependents = graph.dependedBy.get(current) || new Set();\n for (const dependent of dependents) {\n dfs(dependent, [...path, dependent]);\n }\n\n visited.delete(current);\n }\n\n dfs(from, [from]);\n return paths;\n}\n","import { join, resolve } from 'node:path';\nimport type { WorkspacePackage } from '../graph/types';\nimport type {\n FileSystemClient,\n GlobClient,\n YamlClient,\n} from '../types/clients';\n\nexport interface WorkspaceConfig {\n packages: string[];\n}\n\nexport interface WorkspaceDiscoveryConfig {\n fs: FileSystemClient;\n glob: GlobClient;\n yaml: YamlClient;\n}\n\nexport async function discoverWorkspaces(\n rootDir: string,\n config: WorkspaceDiscoveryConfig,\n): Promise<WorkspacePackage[]> {\n const workspaceConfig = await loadWorkspaceConfig(rootDir, config);\n const packages: WorkspacePackage[] = [];\n\n for (const pattern of workspaceConfig.packages) {\n if (pattern.startsWith('!')) continue;\n\n const pkgDirs = await findPackageDirectories(rootDir, pattern, config);\n\n for (const pkgDir of pkgDirs) {\n const pkgJsonPath = join(pkgDir, 'package.json');\n\n try {\n const pkgJsonContent: string = await config.fs.readFile(\n pkgJsonPath,\n 'utf-8',\n );\n const pkgJson = JSON.parse(pkgJsonContent) as {\n name: string;\n version?: string;\n dependencies?: Record<string, string>;\n devDependencies?: Record<string, string>;\n [key: string]: unknown;\n };\n\n packages.push({\n name: pkgJson.name,\n version: pkgJson.version || '0.0.0',\n path: pkgDir,\n packageJson: pkgJson,\n dependencies: pkgJson.dependencies || {},\n devDependencies: pkgJson.devDependencies || {},\n });\n } catch {\n continue;\n }\n }\n }\n\n return packages;\n}\n\nasync function loadWorkspaceConfig(\n rootDir: string,\n config: WorkspaceDiscoveryConfig,\n): Promise<WorkspaceConfig> {\n const workspaceFilePath = join(rootDir, 'pnpm-workspace.yaml');\n\n try {\n const content: string = await config.fs.readFile(\n workspaceFilePath,\n 'utf-8',\n );\n const parsed = config.yaml.parse(content) as WorkspaceConfig;\n return parsed;\n } catch {\n return { packages: [] };\n }\n}\n\nasync function findPackageDirectories(\n rootDir: string,\n pattern: string,\n config: WorkspaceDiscoveryConfig,\n): Promise<string[]> {\n const matches: string[] = await config.glob.glob(pattern, {\n cwd: rootDir,\n absolute: false,\n ignore: ['**/node_modules/**', '**/.next/**', '**/dist/**'],\n });\n\n return matches.map((match: string) => resolve(rootDir, match));\n}\n","import { existsSync } from 'node:fs';\nimport { join, relative, sep } from 'node:path';\nimport type { WorkspacePackage } from '../graph/types';\n\nexport interface MappedWorkspacePath {\n filesystemPath: string;\n workflowPath: string | null;\n}\n\nexport interface WorkspacePackageMap {\n packageMap: Map<string, MappedWorkspacePath>;\n workspaceRoots: Set<string>;\n}\n\nexport function normalizeRelativePath(path: string): string {\n return path.split(sep).join('/');\n}\n\nfunction resolveWorkflowPath(\n relativePath: string,\n rootDir: string,\n): string | null {\n if (!relativePath.startsWith('../')) {\n return relativePath;\n }\n\n const segments = relativePath.split('/');\n\n for (let index = 0; index < segments.length; index += 1) {\n const candidateSegments = segments.slice(index);\n if (candidateSegments.length === 0 || candidateSegments[0] === '..') {\n continue;\n }\n\n const candidatePath = candidateSegments.join('/');\n if (existsSync(join(rootDir, candidatePath))) {\n return candidatePath;\n }\n }\n\n return null;\n}\n\nexport function buildPackageMap(\n packages: WorkspacePackage[],\n rootDir: string,\n): WorkspacePackageMap {\n const packageMap = new Map<string, MappedWorkspacePath>();\n const workspaceRoots = new Set<string>();\n\n for (const pkg of packages) {\n const relativePath = normalizeRelativePath(relative(rootDir, pkg.path));\n const workflowPath = resolveWorkflowPath(relativePath, rootDir);\n\n packageMap.set(pkg.name, {\n filesystemPath: relativePath,\n workflowPath,\n });\n\n if (!workflowPath) {\n continue;\n }\n\n const [workspaceRoot] = workflowPath.split('/');\n if (workspaceRoot) {\n workspaceRoots.add(workspaceRoot);\n }\n }\n\n return {\n packageMap,\n workspaceRoots,\n };\n}\n","import type { WorkflowValidationPolicy } from './types';\n\nexport const defaultWorkflowValidationPolicy: WorkflowValidationPolicy = {\n workflowFilePatterns: ['deploy-*.yml', 'release-*.yml'],\n allowedRootPaths: [\n 'package.json',\n 'pnpm-lock.yaml',\n 'pnpm-workspace.yaml',\n 'turbo.json',\n ],\n includeDevDependenciesForRootPackage: true,\n includeDevDependenciesTransitively: false,\n};\n","import { existsSync, readdirSync } from 'node:fs';\nimport { join } from 'node:path';\nimport type { WorkspacePackage } from '../graph/types';\nimport { buildPackageMap } from '../workspace/package-map';\nimport type { WorkflowTarget, WorkflowValidationPolicy } from './types';\n\nfunction patternToRegExp(pattern: string): RegExp {\n const escaped = pattern.replace(/[.+?^${}()|[\\]\\\\]/g, '\\\\$&');\n return new RegExp(`^${escaped.replace(/\\*/g, '.*')}$`);\n}\n\nexport function discoverWorkflowTargets(\n rootDir: string,\n packages: WorkspacePackage[],\n policy: WorkflowValidationPolicy,\n): WorkflowTarget[] {\n const workflowsDir = join(rootDir, '.github/workflows');\n if (!existsSync(workflowsDir)) {\n return [];\n }\n\n const { packageMap } = buildPackageMap(packages, rootDir);\n const appPackages = packages.filter((pkg) =>\n packageMap.get(pkg.name)?.workflowPath?.startsWith('apps/'),\n );\n const bySlug = new Map<string, string>();\n\n for (const pkg of appPackages) {\n const relativePath = packageMap.get(pkg.name)?.workflowPath;\n if (!relativePath) {\n continue;\n }\n\n const slug = relativePath.split('/').at(-1);\n if (slug) {\n bySlug.set(slug, pkg.name);\n }\n }\n\n const allowedPatterns = policy.workflowFilePatterns.map(patternToRegExp);\n const files = readdirSync(workflowsDir)\n .filter((file) => file.endsWith('.yml'))\n .filter((file) => allowedPatterns.some((pattern) => pattern.test(file)));\n\n return files\n .map((workflowFile) => {\n const match = /^(?:deploy|release)-(.+)\\.yml$/.exec(workflowFile);\n const targetSlug = match?.[1] ?? workflowFile.replace(/\\.yml$/, '');\n\n return {\n workflowFile,\n workflowPath: `.github/workflows/${workflowFile}`,\n targetSlug,\n targetPackage: bySlug.get(targetSlug) ?? null,\n };\n })\n .sort((a, b) => a.workflowFile.localeCompare(b.workflowFile));\n}\n","import type { WorkspacePackage } from '../graph/types';\nimport type { WorkflowTarget, WorkflowValidationPolicy } from './types';\n\ninterface ExpectedPathsOptions {\n workflowTarget: WorkflowTarget;\n packages: WorkspacePackage[];\n packageMap: Map<string, { workflowPath: string | null }>;\n policy: WorkflowValidationPolicy;\n}\n\nfunction uniqueSorted(values: string[]): string[] {\n return Array.from(new Set(values)).sort();\n}\n\nfunction resolveWorkspaceDependency(\n dependencyName: string,\n packages: WorkspacePackage[],\n): WorkspacePackage | undefined {\n let match = packages.find((pkg) => pkg.name === dependencyName);\n if (match) {\n return match;\n }\n\n match = packages.find((pkg) => pkg.name === `@repo/${dependencyName}`);\n if (match) {\n return match;\n }\n\n const nameWithoutRepo = dependencyName.replace(/^@repo\\//, '');\n return packages.find((pkg) => pkg.name === nameWithoutRepo);\n}\n\nfunction collectWorkspaceDependencyNames(\n pkg: WorkspacePackage,\n packages: WorkspacePackage[],\n visited: Set<string>,\n includeDevDependencies: boolean,\n includeDevDependenciesTransitively: boolean,\n): Set<string> {\n const collected = new Set<string>();\n const dependencyEntries = Object.entries(pkg.dependencies);\n const devDependencyEntries = includeDevDependencies\n ? Object.entries(pkg.devDependencies)\n : [];\n\n for (const [dependencyName] of [\n ...dependencyEntries,\n ...devDependencyEntries,\n ]) {\n const dependency = resolveWorkspaceDependency(dependencyName, packages);\n if (!dependency || visited.has(dependency.name)) {\n continue;\n }\n\n visited.add(dependency.name);\n collected.add(dependency.name);\n\n const nestedDependencies = collectWorkspaceDependencyNames(\n dependency,\n packages,\n visited,\n includeDevDependenciesTransitively,\n includeDevDependenciesTransitively,\n );\n\n for (const nestedDependency of nestedDependencies) {\n collected.add(nestedDependency);\n }\n }\n\n return collected;\n}\n\nexport function getExpectedWorkflowPaths({\n workflowTarget,\n packages,\n packageMap,\n policy,\n}: ExpectedPathsOptions): string[] {\n const targetPackageName = workflowTarget.targetPackage;\n if (!targetPackageName) {\n return [];\n }\n\n const targetPackage = packages.find((pkg) => pkg.name === targetPackageName);\n if (!targetPackage) {\n return [];\n }\n\n const dependencyNames = collectWorkspaceDependencyNames(\n targetPackage,\n packages,\n new Set<string>(),\n policy.includeDevDependenciesForRootPackage,\n policy.includeDevDependenciesTransitively,\n );\n\n const expectedPaths: string[] = [];\n const targetPackagePath = packageMap.get(targetPackageName)?.workflowPath;\n if (targetPackagePath) {\n expectedPaths.push(`${targetPackagePath}/**`);\n }\n\n for (const dependencyName of dependencyNames) {\n const dependencyPath = packageMap.get(dependencyName)?.workflowPath;\n if (dependencyPath) {\n expectedPaths.push(`${dependencyPath}/**`);\n }\n }\n\n return uniqueSorted(expectedPaths);\n}\n","import type { WorkspacePackage } from '../graph/types';\nimport { buildPackageMap } from '../workspace/package-map';\nimport { discoverWorkflowTargets } from './discovery';\nimport { getExpectedWorkflowPaths } from './expected-paths';\nimport { defaultWorkflowValidationPolicy } from './policy';\nimport type { WorkflowImpact, WorkflowValidationPolicy } from './types';\n\nfunction uniqueSorted(values: string[]): string[] {\n return Array.from(new Set(values)).sort();\n}\n\nfunction mergeWorkflowPolicy(\n policyOverrides?: Partial<WorkflowValidationPolicy>,\n): WorkflowValidationPolicy {\n return {\n ...defaultWorkflowValidationPolicy,\n ...policyOverrides,\n allowedRootPaths: uniqueSorted([\n ...defaultWorkflowValidationPolicy.allowedRootPaths,\n ...(policyOverrides?.allowedRootPaths ?? []),\n ]),\n };\n}\n\n// Path matching intentionally only supports exact paths and /** prefix matching\n// because that mirrors the workflow filters currently used in these repos.\nexport function matchesWorkflowPathFilter(\n changedPath: string,\n workflowPathFilter: string,\n): boolean {\n if (workflowPathFilter === changedPath) {\n return true;\n }\n\n if (!workflowPathFilter.endsWith('/**')) {\n return false;\n }\n\n const workflowPrefix = workflowPathFilter.slice(0, -3);\n return changedPath.startsWith(`${workflowPrefix}/`);\n}\n\nexport function getWorkflowImpacts(\n rootDir: string,\n packages: WorkspacePackage[],\n changedPaths: string[],\n policyOverrides?: Partial<WorkflowValidationPolicy>,\n): WorkflowImpact[] {\n const policy = mergeWorkflowPolicy(policyOverrides);\n const workflowTargets = discoverWorkflowTargets(rootDir, packages, policy);\n const { packageMap } = buildPackageMap(packages, rootDir);\n\n return workflowTargets\n .map((workflowTarget) => {\n const calculatedPaths = uniqueSorted([\n ...getExpectedWorkflowPaths({\n workflowTarget,\n packages,\n packageMap,\n policy,\n }),\n ...policy.allowedRootPaths,\n workflowTarget.workflowPath,\n ]);\n const matchedPaths = calculatedPaths.filter((calculatedPath) =>\n changedPaths.some((changedPath) =>\n matchesWorkflowPathFilter(changedPath, calculatedPath),\n ),\n );\n\n return {\n ...workflowTarget,\n matchedPaths: uniqueSorted(matchedPaths),\n } satisfies WorkflowImpact;\n })\n .filter((workflowImpact) => workflowImpact.matchedPaths.length > 0);\n}\n","import { existsSync } from 'node:fs';\nimport { readFile } from 'node:fs/promises';\nimport { join } from 'node:path';\nimport { glob } from 'glob';\nimport { parse as parseYaml } from 'yaml';\nimport { discoverWorkspaces } from '../workspace/discovery';\nimport { buildPackageMap } from '../workspace/package-map';\nimport type {\n WorkflowValidationIssue,\n WorkflowValidationPolicy,\n WorkflowValidationResult,\n} from './types';\nimport { discoverWorkflowTargets } from './discovery';\nimport { getExpectedWorkflowPaths } from './expected-paths';\nimport { parseWorkflowFile } from './parser';\nimport { defaultWorkflowValidationPolicy } from './policy';\n\nfunction uniqueSorted(values: string[]): string[] {\n return Array.from(new Set(values)).sort();\n}\n\nfunction buildBroadWildcards(workspaceRoots: Set<string>): Set<string> {\n const wildcards = new Set<string>();\n\n for (const root of workspaceRoots) {\n wildcards.add(`${root}/*`);\n wildcards.add(`${root}/**`);\n }\n\n return wildcards;\n}\n\nfunction hasGlobSyntax(path: string): boolean {\n return /[*?[\\]{}]/.test(path);\n}\n\nfunction getStalePathCheckBase(\n path: string,\n allowedRootPaths: string[],\n workflowPath: string,\n): string | null {\n if (\n path === workflowPath ||\n allowedRootPaths.includes(path) ||\n path.startsWith('!')\n ) {\n return null;\n }\n\n const trailingGlobSuffix = path.endsWith('/**')\n ? '/**'\n : path.endsWith('/*')\n ? '/*'\n : null;\n const checkBase = trailingGlobSuffix\n ? path.slice(0, -trailingGlobSuffix.length)\n : path;\n\n if (!checkBase || hasGlobSyntax(checkBase)) {\n return null;\n }\n\n if (!trailingGlobSuffix && hasGlobSyntax(path)) {\n return null;\n }\n\n return checkBase;\n}\n\nfunction splitActualPaths(\n paths: string[],\n workspaceRoots: Set<string>,\n allowedRootPaths: string[],\n workflowPath: string,\n): { workspacePaths: string[]; ignoredPaths: string[] } {\n const workspacePaths: string[] = [];\n const ignoredPaths: string[] = [];\n\n for (const path of paths) {\n if (path === workflowPath || allowedRootPaths.includes(path)) {\n ignoredPaths.push(path);\n continue;\n }\n\n const [rootSegment] = path.split('/');\n if (rootSegment && workspaceRoots.has(rootSegment)) {\n workspacePaths.push(path);\n continue;\n }\n\n ignoredPaths.push(path);\n }\n\n return {\n workspacePaths: uniqueSorted(workspacePaths),\n ignoredPaths: uniqueSorted(ignoredPaths),\n };\n}\n\nfunction isCoveredByExpectedPath(\n actualPath: string,\n expectedPaths: string[],\n): boolean {\n return expectedPaths.some((expectedPath) => {\n if (expectedPath === actualPath) {\n return true;\n }\n\n if (!expectedPath.endsWith('/**')) {\n return false;\n }\n\n const expectedPrefix = expectedPath.slice(0, -3);\n return actualPath.startsWith(`${expectedPrefix}/`);\n });\n}\n\nfunction validateWorkflowResult(\n workflow: string,\n targetPackage: string,\n expectedPaths: string[],\n actualPaths: string[],\n broadWildcards: Set<string>,\n additionalIssues: WorkflowValidationIssue[] = [],\n): WorkflowValidationResult {\n const issues: WorkflowValidationIssue[] = [];\n const missing = expectedPaths.filter((path) => !actualPaths.includes(path));\n const unnecessary = actualPaths.filter(\n (path) => !isCoveredByExpectedPath(path, expectedPaths),\n );\n\n for (const path of missing) {\n issues.push({\n kind: 'missing',\n path,\n message: `Missing path '${path}'`,\n });\n }\n\n for (const path of unnecessary) {\n issues.push({\n kind: 'unnecessary',\n path,\n message: `Unnecessary path '${path}'`,\n });\n }\n\n for (const path of actualPaths) {\n if (broadWildcards.has(path)) {\n issues.push({\n kind: 'broad-wildcard',\n path,\n message: `Uses broad wildcard '${path}' which triggers on all workspace changes under that root`,\n });\n }\n }\n\n issues.push(...additionalIssues);\n\n return {\n workflow,\n targetPackage,\n valid: issues.length === 0,\n expectedPaths,\n actualPaths,\n missing,\n unnecessary,\n issues,\n };\n}\n\nfunction getStalePathIssues(\n ignoredPaths: string[],\n rootDir: string,\n allowedRootPaths: string[],\n workflowPath: string,\n): WorkflowValidationIssue[] {\n return ignoredPaths.flatMap((path) => {\n const checkBase = getStalePathCheckBase(\n path,\n allowedRootPaths,\n workflowPath,\n );\n\n if (!checkBase || existsSync(join(rootDir, checkBase))) {\n return [];\n }\n\n return [\n {\n kind: 'stale-path',\n path,\n message: `Stale path '${path}' does not exist`,\n },\n ];\n });\n}\n\nexport async function validateWorkflows(\n rootDir: string,\n policyOverrides?: Partial<WorkflowValidationPolicy>,\n): Promise<WorkflowValidationResult[]> {\n const discoveredPackages = await discoverWorkspaces(rootDir, {\n fs: {\n readFile: (path, encoding) => readFile(path, encoding),\n exists: (path) => Promise.resolve(existsSync(path)),\n },\n glob: {\n glob: (pattern, options) => glob(pattern, options),\n },\n yaml: {\n parse: (content): unknown => parseYaml(content),\n },\n });\n\n const policy: WorkflowValidationPolicy = {\n ...defaultWorkflowValidationPolicy,\n ...policyOverrides,\n allowedRootPaths: uniqueSorted([\n ...defaultWorkflowValidationPolicy.allowedRootPaths,\n ...(policyOverrides?.allowedRootPaths ?? []),\n ]),\n };\n\n const { packageMap, workspaceRoots } = buildPackageMap(\n discoveredPackages,\n rootDir,\n );\n const workflowTargets = discoverWorkflowTargets(\n rootDir,\n discoveredPackages,\n policy,\n );\n const broadWildcards = buildBroadWildcards(workspaceRoots);\n\n return workflowTargets.map((workflowTarget) => {\n if (!workflowTarget.targetPackage) {\n return {\n workflow: workflowTarget.workflowFile,\n targetPackage: workflowTarget.targetSlug,\n valid: false,\n expectedPaths: [],\n actualPaths: [],\n missing: [],\n unnecessary: [],\n issues: [\n {\n kind: 'config-error',\n message: `Could not resolve workflow target package for '${workflowTarget.workflowFile}'`,\n },\n ],\n };\n }\n\n try {\n const parsedWorkflow = parseWorkflowFile(\n workflowTarget.workflowFile,\n rootDir,\n );\n\n if (parsedWorkflow.paths.length === 0) {\n return {\n workflow: workflowTarget.workflowFile,\n targetPackage: workflowTarget.targetPackage,\n valid: true,\n expectedPaths: [],\n actualPaths: [],\n missing: [],\n unnecessary: [],\n issues: [],\n };\n }\n\n const { workspacePaths: actualPaths, ignoredPaths } = splitActualPaths(\n parsedWorkflow.paths,\n workspaceRoots,\n policy.allowedRootPaths,\n workflowTarget.workflowPath,\n );\n const expectedPaths = getExpectedWorkflowPaths({\n workflowTarget,\n packages: discoveredPackages,\n packageMap,\n policy,\n });\n\n return validateWorkflowResult(\n workflowTarget.workflowFile,\n workflowTarget.targetPackage,\n expectedPaths,\n actualPaths,\n broadWildcards,\n getStalePathIssues(\n ignoredPaths,\n rootDir,\n policy.allowedRootPaths,\n workflowTarget.workflowPath,\n ),\n );\n } catch (error) {\n return {\n workflow: workflowTarget.workflowFile,\n targetPackage: workflowTarget.targetPackage,\n valid: false,\n expectedPaths: [],\n actualPaths: [],\n missing: [],\n unnecessary: [],\n issues: [\n {\n kind: 'parse-error',\n message: error instanceof Error ? error.message : String(error),\n },\n ],\n };\n }\n });\n}\n","import { existsSync, readFileSync } from 'node:fs';\nimport { join } from 'node:path';\nimport { parse as parseYaml } from 'yaml';\nimport type { ParsedWorkflow } from './types';\n\ninterface WorkflowConfig {\n name?: string;\n on?: {\n push?: {\n paths?: string[];\n };\n pull_request?: {\n paths?: string[];\n };\n };\n}\n\nfunction uniqueSorted(values: string[]): string[] {\n return Array.from(new Set(values)).sort();\n}\n\nexport function parseWorkflowFile(\n workflowFile: string,\n rootDir: string,\n): ParsedWorkflow {\n const workflowPath = join(rootDir, '.github/workflows', workflowFile);\n\n if (!existsSync(workflowPath)) {\n throw new Error(`Workflow file not found: ${workflowFile}`);\n }\n\n const content = readFileSync(workflowPath, 'utf-8');\n const workflow = parseYaml(content) as WorkflowConfig;\n const pushPaths = workflow.on?.push?.paths ?? [];\n const pullRequestPaths = workflow.on?.pull_request?.paths ?? [];\n\n const parsedWorkflow: ParsedWorkflow = {\n pushPaths: uniqueSorted(pushPaths),\n pullRequestPaths: uniqueSorted(pullRequestPaths),\n paths: uniqueSorted([...pushPaths, ...pullRequestPaths]),\n };\n\n if (workflow.name) {\n parsedWorkflow.name = workflow.name;\n }\n\n return parsedWorkflow;\n}\n","#!/usr/bin/env node\n\nimport { existsSync, readFileSync, readdirSync } from 'node:fs';\nimport { join, resolve } from 'node:path';\nimport { parse as parseYaml } from 'yaml';\nimport { validateWorkflows } from '../workflow/validator';\nimport type { WorkflowValidationResult } from '../workflow/types';\n\ninterface PackageJson {\n packageManager?: string;\n}\n\ninterface PnpmValidationResult {\n valid: boolean;\n workflowIssues: { workflow: string; issue: string }[];\n dockerfileIssues: { dockerfile: string; issue: string }[];\n}\n\nfunction discoverPnpmWorkflows(rootDir: string): string[] {\n const workflowsDir = join(rootDir, '.github/workflows');\n if (!existsSync(workflowsDir)) {\n return [];\n }\n\n const workflows: string[] = [];\n const files = readdirSync(workflowsDir).filter((file) =>\n file.endsWith('.yml'),\n );\n\n for (const file of files) {\n const content = readFileSync(join(workflowsDir, file), 'utf-8');\n if (content.includes('pnpm/action-setup')) {\n workflows.push(file);\n }\n }\n\n return workflows;\n}\n\nfunction discoverPnpmDockerfiles(rootDir: string): string[] {\n const dockerfiles: string[] = [];\n\n for (const entry of readdirSync(rootDir)) {\n if (!entry.startsWith('Dockerfile')) {\n continue;\n }\n\n const content = readFileSync(join(rootDir, entry), 'utf-8');\n if (content.includes('pnpm@')) {\n dockerfiles.push(entry);\n }\n }\n\n const appsDir = join(rootDir, 'apps');\n if (!existsSync(appsDir)) {\n return dockerfiles;\n }\n\n for (const entry of readdirSync(appsDir, { withFileTypes: true })) {\n if (!entry.isDirectory()) {\n continue;\n }\n\n const dockerfilePath = join(appsDir, entry.name, 'Dockerfile');\n if (!existsSync(dockerfilePath)) {\n continue;\n }\n\n const content = readFileSync(dockerfilePath, 'utf-8');\n if (content.includes('pnpm@')) {\n dockerfiles.push(`apps/${entry.name}/Dockerfile`);\n }\n }\n\n return dockerfiles;\n}\n\nfunction getExpectedPnpmVersion(rootDir: string): string {\n const packageJson = JSON.parse(\n readFileSync(join(rootDir, 'package.json'), 'utf-8'),\n ) as PackageJson;\n const packageManager = packageJson.packageManager;\n\n if (!packageManager?.startsWith('pnpm@')) {\n throw new Error('packageManager field must specify pnpm version');\n }\n\n return packageManager.replace('pnpm@', '');\n}\n\nfunction checkWorkflowPnpmVersion(\n workflowFile: string,\n rootDir: string,\n): { valid: boolean; issue?: string } {\n const workflowPath = join(rootDir, '.github/workflows', workflowFile);\n if (!existsSync(workflowPath)) {\n return { valid: true };\n }\n\n const content = readFileSync(workflowPath, 'utf-8');\n const workflow = parseYaml(content) as {\n jobs?: Record<\n string,\n { steps?: Array<{ uses?: string; with?: { version?: string | number } }> }\n >;\n };\n\n for (const job of Object.values(workflow.jobs ?? {})) {\n for (const step of job.steps ?? []) {\n if (!step.uses?.startsWith('pnpm/action-setup')) {\n continue;\n }\n\n const version = step.with?.version;\n if (version !== undefined && !/^\\d+$/.test(String(version))) {\n return {\n valid: false,\n issue: `Hardcoded pnpm version '${version}' - remove 'version' key to auto-detect from packageManager`,\n };\n }\n }\n }\n\n return { valid: true };\n}\n\nfunction checkDockerfilePnpmVersion(\n dockerfile: string,\n expectedVersion: string,\n rootDir: string,\n): { valid: boolean; issue?: string } {\n const dockerfilePath = join(rootDir, dockerfile);\n if (!existsSync(dockerfilePath)) {\n return { valid: true };\n }\n\n const content = readFileSync(dockerfilePath, 'utf-8');\n const matches = content.matchAll(/npm install -g pnpm@([\\d.]+)/g);\n\n for (const match of matches) {\n if (match[1] !== expectedVersion) {\n return {\n valid: false,\n issue: `${dockerfile} uses pnpm@${match[1]} but package.json specifies pnpm@${expectedVersion}`,\n };\n }\n }\n\n return { valid: true };\n}\n\nfunction validatePnpmVersions(rootDir: string): PnpmValidationResult {\n const result: PnpmValidationResult = {\n valid: true,\n workflowIssues: [],\n dockerfileIssues: [],\n };\n\n for (const workflowFile of discoverPnpmWorkflows(rootDir)) {\n const check = checkWorkflowPnpmVersion(workflowFile, rootDir);\n if (!check.valid && check.issue) {\n result.valid = false;\n result.workflowIssues.push({\n workflow: workflowFile,\n issue: check.issue,\n });\n }\n }\n\n const expectedVersion = getExpectedPnpmVersion(rootDir);\n for (const dockerfile of discoverPnpmDockerfiles(rootDir)) {\n const check = checkDockerfilePnpmVersion(\n dockerfile,\n expectedVersion,\n rootDir,\n );\n if (!check.valid && check.issue) {\n result.valid = false;\n result.dockerfileIssues.push({ dockerfile, issue: check.issue });\n }\n }\n\n return result;\n}\n\nfunction printResult(result: WorkflowValidationResult): void {\n const icon = result.valid ? 'ā
' : 'ā';\n console.log(`\\n${icon} ${result.workflow} (${result.targetPackage})`);\n\n if (result.valid) {\n console.log(' All paths match dependencies');\n return;\n }\n\n const extraIssues = result.issues.filter(\n (issue) => issue.kind !== 'missing' && issue.kind !== 'unnecessary',\n );\n\n if (extraIssues.length > 0) {\n console.log(' Issues:');\n for (const issue of extraIssues) {\n console.log(` ā ļø ${issue.message}`);\n }\n }\n\n if (result.missing.length > 0) {\n console.log(' Missing paths:');\n for (const path of result.missing) {\n console.log(` - ${path}`);\n }\n }\n\n if (result.unnecessary.length > 0) {\n console.log(' Unnecessary paths:');\n for (const path of result.unnecessary) {\n console.log(` - ${path}`);\n }\n }\n}\n\nexport async function runValidateWorkflows(): Promise<void> {\n const rootDir = resolve(process.cwd());\n let hasErrors = false;\n\n console.log(\n 'š Validating GitHub Actions workflows against dependencies...\\n',\n );\n const results = await validateWorkflows(rootDir);\n console.log(`Found ${results.length} workflow(s) to validate\\n`);\n\n if (results.length === 0) {\n console.log('No deploy-*.yml or release-*.yml workflows found.\\n');\n }\n\n for (const result of results) {\n printResult(result);\n }\n\n const validCount = results.filter((result) => result.valid).length;\n const invalidCount = results.length - validCount;\n\n console.log('\\n' + '='.repeat(60));\n console.log(\n `\\nš Path validation: ${validCount} valid, ${invalidCount} invalid\\n`,\n );\n\n if (invalidCount > 0) {\n console.log('ā Some workflows need updates to match dependencies');\n console.log('\\nTo fix:');\n console.log('1. Update workflow path filters to match missing paths');\n console.log('2. Remove unnecessary paths');\n console.log('3. Replace broad workspace wildcards with specific paths\\n');\n hasErrors = true;\n } else if (results.length > 0) {\n console.log('ā
All workflows match their dependencies!\\n');\n }\n\n console.log('='.repeat(60));\n console.log('\\nš Validating pnpm version consistency...\\n');\n\n const pnpmResult = validatePnpmVersions(rootDir);\n if (!pnpmResult.valid) {\n console.log('ā PNPM version issues found:\\n');\n for (const { workflow, issue } of pnpmResult.workflowIssues) {\n console.log(` ā ļø ${workflow}: ${issue}`);\n }\n for (const { issue } of pnpmResult.dockerfileIssues) {\n console.log(` ā ļø ${issue}`);\n }\n console.log('\\nTo fix:');\n console.log(\n '1. Remove hardcoded pnpm versions from workflows (let pnpm/action-setup auto-detect from packageManager)',\n );\n console.log(\n '2. Update Dockerfile pnpm versions to match package.json packageManager field\\n',\n );\n hasErrors = true;\n } else {\n console.log('ā
PNPM versions are consistent!\\n');\n }\n\n if (hasErrors) {\n process.exit(1);\n }\n}\n\nif (import.meta.url === new URL(process.argv[1] ?? '', 'file:').href) {\n void runValidateWorkflows();\n}\n","#!/usr/bin/env node\n\nimport { runPrPreview } from './pr-preview';\nimport { runValidateWorkflows } from './validate-workflows';\n\nfunction printHelp(): void {\n console.log(`dependency-graph <command>\n\nCommands:\n pr-preview Show affected deploys and releases for the current branch\n validate-workflows Validate workflow path filters and pnpm version consistency`);\n}\n\nasync function main(): Promise<void> {\n const command = process.argv[2];\n\n switch (command) {\n case 'pr-preview':\n await runPrPreview();\n return;\n case 'validate-workflows':\n await runValidateWorkflows();\n return;\n case '--help':\n case '-h':\n case undefined:\n printHelp();\n return;\n default:\n console.error(`Unknown dependency-graph command: ${command}`);\n printHelp();\n process.exit(1);\n }\n}\n\nvoid main();\n"],"mappings":"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","names":["readFileSync","resolve","glob","parseYaml","join","existsSync","join","join","existsSync","uniqueSorted","existsSync","join","parseYaml","existsSync","join","uniqueSorted","uniqueSorted","existsSync","join","parseYaml","resolve","readFileSync","glob","parseYaml","existsSync","readFileSync","readdirSync","join","resolve","parseYaml","join","existsSync","readdirSync","readFileSync","parseYaml","resolve"]}
|
package/dist/index.d.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
export type { WorkspacePackage, DependencyGraph, TraversalOptions, GraphStats, PackageJson, } from './graph/types';
|
|
2
2
|
export type { WorkspaceConfig, WorkspaceDiscoveryConfig, } from './workspace/discovery';
|
|
3
3
|
export type { FileSystemClient, GlobClient, YamlClient } from './types/clients';
|
|
4
|
-
export type { WorkflowValidationIssue, WorkflowValidationPolicy, WorkflowValidationResult, } from './workflow/types';
|
|
4
|
+
export type { WorkflowImpact, WorkflowValidationIssue, WorkflowValidationPolicy, WorkflowValidationResult, } from './workflow/types';
|
|
5
5
|
export { buildDependencyGraph } from './graph/builder';
|
|
6
6
|
export { findAffectedPackages, findDependencyPath, findAllPaths, } from './graph/traversal';
|
|
7
7
|
export { analyzeGraph, detectCycles, getTransitiveDependencies, getTransitiveDependents, } from './graph/analysis';
|
|
@@ -9,5 +9,6 @@ export { discoverWorkspaces } from './workspace/discovery';
|
|
|
9
9
|
export { buildPackageMap, normalizeRelativePath, } from './workspace/package-map';
|
|
10
10
|
export { findPackageForFile, mapFilesToPackages, } from './workspace/file-mapping';
|
|
11
11
|
export { defaultWorkflowValidationPolicy } from './workflow/policy';
|
|
12
|
+
export { getWorkflowImpacts, matchesWorkflowPathFilter, } from './workflow/impact';
|
|
12
13
|
export { validateWorkflows } from './workflow/validator';
|
|
13
14
|
//# sourceMappingURL=index.d.ts.map
|
package/dist/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,YAAY,EACV,gBAAgB,EAChB,eAAe,EACf,gBAAgB,EAChB,UAAU,EACV,WAAW,GACZ,MAAM,eAAe,CAAC;AAEvB,YAAY,EACV,eAAe,EACf,wBAAwB,GACzB,MAAM,uBAAuB,CAAC;AAE/B,YAAY,EAAE,gBAAgB,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAEhF,YAAY,EACV,uBAAuB,EACvB,wBAAwB,EACxB,wBAAwB,GACzB,MAAM,kBAAkB,CAAC;AAE1B,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AAEvD,OAAO,EACL,oBAAoB,EACpB,kBAAkB,EAClB,YAAY,GACb,MAAM,mBAAmB,CAAC;AAE3B,OAAO,EACL,YAAY,EACZ,YAAY,EACZ,yBAAyB,EACzB,uBAAuB,GACxB,MAAM,kBAAkB,CAAC;AAE1B,OAAO,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AAE3D,OAAO,EACL,eAAe,EACf,qBAAqB,GACtB,MAAM,yBAAyB,CAAC;AAEjC,OAAO,EACL,kBAAkB,EAClB,kBAAkB,GACnB,MAAM,0BAA0B,CAAC;AAElC,OAAO,EAAE,+BAA+B,EAAE,MAAM,mBAAmB,CAAC;AAEpE,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC"}
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,YAAY,EACV,gBAAgB,EAChB,eAAe,EACf,gBAAgB,EAChB,UAAU,EACV,WAAW,GACZ,MAAM,eAAe,CAAC;AAEvB,YAAY,EACV,eAAe,EACf,wBAAwB,GACzB,MAAM,uBAAuB,CAAC;AAE/B,YAAY,EAAE,gBAAgB,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAEhF,YAAY,EACV,cAAc,EACd,uBAAuB,EACvB,wBAAwB,EACxB,wBAAwB,GACzB,MAAM,kBAAkB,CAAC;AAE1B,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AAEvD,OAAO,EACL,oBAAoB,EACpB,kBAAkB,EAClB,YAAY,GACb,MAAM,mBAAmB,CAAC;AAE3B,OAAO,EACL,YAAY,EACZ,YAAY,EACZ,yBAAyB,EACzB,uBAAuB,GACxB,MAAM,kBAAkB,CAAC;AAE1B,OAAO,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AAE3D,OAAO,EACL,eAAe,EACf,qBAAqB,GACtB,MAAM,yBAAyB,CAAC;AAEjC,OAAO,EACL,kBAAkB,EAClB,kBAAkB,GACnB,MAAM,0BAA0B,CAAC;AAElC,OAAO,EAAE,+BAA+B,EAAE,MAAM,mBAAmB,CAAC;AAEpE,OAAO,EACL,kBAAkB,EAClB,yBAAyB,GAC1B,MAAM,mBAAmB,CAAC;AAE3B,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC"}
|
package/dist/index.js
CHANGED
|
@@ -378,12 +378,6 @@ var defaultWorkflowValidationPolicy = {
|
|
|
378
378
|
includeDevDependenciesTransitively: false
|
|
379
379
|
};
|
|
380
380
|
|
|
381
|
-
// src/workflow/validator.ts
|
|
382
|
-
import { existsSync as existsSync4 } from "fs";
|
|
383
|
-
import { readFile } from "fs/promises";
|
|
384
|
-
import { glob } from "glob";
|
|
385
|
-
import { parse as parseYaml2 } from "yaml";
|
|
386
|
-
|
|
387
381
|
// src/workflow/discovery.ts
|
|
388
382
|
import { existsSync as existsSync2, readdirSync } from "fs";
|
|
389
383
|
import { join as join3 } from "path";
|
|
@@ -503,11 +497,69 @@ function getExpectedWorkflowPaths({
|
|
|
503
497
|
return uniqueSorted(expectedPaths);
|
|
504
498
|
}
|
|
505
499
|
|
|
500
|
+
// src/workflow/impact.ts
|
|
501
|
+
function uniqueSorted2(values) {
|
|
502
|
+
return Array.from(new Set(values)).sort();
|
|
503
|
+
}
|
|
504
|
+
function mergeWorkflowPolicy(policyOverrides) {
|
|
505
|
+
return {
|
|
506
|
+
...defaultWorkflowValidationPolicy,
|
|
507
|
+
...policyOverrides,
|
|
508
|
+
allowedRootPaths: uniqueSorted2([
|
|
509
|
+
...defaultWorkflowValidationPolicy.allowedRootPaths,
|
|
510
|
+
...policyOverrides?.allowedRootPaths ?? []
|
|
511
|
+
])
|
|
512
|
+
};
|
|
513
|
+
}
|
|
514
|
+
function matchesWorkflowPathFilter(changedPath, workflowPathFilter) {
|
|
515
|
+
if (workflowPathFilter === changedPath) {
|
|
516
|
+
return true;
|
|
517
|
+
}
|
|
518
|
+
if (!workflowPathFilter.endsWith("/**")) {
|
|
519
|
+
return false;
|
|
520
|
+
}
|
|
521
|
+
const workflowPrefix = workflowPathFilter.slice(0, -3);
|
|
522
|
+
return changedPath.startsWith(`${workflowPrefix}/`);
|
|
523
|
+
}
|
|
524
|
+
function getWorkflowImpacts(rootDir, packages, changedPaths, policyOverrides) {
|
|
525
|
+
const policy = mergeWorkflowPolicy(policyOverrides);
|
|
526
|
+
const workflowTargets = discoverWorkflowTargets(rootDir, packages, policy);
|
|
527
|
+
const { packageMap } = buildPackageMap(packages, rootDir);
|
|
528
|
+
return workflowTargets.map((workflowTarget) => {
|
|
529
|
+
const calculatedPaths = uniqueSorted2([
|
|
530
|
+
...getExpectedWorkflowPaths({
|
|
531
|
+
workflowTarget,
|
|
532
|
+
packages,
|
|
533
|
+
packageMap,
|
|
534
|
+
policy
|
|
535
|
+
}),
|
|
536
|
+
...policy.allowedRootPaths,
|
|
537
|
+
workflowTarget.workflowPath
|
|
538
|
+
]);
|
|
539
|
+
const matchedPaths = calculatedPaths.filter(
|
|
540
|
+
(calculatedPath) => changedPaths.some(
|
|
541
|
+
(changedPath) => matchesWorkflowPathFilter(changedPath, calculatedPath)
|
|
542
|
+
)
|
|
543
|
+
);
|
|
544
|
+
return {
|
|
545
|
+
...workflowTarget,
|
|
546
|
+
matchedPaths: uniqueSorted2(matchedPaths)
|
|
547
|
+
};
|
|
548
|
+
}).filter((workflowImpact) => workflowImpact.matchedPaths.length > 0);
|
|
549
|
+
}
|
|
550
|
+
|
|
551
|
+
// src/workflow/validator.ts
|
|
552
|
+
import { existsSync as existsSync4 } from "fs";
|
|
553
|
+
import { readFile } from "fs/promises";
|
|
554
|
+
import { join as join5 } from "path";
|
|
555
|
+
import { glob } from "glob";
|
|
556
|
+
import { parse as parseYaml2 } from "yaml";
|
|
557
|
+
|
|
506
558
|
// src/workflow/parser.ts
|
|
507
559
|
import { existsSync as existsSync3, readFileSync } from "fs";
|
|
508
560
|
import { join as join4 } from "path";
|
|
509
561
|
import { parse as parseYaml } from "yaml";
|
|
510
|
-
function
|
|
562
|
+
function uniqueSorted3(values) {
|
|
511
563
|
return Array.from(new Set(values)).sort();
|
|
512
564
|
}
|
|
513
565
|
function parseWorkflowFile(workflowFile, rootDir) {
|
|
@@ -520,9 +572,9 @@ function parseWorkflowFile(workflowFile, rootDir) {
|
|
|
520
572
|
const pushPaths = workflow.on?.push?.paths ?? [];
|
|
521
573
|
const pullRequestPaths = workflow.on?.pull_request?.paths ?? [];
|
|
522
574
|
const parsedWorkflow = {
|
|
523
|
-
pushPaths:
|
|
524
|
-
pullRequestPaths:
|
|
525
|
-
paths:
|
|
575
|
+
pushPaths: uniqueSorted3(pushPaths),
|
|
576
|
+
pullRequestPaths: uniqueSorted3(pullRequestPaths),
|
|
577
|
+
paths: uniqueSorted3([...pushPaths, ...pullRequestPaths])
|
|
526
578
|
};
|
|
527
579
|
if (workflow.name) {
|
|
528
580
|
parsedWorkflow.name = workflow.name;
|
|
@@ -531,7 +583,7 @@ function parseWorkflowFile(workflowFile, rootDir) {
|
|
|
531
583
|
}
|
|
532
584
|
|
|
533
585
|
// src/workflow/validator.ts
|
|
534
|
-
function
|
|
586
|
+
function uniqueSorted4(values) {
|
|
535
587
|
return Array.from(new Set(values)).sort();
|
|
536
588
|
}
|
|
537
589
|
function buildBroadWildcards(workspaceRoots) {
|
|
@@ -542,6 +594,23 @@ function buildBroadWildcards(workspaceRoots) {
|
|
|
542
594
|
}
|
|
543
595
|
return wildcards;
|
|
544
596
|
}
|
|
597
|
+
function hasGlobSyntax(path) {
|
|
598
|
+
return /[*?[\]{}]/.test(path);
|
|
599
|
+
}
|
|
600
|
+
function getStalePathCheckBase(path, allowedRootPaths, workflowPath) {
|
|
601
|
+
if (path === workflowPath || allowedRootPaths.includes(path) || path.startsWith("!")) {
|
|
602
|
+
return null;
|
|
603
|
+
}
|
|
604
|
+
const trailingGlobSuffix = path.endsWith("/**") ? "/**" : path.endsWith("/*") ? "/*" : null;
|
|
605
|
+
const checkBase = trailingGlobSuffix ? path.slice(0, -trailingGlobSuffix.length) : path;
|
|
606
|
+
if (!checkBase || hasGlobSyntax(checkBase)) {
|
|
607
|
+
return null;
|
|
608
|
+
}
|
|
609
|
+
if (!trailingGlobSuffix && hasGlobSyntax(path)) {
|
|
610
|
+
return null;
|
|
611
|
+
}
|
|
612
|
+
return checkBase;
|
|
613
|
+
}
|
|
545
614
|
function splitActualPaths(paths, workspaceRoots, allowedRootPaths, workflowPath) {
|
|
546
615
|
const workspacePaths = [];
|
|
547
616
|
const ignoredPaths = [];
|
|
@@ -558,8 +627,8 @@ function splitActualPaths(paths, workspaceRoots, allowedRootPaths, workflowPath)
|
|
|
558
627
|
ignoredPaths.push(path);
|
|
559
628
|
}
|
|
560
629
|
return {
|
|
561
|
-
workspacePaths:
|
|
562
|
-
ignoredPaths:
|
|
630
|
+
workspacePaths: uniqueSorted4(workspacePaths),
|
|
631
|
+
ignoredPaths: uniqueSorted4(ignoredPaths)
|
|
563
632
|
};
|
|
564
633
|
}
|
|
565
634
|
function isCoveredByExpectedPath(actualPath, expectedPaths) {
|
|
@@ -574,7 +643,7 @@ function isCoveredByExpectedPath(actualPath, expectedPaths) {
|
|
|
574
643
|
return actualPath.startsWith(`${expectedPrefix}/`);
|
|
575
644
|
});
|
|
576
645
|
}
|
|
577
|
-
function validateWorkflowResult(workflow, targetPackage, expectedPaths, actualPaths, broadWildcards) {
|
|
646
|
+
function validateWorkflowResult(workflow, targetPackage, expectedPaths, actualPaths, broadWildcards, additionalIssues = []) {
|
|
578
647
|
const issues = [];
|
|
579
648
|
const missing = expectedPaths.filter((path) => !actualPaths.includes(path));
|
|
580
649
|
const unnecessary = actualPaths.filter(
|
|
@@ -603,6 +672,7 @@ function validateWorkflowResult(workflow, targetPackage, expectedPaths, actualPa
|
|
|
603
672
|
});
|
|
604
673
|
}
|
|
605
674
|
}
|
|
675
|
+
issues.push(...additionalIssues);
|
|
606
676
|
return {
|
|
607
677
|
workflow,
|
|
608
678
|
targetPackage,
|
|
@@ -614,6 +684,25 @@ function validateWorkflowResult(workflow, targetPackage, expectedPaths, actualPa
|
|
|
614
684
|
issues
|
|
615
685
|
};
|
|
616
686
|
}
|
|
687
|
+
function getStalePathIssues(ignoredPaths, rootDir, allowedRootPaths, workflowPath) {
|
|
688
|
+
return ignoredPaths.flatMap((path) => {
|
|
689
|
+
const checkBase = getStalePathCheckBase(
|
|
690
|
+
path,
|
|
691
|
+
allowedRootPaths,
|
|
692
|
+
workflowPath
|
|
693
|
+
);
|
|
694
|
+
if (!checkBase || existsSync4(join5(rootDir, checkBase))) {
|
|
695
|
+
return [];
|
|
696
|
+
}
|
|
697
|
+
return [
|
|
698
|
+
{
|
|
699
|
+
kind: "stale-path",
|
|
700
|
+
path,
|
|
701
|
+
message: `Stale path '${path}' does not exist`
|
|
702
|
+
}
|
|
703
|
+
];
|
|
704
|
+
});
|
|
705
|
+
}
|
|
617
706
|
async function validateWorkflows(rootDir, policyOverrides) {
|
|
618
707
|
const discoveredPackages = await discoverWorkspaces(rootDir, {
|
|
619
708
|
fs: {
|
|
@@ -630,7 +719,7 @@ async function validateWorkflows(rootDir, policyOverrides) {
|
|
|
630
719
|
const policy = {
|
|
631
720
|
...defaultWorkflowValidationPolicy,
|
|
632
721
|
...policyOverrides,
|
|
633
|
-
allowedRootPaths:
|
|
722
|
+
allowedRootPaths: uniqueSorted4([
|
|
634
723
|
...defaultWorkflowValidationPolicy.allowedRootPaths,
|
|
635
724
|
...policyOverrides?.allowedRootPaths ?? []
|
|
636
725
|
])
|
|
@@ -680,7 +769,7 @@ async function validateWorkflows(rootDir, policyOverrides) {
|
|
|
680
769
|
issues: []
|
|
681
770
|
};
|
|
682
771
|
}
|
|
683
|
-
const { workspacePaths: actualPaths } = splitActualPaths(
|
|
772
|
+
const { workspacePaths: actualPaths, ignoredPaths } = splitActualPaths(
|
|
684
773
|
parsedWorkflow.paths,
|
|
685
774
|
workspaceRoots,
|
|
686
775
|
policy.allowedRootPaths,
|
|
@@ -697,7 +786,13 @@ async function validateWorkflows(rootDir, policyOverrides) {
|
|
|
697
786
|
workflowTarget.targetPackage,
|
|
698
787
|
expectedPaths,
|
|
699
788
|
actualPaths,
|
|
700
|
-
broadWildcards
|
|
789
|
+
broadWildcards,
|
|
790
|
+
getStalePathIssues(
|
|
791
|
+
ignoredPaths,
|
|
792
|
+
rootDir,
|
|
793
|
+
policy.allowedRootPaths,
|
|
794
|
+
workflowTarget.workflowPath
|
|
795
|
+
)
|
|
701
796
|
);
|
|
702
797
|
} catch (error) {
|
|
703
798
|
return {
|
|
@@ -731,7 +826,9 @@ export {
|
|
|
731
826
|
findPackageForFile,
|
|
732
827
|
getTransitiveDependencies,
|
|
733
828
|
getTransitiveDependents,
|
|
829
|
+
getWorkflowImpacts,
|
|
734
830
|
mapFilesToPackages,
|
|
831
|
+
matchesWorkflowPathFilter,
|
|
735
832
|
normalizeRelativePath,
|
|
736
833
|
validateWorkflows
|
|
737
834
|
};
|