Skip to content

Refactor: cleaner top-level directory structure#138

Merged
pbakaus merged 9 commits into
mainfrom
refactor/directory-structure
May 4, 2026
Merged

Refactor: cleaner top-level directory structure#138
pbakaus merged 9 commits into
mainfrom
refactor/directory-structure

Conversation

@pbakaus
Copy link
Copy Markdown
Owner

@pbakaus pbakaus commented May 4, 2026

Summary

Two-phase top-level directory restructure to kill the worst confusions in the repo layout. Three pain points the user named, plus a few I noticed along the way:

  • site/content/ and content/site/ are two different trees for the same files (CLAUDE.md acknowledged the dual-tree as cleanup debt).
  • source/ (skill source) and src/ (CLI source) are visually similar and easy to confuse.
  • Top-level docs/ (internal ADRs and plans) clashed with the site's /docs route.
  • CLI was scattered across bin/, src/, and lib/. Site infra was scattered across site/, public/, and astro.config.mjs.

What changed

Phase 1 — three commits:

  • e952f31 Merge content/site/ into site/content/. Move anti-patterns-catalog.js to site/data/.
  • d1f7553 Rename source/skills/impeccable/skill/. Drop the redundant skills/impeccable/ nesting and rewrite readSourceFiles for the flat single-skill layout.
  • d5b576d Rename internal docs/notes/ to remove the clash with the site's /docs route.

Phase 2 — two commits:

  • c249335 Move public/site/public/ (Astro convention). Configure publicDir in astro.config.mjs. Update generator paths in scripts/build.js, scripts/generate-og-image.js, scripts/screenshot-antipatterns.js, and scripts/lib/sub-pages-data.js. Refresh .gitignore for the relocated build artifacts. Delete the dead _REMOVED() Bun static-site builder. Refresh CLAUDE.md (CSS section, generated-pages section, command-add checklist) for the post-Astro layout.
  • db9b46a Consolidate bin/ + src/ + lib/ under cli/: cli/bin/ (entry + sub-commands), cli/engine/ (detect-antipatterns engine + browser variant), cli/lib/ (download-providers helper). Update package.json bin / main / exports / files. Update relative imports, browser detector + extension build paths, Cloudflare Pages function imports, tests, and docs.

What did not change

  • 12 dotfile harness dirs at the repo root (.claude/, .cursor/, .agents/, …) — npx skills reads them there.
  • plugin/ at the repo root — .claude-plugin/marketplace.json references ./plugin.
  • functions/ at the repo root — Cloudflare Pages auto-discovers ./functions/ and there's no wrangler.toml knob to relocate. Moving it would either break the deploy or require a build-time copy step that adds more complexity than the cleanup is worth.

The CLI's npm-published surface (npx impeccable, import 'impeccable', import 'impeccable/browser') is unchanged — only package.json paths point at the new layout.

Eval framework follow-up

The separate ~/code/impeccable-evals/ repo reads the skill from this repo. Path constants there have been updated in a tracking commit (5691a44 on that repo's main, not pushed). Will go up alongside this PR.

Test plan

  • bun run build clean for skills + Astro
  • bun run build:browser regenerates cli/engine/detect-antipatterns-browser.js
  • bun run build:extension regenerates extension/detector/detect.js + zip
  • bun run test — 186/186 pass
  • node cli/bin/cli.js --version and --help work
  • Build output verified: _headers, _redirects, _routes.json, _data/ all land in build/ for Cloudflare Pages
  • All five eval runner TS files import + execute their top-level setup under bun
  • Cloudflare Pages preview deploy (let CI / preview env catch any deploy-time issues with the publicDir relocation)

🤖 Generated with Claude Code


Note

Medium Risk
Medium risk because it rewires many path assumptions (skill source, CLI engine entrypoints, Astro publicDir, and generated-asset locations), which could break builds, installs, or runtime script loading if any path is missed.

Overview
Aligns the repo with a new top-level layout: skill source references are updated from source/skills/impeccable to skill/, and the site’s public assets/generator outputs are moved under site/public with Astro configured via publicDir.

Updates live-mode tooling to load the detector bundle from the new CLI engine location (cli/engine/detect-antipatterns-browser.js), adjusts CLI entrypoints/imports to point at cli/engine, and refreshes docs/ignores to match the relocated generated content and build artifacts.

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

Paul Bakaus and others added 5 commits May 4, 2026 14:24
Phase 1 step 1 of the directory restructure. The dual content tree was
called out in CLAUDE.md as cleanup; both trees were already in sync
except for anti-patterns-catalog.js, which moves to site/data/.

- Delete content/site/skills/ and content/site/tutorials/ (duplicates of
  site/content/, which is what Astro's content collection actually reads).
- Move content/site/anti-patterns-catalog.js -> site/data/.
- Update scripts/lib/sub-pages-data.js and scripts/build.js to read from
  site/content/ and site/data/.
- Drop content/site/ from validateProse target list (site/content was
  already there).
- Rewrite the "Two content trees" section in CLAUDE.md as a single-tree
  pointer; update stale dev-server text mentioning the deleted
  server/index.js.

Tests: 186/186 pass. Skills build: clean.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Phase 1 step 2 of the directory restructure. The path was redundantly
nested ("source/" wrapper plus "skills/impeccable/" — singular content
hidden behind the plural). Collapses to flat skill/SKILL.md +
skill/reference/ + skill/scripts/.

- Move source/skills/impeccable/ -> skill/.
- Rewrite scripts/lib/utils.js readSourceFiles(): drop the multi-skill
  iteration (CLAUDE.md commits to a single user-invocable skill); read
  skill/SKILL.md directly.
- Update scripts/build.js, scripts/generate-og-image.js, and the
  sub-pages data layer to point at skill/.
- Update tests/lib/utils.test.js: drop the "multi-skill" and "dir-name
  fallback" cases, update single-skill paths to skill/.
- Update tests/build.test.js similarly: drop "multiple skills"
  integration test, update paths.
- Update non-glob path joins in tests/framework-fixtures.test.mjs,
  tests/live-e2e/session.mjs, tests/live-e2e/agents/llm-agent.mjs,
  tools/live-loop.mjs.
- Update prose/text references in CLAUDE.md, AGENTS.md, DEVELOP.md,
  README.md, scripts/lib/sub-pages-data.js, bin/commands/skills.mjs,
  site/data/anti-patterns-catalog.js, site/pages/docs/[...slug].astro,
  docs/adr-live-variant-mode.md, docs/plans/.

Eval framework note: the separate impeccable-evals repo reads
../impeccable/source/skills/impeccable/ and needs a coordinated
rename to ../impeccable/skill/.

Tests: 186/186 pass. Skills build: clean.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Phase 1 step 3 of the directory restructure. The internal docs/ dir
(ADRs and plans) clashed with the site's /docs route. Renaming it
"notes/" makes the difference unambiguous: notes/ is project-internal
process, /docs is the user-facing route under site/pages/docs/.

No code references the dir; the rename is a clean git mv.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Phase 2 step 4 of the directory restructure. Public assets and the
Astro publicDir now live alongside the rest of the site, so site/
is fully self-contained for static content.

- git mv public site/public.
- astro.config.mjs: add publicDir: './site/public'. Astro defaults to
  ./public at the project root, so the override is required.
- scripts/build.js: write generated _data, _headers, _redirects,
  _routes.json, and js/detect-antipatterns-browser.js into
  site/public/. Also delete the dead _REMOVED() Bun static-site
  builder (replaced by Astro at #130; the placeholder no longer earns
  its keep).
- scripts/build.js validateProse: replace the stale public/index.html
  reference (deleted at the Astro migration) with site/pages/index.astro
  in the count-validation file list, restoring homepage drift detection.
- scripts/generate-og-image.js: write OG image into site/public/.
- scripts/screenshot-antipatterns.js: read examples from + write
  screenshots to site/public/antipattern-{examples,images}/.
- scripts/lib/sub-pages-data.js: load command demos from
  site/public/js/demos/commands.
- .gitignore: rename the public/* generator-output entries to
  site/public/*.
- CLAUDE.md: refresh CSS/data-file paths (still pointing at the old
  pre-Astro public/css/ + public/js/ tree), point the changelog and
  command-add checklists at site/pages/index.astro and
  site/scripts/data.js + site/scripts/components/framework-viz.js.

Cloudflare Pages note: functions/ stays at the repo root because
CF Pages auto-discovers it there with no configuration knob to
relocate. Moving it under site/ would either break deployment or
require a build-time copy step that adds more complexity than the
cleanup is worth.

Tests: 186/186 pass. Skills + site build clean. _headers,
_redirects, _routes.json, _data/ all land in build/ correctly.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Phase 2 step 5 of the directory restructure. The CLI surface was split
across three top-level dirs whose names were easy to mistake for each
other (especially src/ vs source/ pre-step-2). Consolidates under cli/.

- git mv bin -> cli/bin (CLI entry + skills sub-command)
- git mv src -> cli/engine (detect-antipatterns engine + browser variant)
- git mv lib -> cli/lib (download-providers helper)

Update package.json:
- bin.impeccable: cli/bin/cli.js
- main + exports: cli/engine/detect-antipatterns.mjs and the
  ./browser variant
- files: ["cli/", "LICENSE"]

Update internal references:
- cli/bin/cli.js: dynamic import points at ../engine/, package.json
  read goes one level deeper (../../package.json).
- functions/api/download/[type]/[provider]/[id].js + bundle/[provider].js:
  cli/lib/download-providers.js path.
- scripts/build.js, scripts/build-browser-detector.js,
  scripts/build-extension.js: cli/engine path constants.
- scripts/lib/sub-pages-data.js, scripts/lib/utils.js, skill/scripts/
  live-server.mjs: comment refs.
- tests/detect-antipatterns{,-browser,-fixtures}.test.{js,mjs},
  tests/windows-path-fix.test.js: import + read paths.
- AGENTS.md, CLAUDE.md: doc paths.

Verified:
- npx node cli/bin/cli.js --version, --help, detect --help all work.
- bun run build, bun run build:browser, bun run build:extension all
  clean. Browser detector lands at cli/engine/detect-antipatterns-browser.js;
  extension/detector/detect.js still emits to the same location.
- bun run test: 186/186 pass.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Comment thread .claude/skills/impeccable/scripts/live-server.mjs
Bugbot caught two runtime path leaks where the comment got renamed
to cli/engine/ but the actual code still used the old src/ segment.

- skill/scripts/live-server.mjs: detectPaths array now joins cli, engine,
  detect-antipatterns-browser.js for both the repo-relative lookup
  (4 dirs up from .claude/skills/impeccable/scripts/ to repo root) and
  the npm node_modules fallback. Without this fix, the detection
  overlay would silently not load during live-server sessions.

- scripts/build.js: the post-build copy of the browser detector into
  site/public/js/ was reading from src/. The if (fs.existsSync(...))
  guard meant the copy was silently skipping, so antipattern-examples
  pages would 404 on /js/detect-antipatterns-browser.js once the site
  was deployed.

Tests: 186/186 pass. Build clean. site/public/js/detect-antipatterns-browser.js
re-emits as expected.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
@cloudflare-workers-and-pages
Copy link
Copy Markdown

cloudflare-workers-and-pages Bot commented May 4, 2026

Deploying impeccable with  Cloudflare Pages  Cloudflare Pages

Latest commit: 39b43a4
Status: ✅  Deploy successful!
Preview URL: https://99c99e12.impeccable-2rv.pages.dev
Branch Preview URL: https://refactor-directory-structure.impeccable-2rv.pages.dev

View logs

Copy link
Copy Markdown

@cursor cursor Bot left a comment

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 and found 1 potential issue.

Fix All in Cursor

❌ Bugbot Autofix is OFF. To automatically fix reported issues with cloud agents, enable autofix in the Cursor dashboard.

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

Comment thread cli/bin/commands/skills.mjs Outdated
Paul Bakaus and others added 3 commits May 4, 2026 16:06
Bugbot caught three call sites in cli/bin/commands/skills.mjs that
import '../../skill/scripts/cleanup-deprecated.mjs'. Pre-rename, that
was correct from bin/commands/ (one parent to bin/, one to repo root).
After moving the file from bin/commands/ to cli/bin/commands/, the
path is one directory deeper, so it needs three .. segments to reach
the repo root. Without the fix, every cleanup invocation throws on
import and gets swallowed by the surrounding try/catch — silent skip.

cli/bin/cli.js's package.json read already uses '../../package.json'
(the same depth pattern), confirming three levels is correct.

Verified: dynamic import resolves and exports the expected functions.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Same root cause as the two bugbot finds: some references in moved or
related files weren't tracked because they didn't match a simple
sed pattern. Caught the rest by walking each moved dir's depth and
each Astro-migration deletion.

Stale path references (post-Astro migration, missed earlier):
- CLAUDE.md: legacy URL redirects "live in server/index.js" -> point
  at the actual sources (scripts/build.js generateCFConfig +
  site/public/_redirects).
- AGENTS.md: counts.js path (public/ -> site/public/), changelog file
  (public/index.html -> site/pages/index.astro), screenshots note
  (public/ -> site/), source-of-truth dirs (source/, src/ -> skill/,
  cli/).
- tests/detect-antipatterns-browser.test.mjs: comment about routes
  "in server/index.js".
- skill/reference/live.md: workflow.css example for "this repo" was
  pre-Astro (public/css/) -> site/styles/. (User-project Vite/Next
  example unchanged.)

Stale path that pointed at moved files:
- tests/skills-cli.test.js: CLI path was '..', 'bin', 'cli.js'; now
  '..', 'cli', 'bin', 'cli.js'. Test isn't wired into bun run test
  but it would have failed if invoked.

Dead files (orphaned by Astro migration, never cleaned up):
- tests/server/download-validation.test.js: imported from
  ../../server/lib/{validation,api-handlers}.js which were deleted in
  b8f09c8. Test was a silent failure waiting to happen.
- scripts/lib/render-markdown.js: 156-line module with zero consumers
  (the only caller, scripts/lib/render-page.js, was deleted in the
  Astro cleanup).
- scripts/build.js: dead commented-out generateSubPages import.

Tests: 186/186 pass. Build clean.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Cloudflare Pages rejects the build with `Unsupported package manager
specification (bun@1.3.11)`. The packageManager field follows
Corepack's syntax which only validates npm/pnpm/yarn — `bun@X.Y.Z`
parses as a malformed Corepack directive even though Bun itself
treats it as a hint.

Pre-existing on main since d874af0 (CF Pages deploy on main also
failing); just surfaces here because the PR triggers a fresh deploy.

CF Pages auto-detects Bun anyway (the build log confirms:
"Detected the following tools from environment: bun@1.3.11,
pnpm@10.11.1, nodejs@22.16.0"). Removing the field unblocks the
deploy without changing local dev behavior.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
@pbakaus pbakaus merged commit e587004 into main May 4, 2026
3 checks passed
@pbakaus pbakaus deleted the refactor/directory-structure branch May 4, 2026 23:38
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

1 participant