@crossplatformai/dependency-graph 0.10.0 → 0.11.0-next.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,3 +1,5 @@
1
1
  #!/usr/bin/env node
2
+ import { type WorkflowValidationResult } from '../index';
3
+ export declare function formatWorkflowPathDrift(validationResults: WorkflowValidationResult[]): string[];
2
4
  export declare function runPrPreview(): Promise<void>;
3
5
  //# sourceMappingURL=pr-preview.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"pr-preview.d.ts","sourceRoot":"","sources":["../../src/cli/pr-preview.ts"],"names":[],"mappings":";AAoLA,wBAAsB,YAAY,IAAI,OAAO,CAAC,IAAI,CAAC,CA6MlD"}
1
+ {"version":3,"file":"pr-preview.d.ts","sourceRoot":"","sources":["../../src/cli/pr-preview.ts"],"names":[],"mappings":";AAQA,OAAO,EAOL,KAAK,wBAAwB,EAC9B,MAAM,UAAU,CAAC;AA0DlB,wBAAgB,uBAAuB,CACrC,iBAAiB,EAAE,wBAAwB,EAAE,GAC5C,MAAM,EAAE,CAyBV;AAsGD,wBAAsB,YAAY,IAAI,OAAO,CAAC,IAAI,CAAC,CA4NlD"}
package/dist/index-cli.js CHANGED
@@ -208,12 +208,6 @@ var defaultWorkflowValidationPolicy = {
208
208
  includeDevDependenciesTransitively: false
209
209
  };
210
210
 
211
- // src/workflow/validator.ts
212
- import { existsSync as existsSync4 } from "fs";
213
- import { readFile } from "fs/promises";
214
- import { glob } from "glob";
215
- import { parse as parseYaml2 } from "yaml";
216
-
217
211
  // src/workflow/discovery.ts
218
212
  import { existsSync as existsSync2, readdirSync } from "fs";
219
213
  import { join as join3 } from "path";
@@ -333,11 +327,68 @@ function getExpectedWorkflowPaths({
333
327
  return uniqueSorted(expectedPaths);
334
328
  }
335
329
 
330
+ // src/workflow/impact.ts
331
+ function uniqueSorted2(values) {
332
+ return Array.from(new Set(values)).sort();
333
+ }
334
+ function mergeWorkflowPolicy(policyOverrides) {
335
+ return {
336
+ ...defaultWorkflowValidationPolicy,
337
+ ...policyOverrides,
338
+ allowedRootPaths: uniqueSorted2([
339
+ ...defaultWorkflowValidationPolicy.allowedRootPaths,
340
+ ...policyOverrides?.allowedRootPaths ?? []
341
+ ])
342
+ };
343
+ }
344
+ function matchesWorkflowPathFilter(changedPath, workflowPathFilter) {
345
+ if (workflowPathFilter === changedPath) {
346
+ return true;
347
+ }
348
+ if (!workflowPathFilter.endsWith("/**")) {
349
+ return false;
350
+ }
351
+ const workflowPrefix = workflowPathFilter.slice(0, -3);
352
+ return changedPath.startsWith(`${workflowPrefix}/`);
353
+ }
354
+ function getWorkflowImpacts(rootDir, packages, changedPaths, policyOverrides) {
355
+ const policy = mergeWorkflowPolicy(policyOverrides);
356
+ const workflowTargets = discoverWorkflowTargets(rootDir, packages, policy);
357
+ const { packageMap } = buildPackageMap(packages, rootDir);
358
+ return workflowTargets.map((workflowTarget) => {
359
+ const calculatedPaths = uniqueSorted2([
360
+ ...getExpectedWorkflowPaths({
361
+ workflowTarget,
362
+ packages,
363
+ packageMap,
364
+ policy
365
+ }),
366
+ ...policy.allowedRootPaths,
367
+ workflowTarget.workflowPath
368
+ ]);
369
+ const matchedPaths = calculatedPaths.filter(
370
+ (calculatedPath) => changedPaths.some(
371
+ (changedPath) => matchesWorkflowPathFilter(changedPath, calculatedPath)
372
+ )
373
+ );
374
+ return {
375
+ ...workflowTarget,
376
+ matchedPaths: uniqueSorted2(matchedPaths)
377
+ };
378
+ }).filter((workflowImpact) => workflowImpact.matchedPaths.length > 0);
379
+ }
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
+
336
387
  // src/workflow/parser.ts
337
388
  import { existsSync as existsSync3, readFileSync } from "fs";
338
389
  import { join as join4 } from "path";
339
390
  import { parse as parseYaml } from "yaml";
340
- function uniqueSorted2(values) {
391
+ function uniqueSorted3(values) {
341
392
  return Array.from(new Set(values)).sort();
342
393
  }
343
394
  function parseWorkflowFile(workflowFile, rootDir) {
@@ -350,9 +401,9 @@ function parseWorkflowFile(workflowFile, rootDir) {
350
401
  const pushPaths = workflow.on?.push?.paths ?? [];
351
402
  const pullRequestPaths = workflow.on?.pull_request?.paths ?? [];
352
403
  const parsedWorkflow = {
353
- pushPaths: uniqueSorted2(pushPaths),
354
- pullRequestPaths: uniqueSorted2(pullRequestPaths),
355
- paths: uniqueSorted2([...pushPaths, ...pullRequestPaths])
404
+ pushPaths: uniqueSorted3(pushPaths),
405
+ pullRequestPaths: uniqueSorted3(pullRequestPaths),
406
+ paths: uniqueSorted3([...pushPaths, ...pullRequestPaths])
356
407
  };
357
408
  if (workflow.name) {
358
409
  parsedWorkflow.name = workflow.name;
@@ -361,7 +412,7 @@ function parseWorkflowFile(workflowFile, rootDir) {
361
412
  }
362
413
 
363
414
  // src/workflow/validator.ts
364
- function uniqueSorted3(values) {
415
+ function uniqueSorted4(values) {
365
416
  return Array.from(new Set(values)).sort();
366
417
  }
367
418
  function buildBroadWildcards(workspaceRoots) {
@@ -388,8 +439,8 @@ function splitActualPaths(paths, workspaceRoots, allowedRootPaths, workflowPath)
388
439
  ignoredPaths.push(path);
389
440
  }
390
441
  return {
391
- workspacePaths: uniqueSorted3(workspacePaths),
392
- ignoredPaths: uniqueSorted3(ignoredPaths)
442
+ workspacePaths: uniqueSorted4(workspacePaths),
443
+ ignoredPaths: uniqueSorted4(ignoredPaths)
393
444
  };
394
445
  }
395
446
  function isCoveredByExpectedPath(actualPath, expectedPaths) {
@@ -460,7 +511,7 @@ async function validateWorkflows(rootDir, policyOverrides) {
460
511
  const policy = {
461
512
  ...defaultWorkflowValidationPolicy,
462
513
  ...policyOverrides,
463
- allowedRootPaths: uniqueSorted3([
514
+ allowedRootPaths: uniqueSorted4([
464
515
  ...defaultWorkflowValidationPolicy.allowedRootPaths,
465
516
  ...policyOverrides?.allowedRootPaths ?? []
466
517
  ])
@@ -550,14 +601,6 @@ async function validateWorkflows(rootDir, policyOverrides) {
550
601
  }
551
602
 
552
603
  // src/cli/pr-preview.ts
553
- var WORKFLOW_MAPPING = {
554
- "deploy-web.yml": "web",
555
- "deploy-api-origin.yml": "api-origin",
556
- "deploy-api-edge.yml": "api-edge",
557
- "release-mobile.yml": "mobile",
558
- "release-desktop.yml": "desktop",
559
- "release-cli.yml": "cli"
560
- };
561
604
  var PLATFORM_INDICATORS = [
562
605
  {
563
606
  platform: "cloudflare-workers",
@@ -596,6 +639,28 @@ var PLATFORM_INDICATORS = [
596
639
  }
597
640
  ];
598
641
  var DEFAULT_APP_DIRECTORIES = ["apps"];
642
+ function formatWorkflowPathDrift(validationResults) {
643
+ const invalidResults = validationResults.filter((result) => !result.valid);
644
+ if (invalidResults.length === 0) {
645
+ return [];
646
+ }
647
+ const lines = [
648
+ "\u26A0\uFE0F Workflow path drift (advisory - does not affect impact above):"
649
+ ];
650
+ for (const result of invalidResults) {
651
+ lines.push(` ${result.workflow} (${result.targetPackage}):`);
652
+ for (const issue of result.issues) {
653
+ if (issue.kind === "missing") {
654
+ lines.push(` + ${issue.message}`);
655
+ } else if (issue.kind === "unnecessary") {
656
+ lines.push(` - ${issue.message}`);
657
+ } else {
658
+ lines.push(` ! ${issue.message}`);
659
+ }
660
+ }
661
+ }
662
+ return lines;
663
+ }
599
664
  function isDeployableApp(pkg) {
600
665
  const isInAppDirectory = DEFAULT_APP_DIRECTORIES.some(
601
666
  (dir) => pkg.path.includes(`/${dir}/`) || pkg.path.endsWith(`/${dir}`)
@@ -713,23 +778,32 @@ async function runPrPreview() {
713
778
  changedPackages.add(pkg.name);
714
779
  }
715
780
  }
716
- const changedWorkflows = changedFiles.filter(
717
- (f) => f.path.startsWith(".github/workflows/")
781
+ const workflowImpacts = getWorkflowImpacts(
782
+ rootDir,
783
+ packages,
784
+ changedFiles.map((file) => file.path)
785
+ );
786
+ const workflowDriftLines = formatWorkflowPathDrift(
787
+ await validateWorkflows(rootDir)
788
+ );
789
+ const workflowAffectedApps = new Set(
790
+ workflowImpacts.map(
791
+ (workflowImpact) => workflowImpact.targetPackage ?? workflowImpact.targetSlug
792
+ )
718
793
  );
719
- const workflowAffectedApps = /* @__PURE__ */ new Set();
720
- for (const workflow of changedWorkflows) {
721
- const workflowName = workflow.path.split("/").pop();
722
- if (workflowName && WORKFLOW_MAPPING[workflowName]) {
723
- workflowAffectedApps.add(WORKFLOW_MAPPING[workflowName]);
724
- }
725
- }
726
794
  console.log(
727
795
  `
728
796
  \u{1F4E6} Changed packages: ${Array.from(changedPackages).join(", ") || "none"}
729
797
  `
730
798
  );
731
- if (changedPackages.size === 0) {
732
- console.log("\n\u2728 No packages changed - no deployments needed\n");
799
+ if (changedPackages.size === 0 && workflowAffectedApps.size === 0) {
800
+ console.log(
801
+ "\n\u2728 No packages or workflow filters changed - no deployments needed\n"
802
+ );
803
+ if (workflowDriftLines.length > 0) {
804
+ console.log(workflowDriftLines.join("\n"));
805
+ console.log("");
806
+ }
733
807
  return;
734
808
  }
735
809
  const affected = findAffectedPackages(changedPackages, graph, {
@@ -826,6 +900,10 @@ async function runPrPreview() {
826
900
  console.log(`\u{1F4DD} = File changed`);
827
901
  console.log(`\u274C = File deleted
828
902
  `);
903
+ if (workflowDriftLines.length > 0) {
904
+ console.log(workflowDriftLines.join("\n"));
905
+ console.log("");
906
+ }
829
907
  } catch (error) {
830
908
  console.error(
831
909
  "Error:",
@@ -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 { 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, 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","parseYaml","existsSync","join","uniqueSorted","uniqueSorted","existsSync","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
@@ -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,68 @@ 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 { glob } from "glob";
555
+ import { parse as parseYaml2 } from "yaml";
556
+
506
557
  // src/workflow/parser.ts
507
558
  import { existsSync as existsSync3, readFileSync } from "fs";
508
559
  import { join as join4 } from "path";
509
560
  import { parse as parseYaml } from "yaml";
510
- function uniqueSorted2(values) {
561
+ function uniqueSorted3(values) {
511
562
  return Array.from(new Set(values)).sort();
512
563
  }
513
564
  function parseWorkflowFile(workflowFile, rootDir) {
@@ -520,9 +571,9 @@ function parseWorkflowFile(workflowFile, rootDir) {
520
571
  const pushPaths = workflow.on?.push?.paths ?? [];
521
572
  const pullRequestPaths = workflow.on?.pull_request?.paths ?? [];
522
573
  const parsedWorkflow = {
523
- pushPaths: uniqueSorted2(pushPaths),
524
- pullRequestPaths: uniqueSorted2(pullRequestPaths),
525
- paths: uniqueSorted2([...pushPaths, ...pullRequestPaths])
574
+ pushPaths: uniqueSorted3(pushPaths),
575
+ pullRequestPaths: uniqueSorted3(pullRequestPaths),
576
+ paths: uniqueSorted3([...pushPaths, ...pullRequestPaths])
526
577
  };
527
578
  if (workflow.name) {
528
579
  parsedWorkflow.name = workflow.name;
@@ -531,7 +582,7 @@ function parseWorkflowFile(workflowFile, rootDir) {
531
582
  }
532
583
 
533
584
  // src/workflow/validator.ts
534
- function uniqueSorted3(values) {
585
+ function uniqueSorted4(values) {
535
586
  return Array.from(new Set(values)).sort();
536
587
  }
537
588
  function buildBroadWildcards(workspaceRoots) {
@@ -558,8 +609,8 @@ function splitActualPaths(paths, workspaceRoots, allowedRootPaths, workflowPath)
558
609
  ignoredPaths.push(path);
559
610
  }
560
611
  return {
561
- workspacePaths: uniqueSorted3(workspacePaths),
562
- ignoredPaths: uniqueSorted3(ignoredPaths)
612
+ workspacePaths: uniqueSorted4(workspacePaths),
613
+ ignoredPaths: uniqueSorted4(ignoredPaths)
563
614
  };
564
615
  }
565
616
  function isCoveredByExpectedPath(actualPath, expectedPaths) {
@@ -630,7 +681,7 @@ async function validateWorkflows(rootDir, policyOverrides) {
630
681
  const policy = {
631
682
  ...defaultWorkflowValidationPolicy,
632
683
  ...policyOverrides,
633
- allowedRootPaths: uniqueSorted3([
684
+ allowedRootPaths: uniqueSorted4([
634
685
  ...defaultWorkflowValidationPolicy.allowedRootPaths,
635
686
  ...policyOverrides?.allowedRootPaths ?? []
636
687
  ])
@@ -731,7 +782,9 @@ export {
731
782
  findPackageForFile,
732
783
  getTransitiveDependencies,
733
784
  getTransitiveDependents,
785
+ getWorkflowImpacts,
734
786
  mapFilesToPackages,
787
+ matchesWorkflowPathFilter,
735
788
  normalizeRelativePath,
736
789
  validateWorkflows
737
790
  };
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/graph/builder.ts","../src/graph/traversal.ts","../src/graph/analysis.ts","../src/workspace/discovery.ts","../src/workspace/package-map.ts","../src/workspace/file-mapping.ts","../src/workflow/policy.ts","../src/workflow/validator.ts","../src/workflow/discovery.ts","../src/workflow/expected-paths.ts","../src/workflow/parser.ts"],"sourcesContent":["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 type { DependencyGraph, GraphStats } from './types';\n\nexport function analyzeGraph(graph: DependencyGraph): GraphStats {\n const leafNodes: string[] = [];\n const rootNodes: string[] = [];\n let totalEdges = 0;\n\n for (const [pkgName, dependents] of graph.dependedBy.entries()) {\n if (dependents.size === 0) {\n leafNodes.push(pkgName);\n }\n totalEdges += dependents.size;\n }\n\n for (const [pkgName, dependencies] of graph.dependsOn.entries()) {\n if (dependencies.size === 0) {\n rootNodes.push(pkgName);\n }\n }\n\n const maxDepth = calculateMaxDepth(graph);\n const cycles = detectCycles(graph);\n\n return {\n totalPackages: graph.packages.size,\n totalEdges,\n maxDepth,\n leafNodes,\n rootNodes,\n cycles,\n };\n}\n\nfunction calculateMaxDepth(graph: DependencyGraph): number {\n let maxDepth = 0;\n\n for (const pkgName of graph.packages.keys()) {\n const depth = getPackageDepth(pkgName, graph);\n if (depth > maxDepth) {\n maxDepth = depth;\n }\n }\n\n return maxDepth;\n}\n\nfunction getPackageDepth(\n pkgName: string,\n graph: DependencyGraph,\n visited = new Set<string>(),\n): number {\n if (visited.has(pkgName)) return 0;\n visited.add(pkgName);\n\n const dependencies = graph.dependsOn.get(pkgName) || new Set();\n if (dependencies.size === 0) return 0;\n\n let maxDepth = 0;\n for (const dep of dependencies) {\n const depth = getPackageDepth(dep, graph, new Set(visited));\n if (depth > maxDepth) {\n maxDepth = depth;\n }\n }\n\n return maxDepth + 1;\n}\n\nexport function detectCycles(graph: DependencyGraph): string[][] {\n const cycles: string[][] = [];\n const visited = new Set<string>();\n const recursionStack = new Set<string>();\n\n function dfs(pkgName: string, path: string[]): void {\n visited.add(pkgName);\n recursionStack.add(pkgName);\n path.push(pkgName);\n\n const dependencies = graph.dependsOn.get(pkgName) || new Set();\n for (const dep of dependencies) {\n if (!visited.has(dep)) {\n dfs(dep, [...path]);\n } else if (recursionStack.has(dep)) {\n const cycleStart = path.indexOf(dep);\n if (cycleStart !== -1) {\n const cycle = path.slice(cycleStart);\n cycle.push(dep);\n cycles.push(cycle);\n }\n }\n }\n\n recursionStack.delete(pkgName);\n }\n\n for (const pkgName of graph.packages.keys()) {\n if (!visited.has(pkgName)) {\n dfs(pkgName, []);\n }\n }\n\n return cycles;\n}\n\nexport function getTransitiveDependencies(\n pkgName: string,\n graph: DependencyGraph,\n): Set<string> {\n const transitive = new Set<string>();\n const visited = new Set<string>();\n\n function collect(current: string): void {\n if (visited.has(current)) return;\n visited.add(current);\n\n const deps = graph.dependsOn.get(current) || new Set();\n for (const dep of deps) {\n transitive.add(dep);\n collect(dep);\n }\n }\n\n collect(pkgName);\n return transitive;\n}\n\nexport function getTransitiveDependents(\n pkgName: string,\n graph: DependencyGraph,\n): Set<string> {\n const transitive = new Set<string>();\n const visited = new Set<string>();\n\n function collect(current: string): void {\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 transitive.add(dependent);\n collect(dependent);\n }\n }\n\n collect(pkgName);\n return transitive;\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 { WorkspacePackage } from '../graph/types';\n\nexport function findPackageForFile(\n filePath: string,\n packages: WorkspacePackage[],\n): WorkspacePackage | undefined {\n const sorted = packages.sort((a, b) => b.path.length - a.path.length);\n\n for (const pkg of sorted) {\n if (filePath.startsWith(pkg.path + '/') || filePath === pkg.path) {\n return pkg;\n }\n }\n\n return undefined;\n}\n\nexport function mapFilesToPackages(\n files: string[],\n packages: WorkspacePackage[],\n): Map<string, string[]> {\n const fileMap = new Map<string, string[]>();\n\n for (const file of files) {\n const pkg = findPackageForFile(file, packages);\n if (pkg) {\n if (!fileMap.has(pkg.name)) {\n fileMap.set(pkg.name, []);\n }\n fileMap.get(pkg.name)!.push(file);\n }\n }\n\n return fileMap;\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"],"mappings":"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","names":["join","existsSync","parseYaml","existsSync","join","join","existsSync","existsSync","join","uniqueSorted","uniqueSorted","existsSync","parseYaml"]}
1
+ {"version":3,"sources":["../src/graph/builder.ts","../src/graph/traversal.ts","../src/graph/analysis.ts","../src/workspace/discovery.ts","../src/workspace/package-map.ts","../src/workspace/file-mapping.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"],"sourcesContent":["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 type { DependencyGraph, GraphStats } from './types';\n\nexport function analyzeGraph(graph: DependencyGraph): GraphStats {\n const leafNodes: string[] = [];\n const rootNodes: string[] = [];\n let totalEdges = 0;\n\n for (const [pkgName, dependents] of graph.dependedBy.entries()) {\n if (dependents.size === 0) {\n leafNodes.push(pkgName);\n }\n totalEdges += dependents.size;\n }\n\n for (const [pkgName, dependencies] of graph.dependsOn.entries()) {\n if (dependencies.size === 0) {\n rootNodes.push(pkgName);\n }\n }\n\n const maxDepth = calculateMaxDepth(graph);\n const cycles = detectCycles(graph);\n\n return {\n totalPackages: graph.packages.size,\n totalEdges,\n maxDepth,\n leafNodes,\n rootNodes,\n cycles,\n };\n}\n\nfunction calculateMaxDepth(graph: DependencyGraph): number {\n let maxDepth = 0;\n\n for (const pkgName of graph.packages.keys()) {\n const depth = getPackageDepth(pkgName, graph);\n if (depth > maxDepth) {\n maxDepth = depth;\n }\n }\n\n return maxDepth;\n}\n\nfunction getPackageDepth(\n pkgName: string,\n graph: DependencyGraph,\n visited = new Set<string>(),\n): number {\n if (visited.has(pkgName)) return 0;\n visited.add(pkgName);\n\n const dependencies = graph.dependsOn.get(pkgName) || new Set();\n if (dependencies.size === 0) return 0;\n\n let maxDepth = 0;\n for (const dep of dependencies) {\n const depth = getPackageDepth(dep, graph, new Set(visited));\n if (depth > maxDepth) {\n maxDepth = depth;\n }\n }\n\n return maxDepth + 1;\n}\n\nexport function detectCycles(graph: DependencyGraph): string[][] {\n const cycles: string[][] = [];\n const visited = new Set<string>();\n const recursionStack = new Set<string>();\n\n function dfs(pkgName: string, path: string[]): void {\n visited.add(pkgName);\n recursionStack.add(pkgName);\n path.push(pkgName);\n\n const dependencies = graph.dependsOn.get(pkgName) || new Set();\n for (const dep of dependencies) {\n if (!visited.has(dep)) {\n dfs(dep, [...path]);\n } else if (recursionStack.has(dep)) {\n const cycleStart = path.indexOf(dep);\n if (cycleStart !== -1) {\n const cycle = path.slice(cycleStart);\n cycle.push(dep);\n cycles.push(cycle);\n }\n }\n }\n\n recursionStack.delete(pkgName);\n }\n\n for (const pkgName of graph.packages.keys()) {\n if (!visited.has(pkgName)) {\n dfs(pkgName, []);\n }\n }\n\n return cycles;\n}\n\nexport function getTransitiveDependencies(\n pkgName: string,\n graph: DependencyGraph,\n): Set<string> {\n const transitive = new Set<string>();\n const visited = new Set<string>();\n\n function collect(current: string): void {\n if (visited.has(current)) return;\n visited.add(current);\n\n const deps = graph.dependsOn.get(current) || new Set();\n for (const dep of deps) {\n transitive.add(dep);\n collect(dep);\n }\n }\n\n collect(pkgName);\n return transitive;\n}\n\nexport function getTransitiveDependents(\n pkgName: string,\n graph: DependencyGraph,\n): Set<string> {\n const transitive = new Set<string>();\n const visited = new Set<string>();\n\n function collect(current: string): void {\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 transitive.add(dependent);\n collect(dependent);\n }\n }\n\n collect(pkgName);\n return transitive;\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 { WorkspacePackage } from '../graph/types';\n\nexport function findPackageForFile(\n filePath: string,\n packages: WorkspacePackage[],\n): WorkspacePackage | undefined {\n const sorted = packages.sort((a, b) => b.path.length - a.path.length);\n\n for (const pkg of sorted) {\n if (filePath.startsWith(pkg.path + '/') || filePath === pkg.path) {\n return pkg;\n }\n }\n\n return undefined;\n}\n\nexport function mapFilesToPackages(\n files: string[],\n packages: WorkspacePackage[],\n): Map<string, string[]> {\n const fileMap = new Map<string, string[]>();\n\n for (const file of files) {\n const pkg = findPackageForFile(file, packages);\n if (pkg) {\n if (!fileMap.has(pkg.name)) {\n fileMap.set(pkg.name, []);\n }\n fileMap.get(pkg.name)!.push(file);\n }\n }\n\n return fileMap;\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 { 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, 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"],"mappings":"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","names":["join","existsSync","join","join","existsSync","uniqueSorted","existsSync","parseYaml","existsSync","join","uniqueSorted","uniqueSorted","existsSync","parseYaml"]}
@@ -0,0 +1,5 @@
1
+ import type { WorkspacePackage } from '../graph/types';
2
+ import type { WorkflowImpact, WorkflowValidationPolicy } from './types';
3
+ export declare function matchesWorkflowPathFilter(changedPath: string, workflowPathFilter: string): boolean;
4
+ export declare function getWorkflowImpacts(rootDir: string, packages: WorkspacePackage[], changedPaths: string[], policyOverrides?: Partial<WorkflowValidationPolicy>): WorkflowImpact[];
5
+ //# sourceMappingURL=impact.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"impact.d.ts","sourceRoot":"","sources":["../../src/workflow/impact.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAKvD,OAAO,KAAK,EAAE,cAAc,EAAE,wBAAwB,EAAE,MAAM,SAAS,CAAC;AAqBxE,wBAAgB,yBAAyB,CACvC,WAAW,EAAE,MAAM,EACnB,kBAAkB,EAAE,MAAM,GACzB,OAAO,CAWT;AAED,wBAAgB,kBAAkB,CAChC,OAAO,EAAE,MAAM,EACf,QAAQ,EAAE,gBAAgB,EAAE,EAC5B,YAAY,EAAE,MAAM,EAAE,EACtB,eAAe,CAAC,EAAE,OAAO,CAAC,wBAAwB,CAAC,GAClD,cAAc,EAAE,CA6BlB"}
@@ -1,5 +1,9 @@
1
1
  export interface WorkflowValidationPolicy {
2
2
  workflowFilePatterns: string[];
3
+ /**
4
+ * Root-level paths that currently act as global workflow-impact paths and
5
+ * validator-ignored runtime config paths.
6
+ */
3
7
  allowedRootPaths: string[];
4
8
  includeDevDependenciesForRootPackage: boolean;
5
9
  includeDevDependenciesTransitively: boolean;
@@ -31,4 +35,7 @@ export interface WorkflowTarget {
31
35
  targetSlug: string;
32
36
  targetPackage: string | null;
33
37
  }
38
+ export interface WorkflowImpact extends WorkflowTarget {
39
+ matchedPaths: string[];
40
+ }
34
41
  //# sourceMappingURL=types.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/workflow/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,wBAAwB;IACvC,oBAAoB,EAAE,MAAM,EAAE,CAAC;IAC/B,gBAAgB,EAAE,MAAM,EAAE,CAAC;IAC3B,oCAAoC,EAAE,OAAO,CAAC;IAC9C,kCAAkC,EAAE,OAAO,CAAC;CAC7C;AAED,MAAM,WAAW,uBAAuB;IACtC,IAAI,EACA,SAAS,GACT,aAAa,GACb,gBAAgB,GAChB,aAAa,GACb,cAAc,CAAC;IACnB,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED,MAAM,WAAW,wBAAwB;IACvC,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,KAAK,EAAE,OAAO,CAAC;IACf,aAAa,EAAE,MAAM,EAAE,CAAC;IACxB,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,MAAM,EAAE,uBAAuB,EAAE,CAAC;CACnC;AAED,MAAM,WAAW,cAAc;IAC7B,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,SAAS,EAAE,MAAM,EAAE,CAAC;IACpB,gBAAgB,EAAE,MAAM,EAAE,CAAC;IAC3B,KAAK,EAAE,MAAM,EAAE,CAAC;CACjB;AAED,MAAM,WAAW,cAAc;IAC7B,YAAY,EAAE,MAAM,CAAC;IACrB,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,MAAM,CAAC;IACnB,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;CAC9B"}
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/workflow/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,wBAAwB;IACvC,oBAAoB,EAAE,MAAM,EAAE,CAAC;IAC/B;;;OAGG;IACH,gBAAgB,EAAE,MAAM,EAAE,CAAC;IAC3B,oCAAoC,EAAE,OAAO,CAAC;IAC9C,kCAAkC,EAAE,OAAO,CAAC;CAC7C;AAED,MAAM,WAAW,uBAAuB;IACtC,IAAI,EACA,SAAS,GACT,aAAa,GACb,gBAAgB,GAChB,aAAa,GACb,cAAc,CAAC;IACnB,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED,MAAM,WAAW,wBAAwB;IACvC,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,KAAK,EAAE,OAAO,CAAC;IACf,aAAa,EAAE,MAAM,EAAE,CAAC;IACxB,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,MAAM,EAAE,uBAAuB,EAAE,CAAC;CACnC;AAED,MAAM,WAAW,cAAc;IAC7B,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,SAAS,EAAE,MAAM,EAAE,CAAC;IACpB,gBAAgB,EAAE,MAAM,EAAE,CAAC;IAC3B,KAAK,EAAE,MAAM,EAAE,CAAC;CACjB;AAED,MAAM,WAAW,cAAc;IAC7B,YAAY,EAAE,MAAM,CAAC;IACrB,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,MAAM,CAAC;IACnB,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;CAC9B;AAED,MAAM,WAAW,cAAe,SAAQ,cAAc;IACpD,YAAY,EAAE,MAAM,EAAE,CAAC;CACxB"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@crossplatformai/dependency-graph",
3
3
  "description": "Workspace dependency graph tooling for CrossPlatform.ai projects.",
4
- "version": "0.10.0",
4
+ "version": "0.11.0-next.0",
5
5
  "bin": {
6
6
  "dependency-graph": "./dist/index-cli.js"
7
7
  },
@@ -22,7 +22,7 @@
22
22
  "typescript": "~5.9.3",
23
23
  "typescript-eslint": "^8.54.0",
24
24
  "vitest": "^4.0.17",
25
- "@crossplatformai/typescript-config": "0.7.0"
25
+ "@crossplatformai/typescript-config": "0.7.1-next.0"
26
26
  },
27
27
  "exports": {
28
28
  ".": {
@@ -50,12 +50,10 @@
50
50
  "types": "./dist/index.d.ts",
51
51
  "scripts": {
52
52
  "build": "tsup && tsc -p tsconfig.build.json --emitDeclarationOnly",
53
- "check-types": "tsc --noEmit --skipLibCheck",
54
53
  "lint": "eslint . --cache --fix --max-warnings 0",
55
- "publish:latest": "node ../../scripts/publish-release.mjs latest",
56
- "publish:next": "node ../../scripts/publish-release.mjs next",
57
- "publish:preview": "node ../../scripts/publish-preview.mjs",
54
+ "release": "node ../../scripts/publish-smart.mjs",
58
55
  "test": "vitest run",
59
- "test:watch": "vitest"
56
+ "test:watch": "vitest",
57
+ "typecheck": "tsc --noEmit --skipLibCheck"
60
58
  }
61
59
  }