duocode 0.1.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.
- package/.env.example +36 -0
- package/LICENSE +21 -0
- package/README.md +52 -0
- package/dist/ast/context.d.ts +16 -0
- package/dist/ast/context.js +37 -0
- package/dist/ast/context.js.map +1 -0
- package/dist/ast/diff.d.ts +27 -0
- package/dist/ast/diff.js +44 -0
- package/dist/ast/diff.js.map +1 -0
- package/dist/ast/locks.d.ts +47 -0
- package/dist/ast/locks.js +88 -0
- package/dist/ast/locks.js.map +1 -0
- package/dist/ast/merge.d.ts +22 -0
- package/dist/ast/merge.js +120 -0
- package/dist/ast/merge.js.map +1 -0
- package/dist/ast/ownership.d.ts +31 -0
- package/dist/ast/ownership.js +111 -0
- package/dist/ast/ownership.js.map +1 -0
- package/dist/ast/parser.d.ts +44 -0
- package/dist/ast/parser.js +134 -0
- package/dist/ast/parser.js.map +1 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +423 -0
- package/dist/cli.js.map +1 -0
- package/dist/commands/doctor.d.ts +5 -0
- package/dist/commands/doctor.js +63 -0
- package/dist/commands/doctor.js.map +1 -0
- package/dist/commands/duo.d.ts +9 -0
- package/dist/commands/duo.js +285 -0
- package/dist/commands/duo.js.map +1 -0
- package/dist/commands/github.d.ts +2 -0
- package/dist/commands/github.js +85 -0
- package/dist/commands/github.js.map +1 -0
- package/dist/commands/init.d.ts +2 -0
- package/dist/commands/init.js +33 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/negotiation.d.ts +2 -0
- package/dist/commands/negotiation.js +160 -0
- package/dist/commands/negotiation.js.map +1 -0
- package/dist/commands/repl_commands.d.ts +26 -0
- package/dist/commands/repl_commands.js +226 -0
- package/dist/commands/repl_commands.js.map +1 -0
- package/dist/commands/shell.d.ts +1 -0
- package/dist/commands/shell.js +110 -0
- package/dist/commands/shell.js.map +1 -0
- package/dist/commands/start.d.ts +2 -0
- package/dist/commands/start.js +231 -0
- package/dist/commands/start.js.map +1 -0
- package/dist/commands/task.d.ts +2 -0
- package/dist/commands/task.js +215 -0
- package/dist/commands/task.js.map +1 -0
- package/dist/config/loader.d.ts +193 -0
- package/dist/config/loader.js +106 -0
- package/dist/config/loader.js.map +1 -0
- package/dist/context/project_context.d.ts +79 -0
- package/dist/context/project_context.js +292 -0
- package/dist/context/project_context.js.map +1 -0
- package/dist/context/token_budget.d.ts +35 -0
- package/dist/context/token_budget.js +81 -0
- package/dist/context/token_budget.js.map +1 -0
- package/dist/db/queries.d.ts +121 -0
- package/dist/db/queries.js +109 -0
- package/dist/db/queries.js.map +1 -0
- package/dist/db/schema.d.ts +110 -0
- package/dist/db/schema.js +346 -0
- package/dist/db/schema.js.map +1 -0
- package/dist/duo/duo_orchestrator.d.ts +50 -0
- package/dist/duo/duo_orchestrator.js +510 -0
- package/dist/duo/duo_orchestrator.js.map +1 -0
- package/dist/duo/duo_session.d.ts +47 -0
- package/dist/duo/duo_session.js +127 -0
- package/dist/duo/duo_session.js.map +1 -0
- package/dist/duo/duo_types.d.ts +168 -0
- package/dist/duo/duo_types.js +53 -0
- package/dist/duo/duo_types.js.map +1 -0
- package/dist/duo/session_store.d.ts +71 -0
- package/dist/duo/session_store.js +177 -0
- package/dist/duo/session_store.js.map +1 -0
- package/dist/git/worktree.d.ts +21 -0
- package/dist/git/worktree.js +86 -0
- package/dist/git/worktree.js.map +1 -0
- package/dist/github/cache.d.ts +23 -0
- package/dist/github/cache.js +67 -0
- package/dist/github/cache.js.map +1 -0
- package/dist/github/issues.d.ts +17 -0
- package/dist/github/issues.js +93 -0
- package/dist/github/issues.js.map +1 -0
- package/dist/github/mcp_client.d.ts +57 -0
- package/dist/github/mcp_client.js +214 -0
- package/dist/github/mcp_client.js.map +1 -0
- package/dist/github/sync.d.ts +11 -0
- package/dist/github/sync.js +65 -0
- package/dist/github/sync.js.map +1 -0
- package/dist/github/webhook.d.ts +25 -0
- package/dist/github/webhook.js +197 -0
- package/dist/github/webhook.js.map +1 -0
- package/dist/negotiation/index.d.ts +1 -0
- package/dist/negotiation/index.js +2 -0
- package/dist/negotiation/index.js.map +1 -0
- package/dist/negotiation/protocol.d.ts +62 -0
- package/dist/negotiation/protocol.js +188 -0
- package/dist/negotiation/protocol.js.map +1 -0
- package/dist/orchestrator/complexity_scorer.d.ts +2 -0
- package/dist/orchestrator/complexity_scorer.js +79 -0
- package/dist/orchestrator/complexity_scorer.js.map +1 -0
- package/dist/orchestrator/dependency_graph.d.ts +7 -0
- package/dist/orchestrator/dependency_graph.js +73 -0
- package/dist/orchestrator/dependency_graph.js.map +1 -0
- package/dist/orchestrator/intent_parser.d.ts +11 -0
- package/dist/orchestrator/intent_parser.js +116 -0
- package/dist/orchestrator/intent_parser.js.map +1 -0
- package/dist/orchestrator/task_runner.d.ts +56 -0
- package/dist/orchestrator/task_runner.js +181 -0
- package/dist/orchestrator/task_runner.js.map +1 -0
- package/dist/orchestrator/types.d.ts +44 -0
- package/dist/orchestrator/types.js +21 -0
- package/dist/orchestrator/types.js.map +1 -0
- package/dist/providers/anthropic.d.ts +12 -0
- package/dist/providers/anthropic.js +258 -0
- package/dist/providers/anthropic.js.map +1 -0
- package/dist/providers/auction.d.ts +42 -0
- package/dist/providers/auction.js +190 -0
- package/dist/providers/auction.js.map +1 -0
- package/dist/providers/base.d.ts +103 -0
- package/dist/providers/base.js +2 -0
- package/dist/providers/base.js.map +1 -0
- package/dist/providers/cost_tracker.d.ts +45 -0
- package/dist/providers/cost_tracker.js +111 -0
- package/dist/providers/cost_tracker.js.map +1 -0
- package/dist/providers/duo_pair_router.d.ts +11 -0
- package/dist/providers/duo_pair_router.js +67 -0
- package/dist/providers/duo_pair_router.js.map +1 -0
- package/dist/providers/factory.d.ts +7 -0
- package/dist/providers/factory.js +130 -0
- package/dist/providers/factory.js.map +1 -0
- package/dist/providers/grading_rubric.d.ts +37 -0
- package/dist/providers/grading_rubric.js +238 -0
- package/dist/providers/grading_rubric.js.map +1 -0
- package/dist/providers/openai.d.ts +12 -0
- package/dist/providers/openai.js +229 -0
- package/dist/providers/openai.js.map +1 -0
- package/dist/providers/openrouter.d.ts +14 -0
- package/dist/providers/openrouter.js +178 -0
- package/dist/providers/openrouter.js.map +1 -0
- package/dist/providers/performance_tracker.d.ts +21 -0
- package/dist/providers/performance_tracker.js +63 -0
- package/dist/providers/performance_tracker.js.map +1 -0
- package/dist/providers/registry_loader.d.ts +6 -0
- package/dist/providers/registry_loader.js +54 -0
- package/dist/providers/registry_loader.js.map +1 -0
- package/dist/providers/retry.d.ts +66 -0
- package/dist/providers/retry.js +203 -0
- package/dist/providers/retry.js.map +1 -0
- package/dist/providers/role_scorer.d.ts +16 -0
- package/dist/providers/role_scorer.js +16 -0
- package/dist/providers/role_scorer.js.map +1 -0
- package/dist/providers/router.d.ts +84 -0
- package/dist/providers/router.js +542 -0
- package/dist/providers/router.js.map +1 -0
- package/dist/security/credentials.d.ts +6 -0
- package/dist/security/credentials.js +16 -0
- package/dist/security/credentials.js.map +1 -0
- package/dist/setup/browser.d.ts +1 -0
- package/dist/setup/browser.js +12 -0
- package/dist/setup/browser.js.map +1 -0
- package/dist/setup/global_config.d.ts +14 -0
- package/dist/setup/global_config.js +54 -0
- package/dist/setup/global_config.js.map +1 -0
- package/dist/setup/wizard.d.ts +2 -0
- package/dist/setup/wizard.js +206 -0
- package/dist/setup/wizard.js.map +1 -0
- package/dist/tools/agent_loop.d.ts +38 -0
- package/dist/tools/agent_loop.js +72 -0
- package/dist/tools/agent_loop.js.map +1 -0
- package/dist/tools/approval.d.ts +64 -0
- package/dist/tools/approval.js +172 -0
- package/dist/tools/approval.js.map +1 -0
- package/dist/tools/checkpoint.d.ts +65 -0
- package/dist/tools/checkpoint.js +342 -0
- package/dist/tools/checkpoint.js.map +1 -0
- package/dist/tools/definitions.d.ts +13 -0
- package/dist/tools/definitions.js +103 -0
- package/dist/tools/definitions.js.map +1 -0
- package/dist/tools/diff_display.d.ts +46 -0
- package/dist/tools/diff_display.js +298 -0
- package/dist/tools/diff_display.js.map +1 -0
- package/dist/tools/executor.d.ts +12 -0
- package/dist/tools/executor.js +340 -0
- package/dist/tools/executor.js.map +1 -0
- package/dist/tools/permissions.d.ts +17 -0
- package/dist/tools/permissions.js +139 -0
- package/dist/tools/permissions.js.map +1 -0
- package/dist/tools/tool_types.d.ts +48 -0
- package/dist/tools/tool_types.js +7 -0
- package/dist/tools/tool_types.js.map +1 -0
- package/dist/ui/banner.d.ts +4 -0
- package/dist/ui/banner.js +104 -0
- package/dist/ui/banner.js.map +1 -0
- package/dist/ui/callbacks.d.ts +30 -0
- package/dist/ui/callbacks.js +132 -0
- package/dist/ui/callbacks.js.map +1 -0
- package/dist/ui/colors.d.ts +14 -0
- package/dist/ui/colors.js +28 -0
- package/dist/ui/colors.js.map +1 -0
- package/dist/ui/dashboard.d.ts +51 -0
- package/dist/ui/dashboard.js +181 -0
- package/dist/ui/dashboard.js.map +1 -0
- package/dist/ui/leaderboard.d.ts +16 -0
- package/dist/ui/leaderboard.js +43 -0
- package/dist/ui/leaderboard.js.map +1 -0
- package/dist/ui/logger.d.ts +28 -0
- package/dist/ui/logger.js +117 -0
- package/dist/ui/logger.js.map +1 -0
- package/dist/ui/progress.d.ts +16 -0
- package/dist/ui/progress.js +62 -0
- package/dist/ui/progress.js.map +1 -0
- package/dist/ui/tokenizer.d.ts +5 -0
- package/dist/ui/tokenizer.js +54 -0
- package/dist/ui/tokenizer.js.map +1 -0
- package/package.json +63 -0
package/.env.example
ADDED
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
# ─── DuoCode Environment Configuration ───
|
|
2
|
+
|
|
3
|
+
# Provider API Keys
|
|
4
|
+
# Preferred: one OpenRouter key for multi-provider routing
|
|
5
|
+
OPENROUTER_API_KEY=sk-or-...
|
|
6
|
+
# Optional fallback provider keys
|
|
7
|
+
OPENAI_API_KEY=sk-...
|
|
8
|
+
ANTHROPIC_API_KEY=sk-ant-...
|
|
9
|
+
|
|
10
|
+
# GitHub Integration
|
|
11
|
+
GITHUB_TOKEN=ghp_...
|
|
12
|
+
GITHUB_WEBHOOK_SECRET=your-webhook-secret-here
|
|
13
|
+
|
|
14
|
+
# MCP Server Configuration
|
|
15
|
+
# The GitHub MCP server command (used by duocode github sync)
|
|
16
|
+
MCP_GITHUB_COMMAND=npx
|
|
17
|
+
MCP_GITHUB_ARGS=-y,@modelcontextprotocol/server-github
|
|
18
|
+
MCP_GITHUB_TIMEOUT_MS=15000
|
|
19
|
+
|
|
20
|
+
# Webhook Server
|
|
21
|
+
WEBHOOK_PORT=3000
|
|
22
|
+
WEBHOOK_HOST=0.0.0.0
|
|
23
|
+
|
|
24
|
+
# Testing
|
|
25
|
+
# Set to "1" to enable smoke tests that require a real git repo
|
|
26
|
+
DUOCODE_SMOKE_TEST=0
|
|
27
|
+
# Set to "1" to enable MCP GitHub smoke connectivity checks
|
|
28
|
+
DUOCODE_MCP_SMOKE=0
|
|
29
|
+
SMOKE_GITHUB_OWNER=your-org
|
|
30
|
+
SMOKE_GITHUB_REPO=your-repo
|
|
31
|
+
|
|
32
|
+
# Database (optional; defaults configured in duocode.yaml)
|
|
33
|
+
# DUOCODE_DB_PATH=.duocode/duocode.db
|
|
34
|
+
|
|
35
|
+
# Budget Limits (optional)
|
|
36
|
+
# DUOCODE_MAX_TOTAL_USD=100
|
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 duocode contributors
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/README.md
ADDED
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
# DuoCode
|
|
2
|
+
|
|
3
|
+
DuoCode is a multi-agent coding orchestration CLI with AST ownership, negotiation, provider routing, and GitHub MCP integration.
|
|
4
|
+
|
|
5
|
+
## Prerequisites
|
|
6
|
+
- Node.js 20+
|
|
7
|
+
- npm
|
|
8
|
+
- Git
|
|
9
|
+
|
|
10
|
+
## Quick Start
|
|
11
|
+
```bash
|
|
12
|
+
npm install
|
|
13
|
+
npm run typecheck
|
|
14
|
+
npm run build
|
|
15
|
+
npm run test
|
|
16
|
+
```
|
|
17
|
+
|
|
18
|
+
## Environment
|
|
19
|
+
Copy `.env.example` values into your shell or `.env`.
|
|
20
|
+
|
|
21
|
+
Important variables:
|
|
22
|
+
- `OPENAI_API_KEY`
|
|
23
|
+
- `ANTHROPIC_API_KEY`
|
|
24
|
+
- `GITHUB_TOKEN`
|
|
25
|
+
- `GITHUB_WEBHOOK_SECRET`
|
|
26
|
+
- `MCP_GITHUB_COMMAND`
|
|
27
|
+
- `MCP_GITHUB_ARGS`
|
|
28
|
+
- `MCP_GITHUB_TIMEOUT_MS`
|
|
29
|
+
|
|
30
|
+
Smoke-test variables:
|
|
31
|
+
- `DUOCODE_SMOKE_TEST=1`
|
|
32
|
+
- `DUOCODE_MCP_SMOKE=1`
|
|
33
|
+
- `SMOKE_GITHUB_OWNER=<owner>`
|
|
34
|
+
- `SMOKE_GITHUB_REPO=<repo>`
|
|
35
|
+
|
|
36
|
+
## Test Stack
|
|
37
|
+
Run full local validation:
|
|
38
|
+
```bash
|
|
39
|
+
./scripts/test-e2e.sh
|
|
40
|
+
```
|
|
41
|
+
|
|
42
|
+
Run strict release gate (includes smoke requirements):
|
|
43
|
+
```bash
|
|
44
|
+
./scripts/release-gate.sh
|
|
45
|
+
```
|
|
46
|
+
|
|
47
|
+
## CLI Commands
|
|
48
|
+
- `duocode init`
|
|
49
|
+
- `duocode task create|list|show`
|
|
50
|
+
- `duocode negotiation run|show`
|
|
51
|
+
- `duocode github sync`
|
|
52
|
+
- `duocode github webhook-serve`
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import type Database from "better-sqlite3";
|
|
2
|
+
export interface SharePairInput {
|
|
3
|
+
taskId: string;
|
|
4
|
+
filePath: string;
|
|
5
|
+
ownershipPaths: string[];
|
|
6
|
+
leftAgentId: string;
|
|
7
|
+
rightAgentId: string;
|
|
8
|
+
leftContext: string;
|
|
9
|
+
rightContext: string;
|
|
10
|
+
}
|
|
11
|
+
export declare class CrossAgentContextHub {
|
|
12
|
+
private readonly db;
|
|
13
|
+
constructor(db: Database.Database);
|
|
14
|
+
shareBetweenAgents(input: SharePairInput): void;
|
|
15
|
+
getContextForAgent(taskId: string, agentId: string, filePath: string): string[];
|
|
16
|
+
}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { createSharedContext, listSharedContextForAgent, } from "../db/schema.js";
|
|
2
|
+
function visibilityCsv(agents) {
|
|
3
|
+
const unique = [...new Set(agents.map((agent) => agent.trim()).filter(Boolean))];
|
|
4
|
+
return `,${unique.join(",")},`;
|
|
5
|
+
}
|
|
6
|
+
export class CrossAgentContextHub {
|
|
7
|
+
db;
|
|
8
|
+
constructor(db) {
|
|
9
|
+
this.db = db;
|
|
10
|
+
}
|
|
11
|
+
shareBetweenAgents(input) {
|
|
12
|
+
const sharedOwnership = input.ownershipPaths.length > 0
|
|
13
|
+
? input.ownershipPaths.join(" | ")
|
|
14
|
+
: null;
|
|
15
|
+
createSharedContext(this.db, {
|
|
16
|
+
taskId: input.taskId,
|
|
17
|
+
filePath: input.filePath,
|
|
18
|
+
ownershipPath: sharedOwnership,
|
|
19
|
+
publisherAgentId: input.leftAgentId,
|
|
20
|
+
content: input.leftContext,
|
|
21
|
+
visibilityCsv: visibilityCsv([input.rightAgentId]),
|
|
22
|
+
});
|
|
23
|
+
createSharedContext(this.db, {
|
|
24
|
+
taskId: input.taskId,
|
|
25
|
+
filePath: input.filePath,
|
|
26
|
+
ownershipPath: sharedOwnership,
|
|
27
|
+
publisherAgentId: input.rightAgentId,
|
|
28
|
+
content: input.rightContext,
|
|
29
|
+
visibilityCsv: visibilityCsv([input.leftAgentId]),
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
getContextForAgent(taskId, agentId, filePath) {
|
|
33
|
+
const rows = listSharedContextForAgent(this.db, taskId, agentId, filePath);
|
|
34
|
+
return rows.map((row) => row.content);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
//# sourceMappingURL=context.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context.js","sourceRoot":"","sources":["../../src/ast/context.ts"],"names":[],"mappings":"AACA,OAAO,EACL,mBAAmB,EACnB,yBAAyB,GAC1B,MAAM,iBAAiB,CAAC;AAYzB,SAAS,aAAa,CAAC,MAAgB;IACrC,MAAM,MAAM,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;IACjF,OAAO,IAAI,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC;AACjC,CAAC;AAED,MAAM,OAAO,oBAAoB;IACF;IAA7B,YAA6B,EAAqB;QAArB,OAAE,GAAF,EAAE,CAAmB;IAAG,CAAC;IAEtD,kBAAkB,CAAC,KAAqB;QACtC,MAAM,eAAe,GAAG,KAAK,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC;YACrD,CAAC,CAAC,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC;YAClC,CAAC,CAAC,IAAI,CAAC;QAET,mBAAmB,CAAC,IAAI,CAAC,EAAE,EAAE;YAC3B,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,aAAa,EAAE,eAAe;YAC9B,gBAAgB,EAAE,KAAK,CAAC,WAAW;YACnC,OAAO,EAAE,KAAK,CAAC,WAAW;YAC1B,aAAa,EAAE,aAAa,CAAC,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;SACnD,CAAC,CAAC;QAEH,mBAAmB,CAAC,IAAI,CAAC,EAAE,EAAE;YAC3B,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,aAAa,EAAE,eAAe;YAC9B,gBAAgB,EAAE,KAAK,CAAC,YAAY;YACpC,OAAO,EAAE,KAAK,CAAC,YAAY;YAC3B,aAAa,EAAE,aAAa,CAAC,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;SAClD,CAAC,CAAC;IACL,CAAC;IAED,kBAAkB,CAAC,MAAc,EAAE,OAAe,EAAE,QAAgB;QAClE,MAAM,IAAI,GAAG,yBAAyB,CAAC,IAAI,CAAC,EAAE,EAAE,MAAM,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;QAC3E,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IACxC,CAAC;CACF"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { type SupportedLanguage } from "./parser.js";
|
|
2
|
+
import { type OwnershipUnit } from "./ownership.js";
|
|
3
|
+
export type AstChangeType = "added" | "removed" | "modified" | "unchanged";
|
|
4
|
+
export interface AstOwnershipChange {
|
|
5
|
+
type: AstChangeType;
|
|
6
|
+
path: string;
|
|
7
|
+
before?: OwnershipUnit;
|
|
8
|
+
after?: OwnershipUnit;
|
|
9
|
+
}
|
|
10
|
+
export interface AstDiffResult {
|
|
11
|
+
language: SupportedLanguage;
|
|
12
|
+
filePath: string;
|
|
13
|
+
changes: AstOwnershipChange[];
|
|
14
|
+
summary: {
|
|
15
|
+
added: number;
|
|
16
|
+
removed: number;
|
|
17
|
+
modified: number;
|
|
18
|
+
unchanged: number;
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
export interface DiffAstParams {
|
|
22
|
+
language: SupportedLanguage;
|
|
23
|
+
filePath: string;
|
|
24
|
+
beforeSource: string;
|
|
25
|
+
afterSource: string;
|
|
26
|
+
}
|
|
27
|
+
export declare function diffAstOwnership(params: DiffAstParams): AstDiffResult;
|
package/dist/ast/diff.js
ADDED
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { MultiLanguageAstParser } from "./parser.js";
|
|
2
|
+
import { createOwnershipSnapshot, ownershipMapByPath, } from "./ownership.js";
|
|
3
|
+
export function diffAstOwnership(params) {
|
|
4
|
+
const parser = new MultiLanguageAstParser();
|
|
5
|
+
const before = createOwnershipSnapshot(parser.parseSource(params.language, params.beforeSource), params.filePath);
|
|
6
|
+
const after = createOwnershipSnapshot(parser.parseSource(params.language, params.afterSource), params.filePath);
|
|
7
|
+
const beforeByPath = ownershipMapByPath(before);
|
|
8
|
+
const afterByPath = ownershipMapByPath(after);
|
|
9
|
+
const allPaths = new Set([...beforeByPath.keys(), ...afterByPath.keys()]);
|
|
10
|
+
const changes = [];
|
|
11
|
+
for (const path of [...allPaths].sort()) {
|
|
12
|
+
const b = beforeByPath.get(path);
|
|
13
|
+
const a = afterByPath.get(path);
|
|
14
|
+
if (!b && a) {
|
|
15
|
+
changes.push({ type: "added", path, after: a });
|
|
16
|
+
continue;
|
|
17
|
+
}
|
|
18
|
+
if (b && !a) {
|
|
19
|
+
changes.push({ type: "removed", path, before: b });
|
|
20
|
+
continue;
|
|
21
|
+
}
|
|
22
|
+
if (b && a) {
|
|
23
|
+
if (b.hash === a.hash) {
|
|
24
|
+
changes.push({ type: "unchanged", path, before: b, after: a });
|
|
25
|
+
}
|
|
26
|
+
else {
|
|
27
|
+
changes.push({ type: "modified", path, before: b, after: a });
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
const summary = {
|
|
32
|
+
added: changes.filter((change) => change.type === "added").length,
|
|
33
|
+
removed: changes.filter((change) => change.type === "removed").length,
|
|
34
|
+
modified: changes.filter((change) => change.type === "modified").length,
|
|
35
|
+
unchanged: changes.filter((change) => change.type === "unchanged").length,
|
|
36
|
+
};
|
|
37
|
+
return {
|
|
38
|
+
language: params.language,
|
|
39
|
+
filePath: params.filePath,
|
|
40
|
+
changes,
|
|
41
|
+
summary,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
//# sourceMappingURL=diff.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"diff.js","sourceRoot":"","sources":["../../src/ast/diff.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,sBAAsB,EAA0B,MAAM,aAAa,CAAC;AAC7E,OAAO,EACL,uBAAuB,EACvB,kBAAkB,GAEnB,MAAM,gBAAgB,CAAC;AA8BxB,MAAM,UAAU,gBAAgB,CAAC,MAAqB;IACpD,MAAM,MAAM,GAAG,IAAI,sBAAsB,EAAE,CAAC;IAC5C,MAAM,MAAM,GAAG,uBAAuB,CACpC,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,YAAY,CAAC,EACxD,MAAM,CAAC,QAAQ,CAChB,CAAC;IACF,MAAM,KAAK,GAAG,uBAAuB,CACnC,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,WAAW,CAAC,EACvD,MAAM,CAAC,QAAQ,CAChB,CAAC;IAEF,MAAM,YAAY,GAAG,kBAAkB,CAAC,MAAM,CAAC,CAAC;IAChD,MAAM,WAAW,GAAG,kBAAkB,CAAC,KAAK,CAAC,CAAC;IAC9C,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,CAAC,GAAG,YAAY,CAAC,IAAI,EAAE,EAAE,GAAG,WAAW,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;IAE1E,MAAM,OAAO,GAAyB,EAAE,CAAC;IAEzC,KAAK,MAAM,IAAI,IAAI,CAAC,GAAG,QAAQ,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC;QACxC,MAAM,CAAC,GAAG,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACjC,MAAM,CAAC,GAAG,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAEhC,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC;YACZ,OAAO,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;YAChD,SAAS;QACX,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC;YACZ,OAAO,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,EAAE,CAAC,CAAC;YACnD,SAAS;QACX,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;YACX,IAAI,CAAC,CAAC,IAAI,KAAK,CAAC,CAAC,IAAI,EAAE,CAAC;gBACtB,OAAO,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;YACjE,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;YAChE,CAAC;QACH,CAAC;IACH,CAAC;IAED,MAAM,OAAO,GAAG;QACd,KAAK,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,MAAM;QACjE,OAAO,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,KAAK,SAAS,CAAC,CAAC,MAAM;QACrE,QAAQ,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,KAAK,UAAU,CAAC,CAAC,MAAM;QACvE,SAAS,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,KAAK,WAAW,CAAC,CAAC,MAAM;KAC1E,CAAC;IAEF,OAAO;QACL,QAAQ,EAAE,MAAM,CAAC,QAAQ;QACzB,QAAQ,EAAE,MAAM,CAAC,QAAQ;QACzB,OAAO;QACP,OAAO;KACR,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
export interface OwnershipLock {
|
|
2
|
+
key: string;
|
|
3
|
+
ownerId: string;
|
|
4
|
+
token: string;
|
|
5
|
+
acquiredAt: number;
|
|
6
|
+
expiresAt: number;
|
|
7
|
+
metadata?: Record<string, string>;
|
|
8
|
+
}
|
|
9
|
+
export interface AcquireLockRequest {
|
|
10
|
+
key: string;
|
|
11
|
+
ownerId: string;
|
|
12
|
+
ttlMs: number;
|
|
13
|
+
metadata?: Record<string, string>;
|
|
14
|
+
now?: number;
|
|
15
|
+
}
|
|
16
|
+
export interface RenewLockRequest {
|
|
17
|
+
key: string;
|
|
18
|
+
token: string;
|
|
19
|
+
ttlMs: number;
|
|
20
|
+
now?: number;
|
|
21
|
+
}
|
|
22
|
+
export interface ReleaseLockRequest {
|
|
23
|
+
key: string;
|
|
24
|
+
token: string;
|
|
25
|
+
}
|
|
26
|
+
export interface OwnershipLockStore {
|
|
27
|
+
get(key: string): Promise<OwnershipLock | null>;
|
|
28
|
+
set(lock: OwnershipLock): Promise<void>;
|
|
29
|
+
delete(key: string): Promise<void>;
|
|
30
|
+
list(): Promise<OwnershipLock[]>;
|
|
31
|
+
}
|
|
32
|
+
export declare class InMemoryOwnershipLockStore implements OwnershipLockStore {
|
|
33
|
+
private readonly locks;
|
|
34
|
+
get(key: string): Promise<OwnershipLock | null>;
|
|
35
|
+
set(lock: OwnershipLock): Promise<void>;
|
|
36
|
+
delete(key: string): Promise<void>;
|
|
37
|
+
list(): Promise<OwnershipLock[]>;
|
|
38
|
+
}
|
|
39
|
+
export declare class DistributedOwnershipLockManager {
|
|
40
|
+
private readonly store;
|
|
41
|
+
constructor(store: OwnershipLockStore);
|
|
42
|
+
acquire(request: AcquireLockRequest): Promise<OwnershipLock | null>;
|
|
43
|
+
renew(request: RenewLockRequest): Promise<OwnershipLock | null>;
|
|
44
|
+
release(request: ReleaseLockRequest): Promise<boolean>;
|
|
45
|
+
listActive(now?: number): Promise<OwnershipLock[]>;
|
|
46
|
+
clearExpired(now?: number): Promise<number>;
|
|
47
|
+
}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
export class InMemoryOwnershipLockStore {
|
|
2
|
+
locks = new Map();
|
|
3
|
+
async get(key) {
|
|
4
|
+
return this.locks.get(key) ?? null;
|
|
5
|
+
}
|
|
6
|
+
async set(lock) {
|
|
7
|
+
this.locks.set(lock.key, lock);
|
|
8
|
+
}
|
|
9
|
+
async delete(key) {
|
|
10
|
+
this.locks.delete(key);
|
|
11
|
+
}
|
|
12
|
+
async list() {
|
|
13
|
+
return [...this.locks.values()];
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
function makeToken() {
|
|
17
|
+
return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 10)}`;
|
|
18
|
+
}
|
|
19
|
+
export class DistributedOwnershipLockManager {
|
|
20
|
+
store;
|
|
21
|
+
constructor(store) {
|
|
22
|
+
this.store = store;
|
|
23
|
+
}
|
|
24
|
+
async acquire(request) {
|
|
25
|
+
const now = request.now ?? Date.now();
|
|
26
|
+
const existing = await this.store.get(request.key);
|
|
27
|
+
if (existing && existing.expiresAt > now) {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
const lock = {
|
|
31
|
+
key: request.key,
|
|
32
|
+
ownerId: request.ownerId,
|
|
33
|
+
token: makeToken(),
|
|
34
|
+
acquiredAt: now,
|
|
35
|
+
expiresAt: now + request.ttlMs,
|
|
36
|
+
metadata: request.metadata,
|
|
37
|
+
};
|
|
38
|
+
await this.store.set(lock);
|
|
39
|
+
return lock;
|
|
40
|
+
}
|
|
41
|
+
async renew(request) {
|
|
42
|
+
const now = request.now ?? Date.now();
|
|
43
|
+
const existing = await this.store.get(request.key);
|
|
44
|
+
if (!existing) {
|
|
45
|
+
return null;
|
|
46
|
+
}
|
|
47
|
+
if (existing.token !== request.token) {
|
|
48
|
+
return null;
|
|
49
|
+
}
|
|
50
|
+
if (existing.expiresAt <= now) {
|
|
51
|
+
await this.store.delete(request.key);
|
|
52
|
+
return null;
|
|
53
|
+
}
|
|
54
|
+
const renewed = {
|
|
55
|
+
...existing,
|
|
56
|
+
expiresAt: now + request.ttlMs,
|
|
57
|
+
};
|
|
58
|
+
await this.store.set(renewed);
|
|
59
|
+
return renewed;
|
|
60
|
+
}
|
|
61
|
+
async release(request) {
|
|
62
|
+
const existing = await this.store.get(request.key);
|
|
63
|
+
if (!existing) {
|
|
64
|
+
return false;
|
|
65
|
+
}
|
|
66
|
+
if (existing.token !== request.token) {
|
|
67
|
+
return false;
|
|
68
|
+
}
|
|
69
|
+
await this.store.delete(request.key);
|
|
70
|
+
return true;
|
|
71
|
+
}
|
|
72
|
+
async listActive(now = Date.now()) {
|
|
73
|
+
const locks = await this.store.list();
|
|
74
|
+
return locks.filter((lock) => lock.expiresAt > now);
|
|
75
|
+
}
|
|
76
|
+
async clearExpired(now = Date.now()) {
|
|
77
|
+
const locks = await this.store.list();
|
|
78
|
+
let cleared = 0;
|
|
79
|
+
for (const lock of locks) {
|
|
80
|
+
if (lock.expiresAt <= now) {
|
|
81
|
+
await this.store.delete(lock.key);
|
|
82
|
+
cleared += 1;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
return cleared;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
//# sourceMappingURL=locks.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"locks.js","sourceRoot":"","sources":["../../src/ast/locks.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { type SupportedLanguage } from "./parser.js";
|
|
2
|
+
import { type OwnershipUnit } from "./ownership.js";
|
|
3
|
+
export interface MergeConflict {
|
|
4
|
+
path: string;
|
|
5
|
+
reason: string;
|
|
6
|
+
base?: OwnershipUnit;
|
|
7
|
+
left?: OwnershipUnit;
|
|
8
|
+
right?: OwnershipUnit;
|
|
9
|
+
}
|
|
10
|
+
export interface AstMergeResult {
|
|
11
|
+
mergedSource: string;
|
|
12
|
+
conflicts: MergeConflict[];
|
|
13
|
+
appliedChanges: string[];
|
|
14
|
+
}
|
|
15
|
+
export interface MergeAstParams {
|
|
16
|
+
language: SupportedLanguage;
|
|
17
|
+
filePath: string;
|
|
18
|
+
baseSource: string;
|
|
19
|
+
leftSource: string;
|
|
20
|
+
rightSource: string;
|
|
21
|
+
}
|
|
22
|
+
export declare function mergeAstThreeWay(params: MergeAstParams): AstMergeResult;
|
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
import { MultiLanguageAstParser } from "./parser.js";
|
|
2
|
+
import { createOwnershipSnapshot, ownershipMapByPath, } from "./ownership.js";
|
|
3
|
+
function isSame(a, b) {
|
|
4
|
+
if (!a && !b) {
|
|
5
|
+
return true;
|
|
6
|
+
}
|
|
7
|
+
if (!a || !b) {
|
|
8
|
+
return false;
|
|
9
|
+
}
|
|
10
|
+
return a.hash === b.hash;
|
|
11
|
+
}
|
|
12
|
+
function changedFromBase(base, candidate) {
|
|
13
|
+
return !isSame(base, candidate);
|
|
14
|
+
}
|
|
15
|
+
function applyTextEdits(baseSource, edits) {
|
|
16
|
+
const ordered = [...edits].sort((a, b) => b.start - a.start);
|
|
17
|
+
let output = baseSource;
|
|
18
|
+
for (const edit of ordered) {
|
|
19
|
+
output = `${output.slice(0, edit.start)}${edit.text}${output.slice(edit.end)}`;
|
|
20
|
+
}
|
|
21
|
+
return output;
|
|
22
|
+
}
|
|
23
|
+
export function mergeAstThreeWay(params) {
|
|
24
|
+
const parser = new MultiLanguageAstParser();
|
|
25
|
+
const base = createOwnershipSnapshot(parser.parseSource(params.language, params.baseSource), params.filePath);
|
|
26
|
+
const left = createOwnershipSnapshot(parser.parseSource(params.language, params.leftSource), params.filePath);
|
|
27
|
+
const right = createOwnershipSnapshot(parser.parseSource(params.language, params.rightSource), params.filePath);
|
|
28
|
+
const baseByPath = ownershipMapByPath(base);
|
|
29
|
+
const leftByPath = ownershipMapByPath(left);
|
|
30
|
+
const rightByPath = ownershipMapByPath(right);
|
|
31
|
+
const allPaths = [...new Set([
|
|
32
|
+
...baseByPath.keys(),
|
|
33
|
+
...leftByPath.keys(),
|
|
34
|
+
...rightByPath.keys(),
|
|
35
|
+
])].sort();
|
|
36
|
+
const edits = [];
|
|
37
|
+
const appendBlocks = [];
|
|
38
|
+
const conflicts = [];
|
|
39
|
+
const appliedChanges = [];
|
|
40
|
+
for (const path of allPaths) {
|
|
41
|
+
const b = baseByPath.get(path);
|
|
42
|
+
const l = leftByPath.get(path);
|
|
43
|
+
const r = rightByPath.get(path);
|
|
44
|
+
const leftChanged = changedFromBase(b, l);
|
|
45
|
+
const rightChanged = changedFromBase(b, r);
|
|
46
|
+
if (!leftChanged && !rightChanged) {
|
|
47
|
+
continue;
|
|
48
|
+
}
|
|
49
|
+
if (leftChanged && !rightChanged) {
|
|
50
|
+
if (b && !l) {
|
|
51
|
+
edits.push({ start: b.range.startIndex, end: b.range.endIndex, text: "", path });
|
|
52
|
+
appliedChanges.push(`${path}: removed (left)`);
|
|
53
|
+
continue;
|
|
54
|
+
}
|
|
55
|
+
if (!b && l) {
|
|
56
|
+
appendBlocks.push(l.text);
|
|
57
|
+
appliedChanges.push(`${path}: added (left)`);
|
|
58
|
+
continue;
|
|
59
|
+
}
|
|
60
|
+
if (b && l) {
|
|
61
|
+
edits.push({ start: b.range.startIndex, end: b.range.endIndex, text: l.text, path });
|
|
62
|
+
appliedChanges.push(`${path}: modified (left)`);
|
|
63
|
+
continue;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
if (!leftChanged && rightChanged) {
|
|
67
|
+
if (b && !r) {
|
|
68
|
+
edits.push({ start: b.range.startIndex, end: b.range.endIndex, text: "", path });
|
|
69
|
+
appliedChanges.push(`${path}: removed (right)`);
|
|
70
|
+
continue;
|
|
71
|
+
}
|
|
72
|
+
if (!b && r) {
|
|
73
|
+
appendBlocks.push(r.text);
|
|
74
|
+
appliedChanges.push(`${path}: added (right)`);
|
|
75
|
+
continue;
|
|
76
|
+
}
|
|
77
|
+
if (b && r) {
|
|
78
|
+
edits.push({ start: b.range.startIndex, end: b.range.endIndex, text: r.text, path });
|
|
79
|
+
appliedChanges.push(`${path}: modified (right)`);
|
|
80
|
+
continue;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
// Both sides changed from base.
|
|
84
|
+
if (isSame(l, r)) {
|
|
85
|
+
if (b && !l) {
|
|
86
|
+
edits.push({ start: b.range.startIndex, end: b.range.endIndex, text: "", path });
|
|
87
|
+
appliedChanges.push(`${path}: removed (both)`);
|
|
88
|
+
continue;
|
|
89
|
+
}
|
|
90
|
+
if (!b && l) {
|
|
91
|
+
appendBlocks.push(l.text);
|
|
92
|
+
appliedChanges.push(`${path}: added (both)`);
|
|
93
|
+
continue;
|
|
94
|
+
}
|
|
95
|
+
if (b && l) {
|
|
96
|
+
edits.push({ start: b.range.startIndex, end: b.range.endIndex, text: l.text, path });
|
|
97
|
+
appliedChanges.push(`${path}: modified (both)`);
|
|
98
|
+
continue;
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
conflicts.push({
|
|
102
|
+
path,
|
|
103
|
+
reason: "Both sides changed this ownership node differently",
|
|
104
|
+
base: b,
|
|
105
|
+
left: l,
|
|
106
|
+
right: r,
|
|
107
|
+
});
|
|
108
|
+
}
|
|
109
|
+
let mergedSource = applyTextEdits(params.baseSource, edits);
|
|
110
|
+
if (appendBlocks.length > 0) {
|
|
111
|
+
const suffix = appendBlocks.join("\n\n");
|
|
112
|
+
mergedSource = mergedSource.trimEnd() + "\n\n" + suffix + "\n";
|
|
113
|
+
}
|
|
114
|
+
return {
|
|
115
|
+
mergedSource,
|
|
116
|
+
conflicts,
|
|
117
|
+
appliedChanges,
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
//# sourceMappingURL=merge.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"merge.js","sourceRoot":"","sources":["../../src/ast/merge.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { type AstParseResult, type SupportedLanguage } from "./parser.js";
|
|
2
|
+
export interface OwnershipRange {
|
|
3
|
+
startIndex: number;
|
|
4
|
+
endIndex: number;
|
|
5
|
+
startLine: number;
|
|
6
|
+
startColumn: number;
|
|
7
|
+
endLine: number;
|
|
8
|
+
endColumn: number;
|
|
9
|
+
}
|
|
10
|
+
export interface OwnershipUnit {
|
|
11
|
+
id: string;
|
|
12
|
+
language: SupportedLanguage;
|
|
13
|
+
filePath: string;
|
|
14
|
+
nodeType: string;
|
|
15
|
+
name: string;
|
|
16
|
+
path: string;
|
|
17
|
+
parentPath: string | null;
|
|
18
|
+
range: OwnershipRange;
|
|
19
|
+
hash: string;
|
|
20
|
+
text: string;
|
|
21
|
+
}
|
|
22
|
+
export interface OwnershipSnapshot {
|
|
23
|
+
language: SupportedLanguage;
|
|
24
|
+
filePath: string;
|
|
25
|
+
units: OwnershipUnit[];
|
|
26
|
+
}
|
|
27
|
+
export declare function extractOwnershipUnits(parseResult: AstParseResult, filePath: string): OwnershipUnit[];
|
|
28
|
+
export declare function createOwnershipSnapshot(parseResult: AstParseResult, filePath: string): OwnershipSnapshot;
|
|
29
|
+
export declare function serializeOwnershipSnapshot(snapshot: OwnershipSnapshot): string;
|
|
30
|
+
export declare function deserializeOwnershipSnapshot(serialized: string): OwnershipSnapshot;
|
|
31
|
+
export declare function ownershipMapByPath(snapshot: OwnershipSnapshot): Map<string, OwnershipUnit>;
|