Synced 79 commits from EveryInc/compound-engineering-plugin upstream while
preserving fork-specific customizations (Python/FastAPI pivot, Zoominfo-internal
review agents, deploy-wiring operational lessons, custom personas).
## Triage decisions (15 conflicts resolved)
Keep deleted (7) -- fork already removed these in prior cleanups:
- agents/design/{design-implementation-reviewer,design-iterator,figma-design-sync}
(no fork successor; backend-Python focus doesn't need UI/Figma agents)
- agents/docs/ankane-readme-writer (replaced by python-package-readme-writer)
- agents/review/{data-migration-expert,performance-oracle,security-sentinel}
(replaced by *-reviewer naming convention: data-migrations-reviewer,
performance-reviewer, security-reviewer)
Keep local (1):
- agents/workflow/lint.md (Python tooling: ruff/mypy/djlint/bandit; upstream
deleted the file). Fixed pre-existing duplicate "2." numbering bug.
Restore from upstream (1):
- agents/review/data-integrity-guardian.md (kept for GDPR/CCPA privacy
compliance angle not covered by data-migrations-reviewer)
Merge both (6) -- upstream structural wins layered with fork intent:
- agents/research/best-practices-researcher.md (upstream <examples> removal +
fork's Rails/Ruby -> Python/FastAPI translations)
- skills/ce-brainstorm/SKILL.md (universal-brainstorming routing + Slack
context + non-obvious angles + fork's Deploy wiring flag)
- skills/ce-plan/SKILL.md (universal-planning routing + planning-bootstrap +
fork's two Deploy wiring check bullets)
- skills/ce-review/SKILL.md (Run ID, model tiering haiku->sonnet, compact-JSON
artifact contract, file-type awareness, cli-readiness-reviewer + fork's
zip-agent-validator, design-conformance-reviewer, Stage 6 Zip Agent
Validation)
- skills/ce-review/references/persona-catalog.md (cli-readiness row + adversarial
refinement + fork's Language & Framework Conditional layer; 22 personas total)
- skills/ce-work/SKILL.md (Parallel Safety Check, parallel-subagent constraints,
Phase 3-4 compression + fork's deploy-values self-review row, with duplicate
checklist bullet collapsed to single occurrence)
## Auto-applied (no triage needed)
- 225 remote-only files: accepted as-is (new docs, brainstorms, plans,
upstream skills, tests, scripts)
- 70 local-only files: 46 preserved as-is (kieran-python, tiangolo-fastapi,
zip-agent-validator, design-conformance-reviewer, essay/proof commands,
excalidraw-png-export, etc.); 24 stayed deleted (dhh-rails-style,
andrew-kane-gem-writer, dspy-ruby Ruby skills no longer needed)
## README updated
- Removed Design section (3 deleted agents)
- Removed deleted Review entries (data-migration-expert, dhh-rails-reviewer,
kieran-rails-reviewer, performance-oracle, security-sentinel)
- Added new Review entries: design-conformance-reviewer, previous-comments-reviewer,
tiangolo-fastapi-reviewer, zip-agent-validator
- Workflow: added lint
- Docs: replaced ankane-readme-writer with python-package-readme-writer
## Known issues (not introduced by merge decisions)
- 9 detect-project-type.sh tests fail on macOS bash 3.2 (script uses
`declare -A` which requires bash 4+). Upstream regression in commit 070092d
(#568). Resolution: install bash 4+ via `brew install bash` locally;
upstream fix tracked separately.
- 2 review-skill-contract tests reference deleted agents (dhh-rails-reviewer,
data-migration-expert). Pre-existing fork inconsistency, not new.
bun run release:validate: passes (46 agents, 51 skills, 0 MCP servers)
91 lines
2.9 KiB
JavaScript
Executable File
91 lines
2.9 KiB
JavaScript
Executable File
#!/usr/bin/env node
|
|
/**
|
|
* Export an Excalidraw JSON file to PNG using Playwright + the official Excalidraw library.
|
|
*
|
|
* Usage: node export_png.mjs <input.excalidraw> [output.png]
|
|
*
|
|
* All rendering happens locally. Diagram data never leaves the machine.
|
|
* The Excalidraw JS library is fetched from esm.sh CDN (code only, not user data).
|
|
*/
|
|
|
|
import { createRequire } from "module";
|
|
import { readFileSync, writeFileSync, copyFileSync } from "fs";
|
|
import { createServer } from "http";
|
|
import { join, extname, dirname } from "path";
|
|
import { fileURLToPath } from "url";
|
|
|
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
const RUNTIME_DIR = join(__dirname, ".export-runtime");
|
|
const HTML_PATH = join(__dirname, "export.html");
|
|
|
|
// Resolve playwright from the runtime directory, not the script's location
|
|
const require = createRequire(join(RUNTIME_DIR, "node_modules", "playwright", "index.mjs"));
|
|
const { chromium } = await import(join(RUNTIME_DIR, "node_modules", "playwright", "index.mjs"));
|
|
|
|
const inputPath = process.argv[2];
|
|
if (!inputPath) {
|
|
console.error("Usage: node export_png.mjs <input.excalidraw> [output.png]");
|
|
process.exit(1);
|
|
}
|
|
|
|
const outputPath = process.argv[3] || inputPath.replace(/\.excalidraw$/, ".png");
|
|
|
|
// Set up a temp serving directory
|
|
const SERVE_DIR = join(__dirname, ".export-tmp");
|
|
const { mkdirSync, rmSync } = await import("fs");
|
|
mkdirSync(SERVE_DIR, { recursive: true });
|
|
copyFileSync(HTML_PATH, join(SERVE_DIR, "export.html"));
|
|
copyFileSync(inputPath, join(SERVE_DIR, "diagram.excalidraw"));
|
|
|
|
const MIME = {
|
|
".html": "text/html",
|
|
".json": "application/json",
|
|
".excalidraw": "application/json",
|
|
};
|
|
|
|
const server = createServer((req, res) => {
|
|
const file = join(SERVE_DIR, req.url === "/" ? "export.html" : req.url);
|
|
try {
|
|
const data = readFileSync(file);
|
|
res.writeHead(200, { "Content-Type": MIME[extname(file)] || "application/octet-stream" });
|
|
res.end(data);
|
|
} catch {
|
|
res.writeHead(404);
|
|
res.end("Not found");
|
|
}
|
|
});
|
|
|
|
server.listen(0, "127.0.0.1", async () => {
|
|
const port = server.address().port;
|
|
|
|
let browser;
|
|
try {
|
|
browser = await chromium.launch({ headless: true });
|
|
const page = await browser.newPage();
|
|
|
|
page.on("pageerror", err => console.error("Page error:", err.message));
|
|
|
|
await page.goto(`http://127.0.0.1:${port}`);
|
|
|
|
await page.waitForFunction(
|
|
() => document.title.startsWith("READY") || document.title.startsWith("ERROR"),
|
|
{ timeout: 30000 }
|
|
);
|
|
|
|
const title = await page.title();
|
|
if (title.startsWith("ERROR")) {
|
|
console.error("Export failed:", title);
|
|
process.exit(1);
|
|
}
|
|
|
|
const dataUrl = await page.evaluate(() => window.__PNG_DATA__);
|
|
const base64 = dataUrl.replace(/^data:image\/png;base64,/, "");
|
|
writeFileSync(outputPath, Buffer.from(base64, "base64"));
|
|
console.log(outputPath);
|
|
} finally {
|
|
if (browser) await browser.close();
|
|
server.close();
|
|
rmSync(SERVE_DIR, { recursive: true, force: true });
|
|
}
|
|
});
|