Skip to content

Preserve asset access failure causes#3342

Merged
juliusmarminge merged 3 commits into
mainfrom
codex/preserve-asset-access-causes
Jun 20, 2026
Merged

Preserve asset access failure causes#3342
juliusmarminge merged 3 commits into
mainfrom
codex/preserve-asset-access-causes

Conversation

@juliusmarminge

@juliusmarminge juliusmarminge commented Jun 20, 2026

Copy link
Copy Markdown
Member

Summary

  • preserve non-missing filesystem failures while resolving canonical asset paths
  • attach exact causes to issuance errors and log causes at read-side null boundaries
  • remove the valueless AssetAccessError constructor wrapper

Verification

  • vp test apps/server/src/assets/AssetAccess.test.ts
  • vp check (passes with 20 pre-existing warnings)
  • vp run typecheck

Note

Medium Risk
Changes error semantics for asset URL creation and favicon resolution (real FS failures now fail the Effect with structured errors instead of null), which affects RPC clients and observability paths around signed asset access.

Overview
Asset URL issuance and resolution now return AssetAccessError values with required operation and resource fields (contract in packages/contracts/src/assets.ts), plus stable messages and optional cause, instead of a generic helper. Workspace-file issuance maps path validation, canonical inspection, favicon, signing key, and related steps to named operations; ws.ts aligns workspace-context failures with the same shape.

Filesystem handling adds optionOnNotFound so only NotFound is treated as “missing”; permission and other **PlatformError**s propagate. issueAssetUrl surfaces non-missing failures during canonical workspace inspection (e.g. realPath) as inspect-workspace-asset errors with the original cause. resolveAsset still returns null at the HTTP boundary but logs causes via resolveCanonicalWorkspaceFileForRequest and attachment stat failures.

Project favicon introduces ProjectFaviconResolutionError with stage-specific operation values; resolvePath fails on bad workspace normalization, stat/read errors, and path resolution failures instead of collapsing them to null, while outside-root icon hrefs are skipped and search continues. Tests cover structured errors and preserved causes for asset issuance and favicon resolution.

Reviewed by Cursor Bugbot for commit 639917f. Bugbot is set up for automated code reviews on this repo. Configure here.

Note

Add structured operation and resource fields to AssetAccessError and preserve failure causes

  • Extends AssetAccessError in assets.ts with required operation (typed union of all operation labels) and resource fields, replacing generic failures with structured context.
  • Introduces ProjectFaviconResolutionError in ProjectFaviconResolver.ts with an operation field covering normalization, path resolution, stat, and read failures; previously these cases returned null silently.
  • Adds optionOnNotFound helpers in both AssetAccess.ts and ProjectFaviconResolver.ts to convert PlatformError.NotFound into Option.none while re-throwing other PlatformError cases as structured errors.
  • Updates ws.ts to include operation and resource when constructing AssetAccessError for workspace context resolution failures.
  • Behavioral Change: non-NotFound filesystem errors that were previously swallowed or returned as null now propagate as typed errors with contextual fields.

Macroscope summarized 639917f.

@juliusmarminge juliusmarminge added the vouch:trusted PR author is trusted by repo permissions or the VOUCHED list. label Jun 20, 2026
@coderabbitai

coderabbitai Bot commented Jun 20, 2026

Copy link
Copy Markdown

Important

Review skipped

Auto reviews are disabled on this repository. Please check the settings in the CodeRabbit UI or the .coderabbit.yaml file in this repository. To trigger a single review, invoke the @coderabbitai review command.

⚙️ Run configuration

Configuration used: Repository UI

Review profile: CHILL

Plan: Pro

Run ID: 5614b2af-aac2-4c3b-b566-b953f326e3ee

You can disable this status message by setting the reviews.review_status to false in the CodeRabbit configuration file.

Use the checkbox below for a quick retry:

  • 🔍 Trigger review
✨ Finishing Touches
🧪 Generate unit tests (beta)
  • Create PR with unit tests
  • Commit unit tests in branch codex/preserve-asset-access-causes

Comment @coderabbitai help to get the list of available commands and usage tips.

@github-actions github-actions Bot added the size:L 100-499 changed lines (additions + deletions). label Jun 20, 2026
@macroscopeapp

macroscopeapp Bot commented Jun 20, 2026

Copy link
Copy Markdown
Contributor

Approvability

Verdict: Approved

This PR improves error handling by preserving structured error causes rather than silently swallowing all filesystem errors. The changes are well-tested, improve debuggability, and the author is the primary owner of the modified files.

You can customize Macroscope's approvability policy. Learn more.

@cursor cursor Bot left a comment

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cursor Bugbot has reviewed your changes using high effort and found 1 potential issue.

Fix All in Cursor

Bugbot Autofix prepared a fix for the issue found in the latest run.

  • ✅ Fixed: Bad icon href blocks later sources
    • Changed findExistingFile to skip candidates where resolveRelativePathWithinRoot fails (using Effect.orElseSucceed with Option.none) instead of propagating the error, restoring the scan-order behavior so later source files are still consulted.

Create PR

Or push these changes by commenting:

@cursor push 7ace30344c
Preview (7ace30344c)
diff --git a/apps/server/src/project/ProjectFaviconResolver.test.ts b/apps/server/src/project/ProjectFaviconResolver.test.ts
--- a/apps/server/src/project/ProjectFaviconResolver.test.ts
+++ b/apps/server/src/project/ProjectFaviconResolver.test.ts
@@ -167,22 +167,31 @@
       }),
     );
 
-    it.effect("rejects icon metadata paths outside the workspace", () =>
+    it.effect("skips icon metadata paths outside the workspace", () =>
       Effect.gen(function* () {
         const resolver = yield* ProjectFaviconResolver.ProjectFaviconResolver;
         const cwd = yield* makeTempDir;
         yield* writeTextFile(cwd, "index.html", '<link rel="icon" href="../../secret.svg">');
 
-        const error = yield* resolver.resolvePath(cwd).pipe(Effect.flip);
+        const resolved = yield* resolver.resolvePath(cwd);
 
-        expect(error).toMatchObject({
-          _tag: "ProjectFaviconResolutionError",
-          operation: "resolve-path",
-          workspaceRoot: cwd,
-          relativePath: "../secret.svg",
-        });
-        expect(error.cause).toBeInstanceOf(WorkspacePaths.WorkspacePathOutsideRootError);
+        expect(resolved).toBeNull();
       }),
     );
+
+    it.effect("skips outside-root href and finds valid icon from later source", () =>
+      Effect.gen(function* () {
+        const resolver = yield* ProjectFaviconResolver.ProjectFaviconResolver;
+        const cwd = yield* makeTempDir;
+        yield* writeTextFile(cwd, "index.html", '<link rel="icon" href="../../secret.svg">');
+        yield* writeTextFile(cwd, "public/index.html", '<link rel="icon" href="/brand/logo.svg">');
+        yield* writeTextFile(cwd, "public/brand/logo.svg", "<svg>brand</svg>");
+
+        const resolved = yield* resolver.resolvePath(cwd);
+
+        expect(resolved).not.toBeNull();
+        expect(resolved).toContain("public/brand/logo.svg");
+      }),
+    );
   });
 });

diff --git a/apps/server/src/project/ProjectFaviconResolver.ts b/apps/server/src/project/ProjectFaviconResolver.ts
--- a/apps/server/src/project/ProjectFaviconResolver.ts
+++ b/apps/server/src/project/ProjectFaviconResolver.ts
@@ -134,30 +134,26 @@
           relativePath,
         })
         .pipe(
-          Effect.mapError(
-            (cause) =>
-              new ProjectFaviconResolutionError({
-                operation: "resolve-path",
-                workspaceRoot: projectCwd,
-                relativePath,
-                cause,
-              }),
-          ),
+          Effect.map(Option.some),
+          Effect.orElseSucceed(() => Option.none<{ absolutePath: string; relativePath: string }>()),
         );
-      const stats = yield* optionOnNotFound(fileSystem.stat(candidate.absolutePath)).pipe(
+      if (Option.isNone(candidate)) {
+        continue;
+      }
+      const stats = yield* optionOnNotFound(fileSystem.stat(candidate.value.absolutePath)).pipe(
         Effect.mapError(
           (cause) =>
             new ProjectFaviconResolutionError({
               operation: "stat-candidate",
               workspaceRoot: projectCwd,
               relativePath,
-              absolutePath: candidate.absolutePath,
+              absolutePath: candidate.value.absolutePath,
               cause,
             }),
         ),
       );
       if (Option.isSome(stats) && stats.value.type === "File") {
-        return candidate.absolutePath;
+        return candidate.value.absolutePath;
       }
     }
     return null;

You can send follow-ups to the cloud agent here.

Reviewed by Cursor Bugbot for commit 26f8074. Configure here.

Comment thread apps/server/src/project/ProjectFaviconResolver.ts
@juliusmarminge juliusmarminge force-pushed the codex/preserve-asset-access-causes branch from 26f8074 to 5d5fb30 Compare June 20, 2026 22:45
juliusmarminge and others added 2 commits June 20, 2026 15:51
Co-authored-by: codex <codex@users.noreply.github.com>
Co-authored-by: codex <codex@users.noreply.github.com>
@juliusmarminge juliusmarminge force-pushed the codex/preserve-asset-access-causes branch from 5d5fb30 to 09ead13 Compare June 20, 2026 22:51
Comment thread apps/server/src/project/ProjectFaviconResolver.ts
Co-authored-by: codex <codex@users.noreply.github.com>
@juliusmarminge juliusmarminge merged commit 90dc76b into main Jun 20, 2026
16 checks passed
@juliusmarminge juliusmarminge deleted the codex/preserve-asset-access-causes branch June 20, 2026 23:15
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

size:L 100-499 changed lines (additions + deletions). vouch:trusted PR author is trusted by repo permissions or the VOUCHED list.

Projects

None yet

Development

Successfully merging this pull request may close these issues.

1 participant