Blocking (review id 2514): - [security] Forbid symlinks in vaults. ensureServable now sets core.symlinks=false in each vault's local git config (a pushed symlink is checked out as a plain file, never a real link), and the engine cycle wraps every read/write/mkdir in an lstat/realpath guard (new path-guard.ts) that refuses a path that is — or traverses — a symlink, or whose realpath escapes the vault root. Prevents a writer from publishing /etc/passwd or the server .env, or writing outside the vault. Adds unit tests (path-guard.test.ts) + a read-guard integration test (cycle.test.ts) + real lstat/realpath in the roundtrip integration test. - [simplification] Delete dead lib/diff.ts + test/diff.test.ts and drop the now-unused @fellow/prosemirror-recreate-transform dependency. - [documentation] Add a CHANGELOG [Unreleased] → Added entry for git-sync. Warnings: - [test-coverage] Cover the CREATE-branch conflict-markers guard (a new .md with markers and no gitmost_id is recorded as a create failure, never created). Suggestions: - [stability] Bound each `git config` in ensureServable with a timeout. - [authz] Trigger endpoint resolves spaceId workspace-scoped and 404s a foreign space before any vault directory is created. - [stability] Attribute git-initiated moves to the service account (lastUpdatedById), via an optional actor param on PageService.movePage. - [documentation] Document the per-space autoMergeConflicts toggle in AGENTS.md. - [test-coverage] Cover the unterminated `:::` callout fence fallback. - [simplification] Move test-only roundtrip-helpers.ts out of src/ into test/. Architecture: - Move the Yjs/ProseMirror merge primitives (yjs-body-merge, three-way-merge, lcs + specs) into collaboration/merge/, breaking the collaboration → integrations/git-sync dependency cycle this PR introduced. - Port the schema-surface drift gate to packages/mcp (the mcp schema mirror had none); pins 52 entries. Deferred (with rationale in the review thread): the incremental-pull perf warning (correctness-neutral; needs a high-water-mark design + its own tests on the data-loss-critical path) and the redis-sync rolling-deploy mixed-version edge (the deficient behavior is in already-released old-instance code; the new code is correct on both sides; impact is a transient rollout-window artifact). Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
210 lines
8.1 KiB
TypeScript
210 lines
8.1 KiB
TypeScript
import { describe, it, expect, vi } from "vitest";
|
|
import { runCycle, type RunCycleDeps } from "../src/engine/cycle";
|
|
|
|
// A fake VaultGit recording the staging calls. An EMPTY vault/tree lets the real
|
|
// readExisting/computePullActions/applyPullActions/runPush run trivially (no
|
|
// files, no pages) so we can assert runCycle's choreography without real git.
|
|
function fakeVault(overrides: Record<string, any> = {}) {
|
|
const order: string[] = [];
|
|
const rec =
|
|
(name: string, ret?: any) =>
|
|
async (...args: any[]) => {
|
|
order.push(args.length ? `${name}:${args.join(",")}` : name);
|
|
return ret;
|
|
};
|
|
const vault: any = {
|
|
order,
|
|
assertGitAvailable: rec("assertGitAvailable"),
|
|
ensureRepo: rec("ensureRepo"),
|
|
isMergeInProgress: vi.fn(async () => false),
|
|
ensureBranch: rec("ensureBranch"),
|
|
checkout: rec("checkout"),
|
|
listTrackedFiles: vi.fn(async () => [] as string[]),
|
|
// push-side git surface (empty diff -> a clean no-op push)
|
|
stageAll: rec("stageAll"),
|
|
commit: rec("commit", { committed: false }),
|
|
merge: rec("merge", { ok: true, conflict: false, output: "" }),
|
|
listUnmergedPaths: vi.fn(async () => [] as string[]),
|
|
commitMerge: rec("commitMerge"),
|
|
abortMerge: rec("abortMerge"),
|
|
resetHardToHead: rec("resetHardToHead"),
|
|
readRef: vi.fn(async () => null),
|
|
revParse: vi.fn(async () => "0000000000000000000000000000000000000000"),
|
|
diffNameStatus: vi.fn(async () => [] as any[]),
|
|
showFileAtRef: vi.fn(async () => ""),
|
|
updateRef: rec("updateRef"),
|
|
fastForwardBranch: rec("fastForwardBranch", { ok: true }),
|
|
...overrides,
|
|
};
|
|
return vault;
|
|
}
|
|
|
|
function baseDeps(vault: any, over: Partial<RunCycleDeps> = {}): RunCycleDeps {
|
|
return {
|
|
spaceId: "space-1",
|
|
client: {
|
|
listSpaceTree: vi.fn(async () => ({ pages: [], complete: true })),
|
|
getPageJson: vi.fn(),
|
|
importPageMarkdown: vi.fn(),
|
|
createPage: vi.fn(),
|
|
deletePage: vi.fn(),
|
|
movePage: vi.fn(),
|
|
renamePage: vi.fn(),
|
|
listRecentSince: vi.fn(),
|
|
listTrash: vi.fn(),
|
|
restorePage: vi.fn(),
|
|
} as any,
|
|
vault,
|
|
settings: { vaultPath: "/vault" } as any,
|
|
fs: {
|
|
readFile: vi.fn(async () => ""),
|
|
writeFile: vi.fn(async () => undefined),
|
|
mkdir: vi.fn(async () => undefined),
|
|
rm: vi.fn(async () => undefined),
|
|
// Default: nothing is a symlink and everything resolves in place (no
|
|
// escape). The symlink-guard tests below override these.
|
|
lstat: vi.fn(async () => ({ isSymbolicLink: false })),
|
|
realpath: vi.fn(async (p: string) => p),
|
|
},
|
|
log: vi.fn(),
|
|
...over,
|
|
};
|
|
}
|
|
|
|
describe("runCycle (composition)", () => {
|
|
it("RECOVERS from a vault left mid-merge: aborts the stale merge and continues (no wedge)", async () => {
|
|
// Regression for the WEDGE bug (QA #119): a vault left mid-merge by a prior
|
|
// cycle used to skip the WHOLE space forever. Now the cycle aborts the stale
|
|
// merge and proceeds so the space self-heals.
|
|
let midMerge = true;
|
|
const vault = fakeVault({
|
|
// mid-merge until `abortMerge` clears it (then the cycle continues).
|
|
isMergeInProgress: vi.fn(async () => midMerge),
|
|
abortMerge: vi.fn(async () => {
|
|
midMerge = false;
|
|
}),
|
|
});
|
|
const deps = baseDeps(vault);
|
|
|
|
const res = await runCycle(deps);
|
|
|
|
// The stale merge was aborted and the cycle RAN (no permanent wedge).
|
|
expect(vault.abortMerge).toHaveBeenCalledTimes(1);
|
|
expect(res.ran).toBe(true);
|
|
expect(deps.client.listSpaceTree).toHaveBeenCalledTimes(1);
|
|
expect(vault.order).toContain("checkout:docmost");
|
|
});
|
|
|
|
it("hard-resets when 'merge --abort' cannot clear a stray unmerged index", async () => {
|
|
// abortMerge does NOT clear it (no MERGE_HEAD but stray unmerged entries);
|
|
// the cycle falls back to a hard reset, then proceeds.
|
|
let midMerge = true;
|
|
const vault = fakeVault({
|
|
isMergeInProgress: vi.fn(async () => midMerge),
|
|
abortMerge: vi.fn(async () => undefined), // leaves it mid-merge
|
|
resetHardToHead: vi.fn(async () => {
|
|
midMerge = false;
|
|
}),
|
|
});
|
|
const deps = baseDeps(vault);
|
|
|
|
const res = await runCycle(deps);
|
|
|
|
expect(vault.abortMerge).toHaveBeenCalledTimes(1);
|
|
expect(vault.resetHardToHead).toHaveBeenCalledTimes(1);
|
|
expect(res.ran).toBe(true);
|
|
});
|
|
|
|
it("stages ensureRepo -> ensureBranch(docmost,main) -> checkout(docmost) BEFORE pulling", async () => {
|
|
const vault = fakeVault();
|
|
const deps = baseDeps(vault);
|
|
|
|
const res = await runCycle(deps);
|
|
|
|
expect(res.ran).toBe(true);
|
|
const ensureRepoIdx = vault.order.indexOf("ensureRepo");
|
|
const ensureBranchIdx = vault.order.indexOf("ensureBranch:docmost,main");
|
|
const checkoutIdx = vault.order.indexOf("checkout:docmost");
|
|
expect(ensureRepoIdx).toBeGreaterThanOrEqual(0);
|
|
expect(ensureBranchIdx).toBeGreaterThan(ensureRepoIdx);
|
|
expect(checkoutIdx).toBeGreaterThan(ensureBranchIdx);
|
|
expect(deps.client.listSpaceTree).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it("runs a SINGLE push planning pass (no dry-run; the delete-cap hook is gone)", async () => {
|
|
const vault = fakeVault();
|
|
const deps = baseDeps(vault);
|
|
|
|
const res = await runCycle(deps);
|
|
expect(res.ran).toBe(true);
|
|
// There is exactly one runPush (the apply) — no separate dry-run pass.
|
|
// diffNameStatus is read once per runPush; assert a single planning pass.
|
|
expect(vault.diffNameStatus).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it("throws on a PRE-aborted signal BEFORE applying the pull (first destructive phase)", async () => {
|
|
const vault = fakeVault();
|
|
const controller = new AbortController();
|
|
controller.abort();
|
|
const deps = baseDeps(vault, { signal: controller.signal });
|
|
|
|
await expect(runCycle(deps)).rejects.toThrow();
|
|
|
|
// The signal is checked AFTER planning but BEFORE the first write phase:
|
|
// the tree was listed (planning) but neither destructive phase advanced —
|
|
// no pull merge and no push diff.
|
|
expect(deps.client.listSpaceTree).toHaveBeenCalledTimes(1);
|
|
expect(vault.order).not.toContain("merge:main");
|
|
expect(vault.diffNameStatus).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("SYMLINK GUARD: never reads a tracked .md that is a symlink (no .env/passwd disclosure)", async () => {
|
|
// Security regression (PR #119 review): a writer who pushes `leak.md` as a
|
|
// SYMLINK to a server file (e.g. `.env`) must NOT have its target read and
|
|
// published. readExisting reads each tracked .md to recover its gitmost_id;
|
|
// the guard refuses the symlink BEFORE the raw read, so the target's bytes
|
|
// are never touched and the cycle keeps running for the rest of the space.
|
|
const vault = fakeVault({
|
|
listTrackedFiles: vi.fn(async () => ["leak.md"]),
|
|
});
|
|
const deps = baseDeps(vault);
|
|
const rawReadFile = vi.fn(async () => "GIT_SYNC_SECRET=topsecret");
|
|
deps.fs.readFile = rawReadFile as any;
|
|
// `/vault/leak.md` is reported as a symlink by lstat.
|
|
deps.fs.lstat = vi.fn(async (p: string) =>
|
|
p === "/vault/leak.md"
|
|
? { isSymbolicLink: true }
|
|
: { isSymbolicLink: false },
|
|
) as any;
|
|
|
|
const res = await runCycle(deps);
|
|
|
|
expect(res.ran).toBe(true);
|
|
// The poisoned symlink's target was NEVER read (the guard short-circuited).
|
|
expect(rawReadFile).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("throws BEFORE the push apply when the signal aborts during the pull phase", async () => {
|
|
// Abort mid-cycle: the signal fires while listSpaceTree (the pull read)
|
|
// runs, so the SECOND checkpoint (before runPush) trips and the push apply
|
|
// never starts.
|
|
const controller = new AbortController();
|
|
const vault = fakeVault();
|
|
const deps = baseDeps(vault, {
|
|
signal: controller.signal,
|
|
client: {
|
|
...baseDeps(vault).client,
|
|
listSpaceTree: vi.fn(async () => {
|
|
controller.abort();
|
|
return { pages: [], complete: true };
|
|
}),
|
|
} as any,
|
|
});
|
|
|
|
await expect(runCycle(deps)).rejects.toThrow();
|
|
// Pull planning ran but the push never did (aborted at a checkpoint).
|
|
expect(deps.client.listSpaceTree).toHaveBeenCalledTimes(1);
|
|
expect(vault.diffNameStatus).not.toHaveBeenCalled();
|
|
});
|
|
});
|