@kodus/kodus-graph 0.2.2 → 0.2.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@kodus/kodus-graph",
3
- "version": "0.2.2",
3
+ "version": "0.2.3",
4
4
  "description": "Code graph builder for Kodus code review — parses source code into structural graphs with nodes, edges, and analysis",
5
5
  "type": "module",
6
6
  "bin": {
@@ -50,7 +50,24 @@ export async function executeContext(opts: ContextOptions): Promise<void> {
50
50
  process.stderr.write(`Error: Invalid graph JSON: ${validated.error.message}\n`);
51
51
  process.exit(1);
52
52
  }
53
- oldGraph = { nodes: validated.data.nodes, edges: validated.data.edges };
53
+ const changedSet = new Set(opts.files);
54
+ const sameBranch = detectSameBranch(validated.data.nodes, parseResult.nodes, changedSet);
55
+
56
+ if (sameBranch) {
57
+ // --graph was built from the same commit (e.g. kodus-ai's parse --all on PR branch).
58
+ // Exclude changed files from oldGraph so diff detects their functions as "added"
59
+ // instead of falsely marking everything "unchanged".
60
+ oldGraph = {
61
+ nodes: validated.data.nodes.filter((n: { file_path: string }) => !changedSet.has(n.file_path)),
62
+ edges: validated.data.edges.filter((e: { file_path: string }) => !changedSet.has(e.file_path)),
63
+ };
64
+ log.debug('Same-branch detected: excluding changed files from baseline', {
65
+ changedFiles: opts.files.length,
66
+ });
67
+ } else {
68
+ oldGraph = { nodes: validated.data.nodes, edges: validated.data.edges };
69
+ }
70
+
54
71
  const mainGraph: MainGraphInput = {
55
72
  repo_id: '',
56
73
  sha: '',
@@ -84,3 +101,39 @@ export async function executeContext(opts: ContextOptions): Promise<void> {
84
101
  }
85
102
  }
86
103
  }
104
+
105
+ /**
106
+ * Detect if --graph was built from the same commit as the current repo.
107
+ * Compares file_hash values for changed files between the graph and the fresh parse.
108
+ * When hashes match, the graph can't serve as a baseline for diff — it IS the new state.
109
+ */
110
+ function detectSameBranch(
111
+ graphNodes: Array<{ file_path: string; file_hash: string }>,
112
+ parseNodes: Array<{ file_path: string; file_hash: string }>,
113
+ changedFiles: Set<string>,
114
+ ): boolean {
115
+ const graphHashes = new Map<string, string>();
116
+ for (const n of graphNodes) {
117
+ if (changedFiles.has(n.file_path) && n.file_hash && !graphHashes.has(n.file_path)) {
118
+ graphHashes.set(n.file_path, n.file_hash);
119
+ }
120
+ }
121
+
122
+ // No overlap means graph has no nodes for changed files — not same-branch scenario
123
+ if (graphHashes.size === 0) return false;
124
+
125
+ const parseHashes = new Map<string, string>();
126
+ for (const n of parseNodes) {
127
+ if (n.file_hash && !parseHashes.has(n.file_path)) {
128
+ parseHashes.set(n.file_path, n.file_hash);
129
+ }
130
+ }
131
+
132
+ // If any overlapping file has different hash → different branch
133
+ for (const [file, hash] of graphHashes) {
134
+ const parseHash = parseHashes.get(file);
135
+ if (parseHash && parseHash !== hash) return false;
136
+ }
137
+
138
+ return true;
139
+ }