mirror of
https://github.com/diegosouzapw/OmniRoute.git
synced 2026-05-22 19:57:07 +00:00
* fix(cli-tools): guard modelId type before calling indexOf E2E shakedown v3.8.0: cli-tools quebrava com TypeError quando dynamicModels continha entradas sem .id (objeto retornado diretamente em vez de string). * fix(offline): avoid SSR/CSR hydration mismatch on navigator.onLine Replace useState+lazy-initializer with useSyncExternalStore so the server snapshot (() => false) and client snapshot (() => navigator.onLine) are declared separately. React hydrates with the server value and switches to the real online status client-side without a mismatch. * chore(i18n): add missing en.json keys for translator, cli-tools, memory, onboarding Adds 58 missing keys identified by the new dashboard audit script: - cliTools: 18 custom CLI builder keys (CustomCliCard) - translator: 24 keys covering stream transformer, live monitor, test bench - memory: 12 health/pagination/dialog keys - onboarding.tier: 8 keys for the tier tour walkthrough Also adds scripts/i18n/audit-dashboard-pages.mjs which scans all dashboard pages, reports t() calls referencing missing en.json keys, and flags candidate hardcoded JSX/attribute strings. * chore(i18n): replace hardcoded UI text with t() calls across dashboard (round 1) Subagents refactored 8 high-impact dashboard pages, replacing 81 of the 407 hardcoded English/PT strings flagged by the audit with proper useTranslations() lookups. Added 73 corresponding keys to en.json across the home, apiManager, providers, settings, and usage namespaces. Pages affected: - BudgetTab (27 → 0) - HomePageClient (2 → 0) - RoutingTab (25 → 7) - ResilienceTab (38 → 18) - SystemStorageTab (42 → 21) - providers/[id] (17 → 15) - ApiManagerPageClient (14 → 13) - OneproxyTab (13 → 10) Also adds two helper scripts: - scripts/i18n/extract-keys-from-diff.mjs — extracts new keys from git diff - scripts/i18n/merge-keys.mjs — merges a pending-keys JSON into en.json Remaining hardcoded strings will be addressed in follow-up rounds. * chore(i18n): replace hardcoded UI text with t() calls across dashboard (round 2) Continues round 1 (commit8d34f4c65). Round-2 subagents refactored additional dashboard pages, replacing 77 more hardcoded strings with useTranslations() lookups. Added 79 corresponding keys to en.json across the a2aDashboard, agents, analytics, apiManager, cliTools, common, and settings namespaces. Pages affected: - a2a/page (new useTranslations + 6 keys) - agent-skills/page (new useTranslations + 9 keys) - AutoRoutingAnalyticsTab (new useTranslations + 6 keys) - AppearanceTab (8 → 6 remaining) - OneproxyTab (10 → 0) - ResilienceTab (18 → 0 missing key) - RoutingTab (7 → 0 missing key) - VisionBridgeSettingsTab (new useTranslations + 6 keys) - CopilotToolCard (7 → 0 missing key) - ApiManagerPageClient (13 → 0 missing key) - gamification/admin (new useTranslations + 7 keys) Hardcoded total: 326 → 249. Real missing keys: 0 (the 6 still flagged are false positives in exampleTemplates.tsx where t is passed as a parameter — keys exist at translator.templatePayloads.*). * chore(i18n): replace hardcoded UI text with t() calls across dashboard (round 3) Round-3 subagents and manual edits refactored 9 more dashboard pages (plus 2 small extras), replacing ~80 hardcoded strings with useTranslations() lookups. Added 79 corresponding keys to en.json across analytics, cloudAgents, combos, common, health, settings, and usage namespaces. Pages affected: - analytics/ComboHealthTab (new useTranslations + 15 keys) - analytics/CompressionAnalyticsTab (new useTranslations + 11 keys) - settings/SystemStorageTab (21 → 0 missing key) - tokens/page (new useTranslations + 13 keys) - usage/BudgetTab (9 missing fixed) - health/page (manual: 6 keys) - cloud-agents/page (manual: 3 keys) - combos/page (manual: 1 key) Hardcoded total: 249 → 164. Real missing keys: 0 (6 remaining are exampleTemplates.tsx false positives). Also adds scripts/i18n/build-pending-from-missing.mjs which reads _audit.json and locates English values from HEAD to rebuild _pending-keys.json after race-condition resets between subagent edits. * chore(i18n): localize remaining dashboard settings labels Replace hardcoded labels in compression and resilience settings with translation lookups to continue the dashboard i18n cleanup. Add the v3.8.0 dashboard shakedown runbook to document the manual smoke-test process and known dev environment pitfalls. * chore(i18n): replace hardcoded UI text with t() calls across dashboard (round 4) Round-4 subagent + manual key-resolution refactored remaining strings in 3 high-traffic settings/API tabs, plus extracted English values for keys that were already added as t() calls but lost during the previous en.json race-condition resets. Pages affected: - api-manager/ApiManagerPageClient (7 → 0 missing key) - settings/CompressionSettingsTab (8 → 0 missing key) - settings/MemorySkillsTab (8 → 0 missing key) - settings/ResilienceTab (4 more keys recovered) Hardcoded total: 164 → 140. Real missing keys: 0 (6 remaining are the exampleTemplates.tsx false positives — t passed as parameter). * chore(i18n): replace hardcoded UI text with t() calls across dashboard (round 5) Round-5 agent began processing the remaining smaller dashboard files. Added 5 more keys to en.json for providers/[id]/page.tsx OAuth flow labels and the cross-OS auto-detection hint. Pages affected: - providers/[id]/page.tsx (5 keys) Hardcoded total: 140 → 136. Real missing keys: 0. * chore(i18n): resolve last 2 missing providers/[id] keys Adds providerDetailMyClaudeAccountPlaceholder and providerDetailPathAutoDetected — the final user-visible labels in the providers/[id] page that the round-5 subagent rewrote to t() calls without yet adding to en.json. Real missing keys: 0 (6 remaining are exampleTemplates.tsx false positives — t is passed as a parameter so the audit cannot resolve the namespace; keys do exist at translator.templatePayloads.*). * chore(i18n): replace hardcoded UI text with t() calls across dashboard (round 6 — 10 parallel agents) Round-6 dispatched 10 parallel subagents covering all 57 remaining dashboard files. Each agent worked on a disjoint file set to avoid en.json race conditions. Added ~60 new i18n keys across 9 namespaces covering small UI labels, table headers, search placeholders, and empty-state messages. Major changes: - analytics: SearchAnalyticsTab, ProviderUtilizationTab, DiversityScoreCard, CompressionAnalyticsTab (new useTranslations + keys) - batch: BatchDetailModal, BatchListTab, FileDetailModal, FilesListTab (new useTranslations + keys) - settings: CliproxyapiSettingsTab, PayloadRulesTab, ModelCooldownsCard, AppearanceTab, PricingTab (mostly new useTranslations) - endpoint: TokenSaverCard, ApiEndpointsTab, EndpointPageClient - cache: CachePerformance, IdempotencyLayer, ReasoningCacheTab, MediaPageClient, page - combos: IntelligentComboPanel, page - playground: ChatPlayground, SearchPlayground - providers: ProviderCard - onboarding: TierFlowDiagram - changelog: ChangelogViewer - home: ProviderTopology, TierCoverageWidget, BootstrapBanner, BadgeToast - usage: BudgetTab, BudgetTelemetryCards, QuotaTable - quotaShare: QuotaSharePageClient - profile: page - leaderboard: page - skills: page Hardcoded total: 131 → 60. Real missing keys: 0 plus 1 false-positive for combos.modePack (lookup via prop-passed t). * chore(i18n): finalize round-6 keys for batch/cache/endpoint/usage Adds the remaining keys produced by parallel agents A4, A6, A8, A9: - common: batch-related labels (BatchDetailModal, BatchListTab, FileDetailModal, FilesListTab, page) + profile/leaderboard - cache: hit rate, latency, retry, avg chars - endpoint: token saver, API endpoints, copy URL, cloud/local labels - usage: noSpend, activeSessions, quotaAlerts, budget timing - skills: install/marketplace/filter - proxyRegistry/quotaShare/mcpDashboard: misc labels Hardcoded total: 60 → 48. Real missing keys: 0 (modePack remaining is a false positive — combos.modePack exists but the audit can't resolve it since IntelligentComboPanel receives t as a prop). * fix(playground): dedupe filteredModels to avoid duplicate React key warning The /v1/models endpoint can return the same model id twice (e.g., when a model is listed by both an alias and its canonical provider), which made the <Select> emit two <option> elements with the same key — triggering "Encountered two children with the same key, codex/gpt-5.5". Replace the chained filter + map with a single pass that skips ids already added. * fix(playground): guard against non-string model ids before .split/.startsWith The /v1/models endpoint can include synthetic entries (combos, locals, in-progress imports) with a null/undefined id. The playground used to call m.id.split("/") in the provider-discovery loop, which threw on the first non-string entry; the surrounding .catch(() => {}) silently swallowed the error, so the provider/model/account dropdowns ended up empty even though /v1/models returned thousands of valid entries. - Skip entries without a string id before split/startsWith. - Log the rejection in the .catch handler so future regressions are visible in DevTools instead of silently emptying the UI. * fix(playground): guard ChatPlayground filteredModels for non-string ids Same root cause as commit49fe356b9: ChatPlayground filtered models with m.id.startsWith(...) which crashed on null/undefined ids returned by /v1/models (synthetic combo entries). Apply the same defensive guard and dedupe used in the parent page. * fix(claude): drop orphan tool_result after fixToolAdjacency strip (discussion #2410) Discussion #2410 reports Claude returning 400 for sequences like: assistant: tool_use(id=X) user: <plain text> ← breaks adjacency user: tool_result(id=X) The previous round added `fixToolAdjacency` (commit44d9abac9) which correctly strips the orphan tool_use from the assistant message. But that left the now-unmatched tool_result intact, so the upstream rejected the request with: messages.N.content.M: unexpected `tool_use_id` found in `tool_result` blocks: X. Each tool_result block must have a corresponding tool_use block in the previous message. Fix: after running `fixToolAdjacency`, re-run `fixToolPairs` to drop the orphaned tool_result blocks. All three call sites updated: - contextManager.purifyHistory (both inside the binary-search loop and the final pass) - BaseExecutor message-prep (Claude path) - claudeCodeCompatible request signer Also tightens an unrelated dynamic-key access in readNestedString (claudeCodeCompatible) to satisfy the prototype- pollution scanner triggered by the post-tool semgrep hook. * fix(mitm): point runtime manager re-export to js entrypoint Use the emitted `.js` path for the runtime manager re-export so dynamic runtime loading resolves correctly outside the Turbopack alias handling. * docs: add AgentRouter setup guide (#2422) Integrated into release/v3.8.0 — AgentRouter setup guide docs. * feat: add new feature on combos - falloverBeforeRetry (#2417) Integrated into release/v3.8.0 — falloverBeforeRetry for per-model quota skipping in combos. * feat(batch): implement 10 feature requests harvested (#2414) Integrated into release/v3.8.0 — batch of 10 feature requests: llama.cpp local provider, upstream error exposure, Termux detection, providers rotate CLI, t3.chat web skeleton, Zed Docker integration, Kiro multi-account OAuth isolation, auto-combo cost blending, auto-combo context filter, combo provider-level exhaustion tracking (#1731). Conflicts with #2417 (falloverBeforeRetry) resolved. * fix(gamification): resolve SQL bug, auth gap, pagination, and anomaly scoring (#2421) Integrated into release/v3.8.0 — 6 critical gamification bug fixes: SQL SELECT in checkActionCountBadges, federation auth enforcement, leaderboard pagination offset, real z-score computation, addXp level calculation, and barrel index.ts * docs(changelog): add post-release entries for #2414 #2417 #2421 #2422 - feat(batch): T3-Chat-Web executor, exhaustedProviders set (#1731), Zed Docker - feat(combos): falloverBeforeRetry + setTry loop (#2417 — @hartmark) - fix(gamification): SQL SELECT bug, federation auth, pagination, z-score (#2421 — @oyi77) - docs: AgentRouter setup guide (#2422 — @leninejunior) * fix(security): resolve CodeQL random/password-hash alerts and sync docs & tests --------- Co-authored-by: diegosouzapw <diego.souza.pw@gmail.com> Co-authored-by: Lenine Júnior <lenine@engrene.com.br> Co-authored-by: Markus Hartung <mail@hartmark.se> Co-authored-by: Paijo <14921983+oyi77@users.noreply.github.com>
341 lines
11 KiB
JavaScript
341 lines
11 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
/**
|
|
* OmniRoute — Postinstall Native Module Fix
|
|
*
|
|
* The npm package ships with a Next.js standalone build that includes
|
|
* native modules compiled for the build platform (Linux x64) inside
|
|
* app/node_modules/. However, npm also installs these as top-level
|
|
* dependencies (in the root node_modules/), correctly compiled for
|
|
* the user's platform.
|
|
*
|
|
* This script copies the correctly-built native binaries from the root
|
|
* into the standalone app directory — no rebuild or build tools needed.
|
|
*
|
|
* Modules repaired:
|
|
* - better-sqlite3 (SQLite bindings)
|
|
* - wreq-js (TLS client for OAuth providers)
|
|
*
|
|
* Fixes: https://github.com/diegosouzapw/OmniRoute/issues/129
|
|
* Fixes: https://github.com/diegosouzapw/OmniRoute/issues/321
|
|
* Fixes: https://github.com/diegosouzapw/OmniRoute/issues/426
|
|
* Fixes: https://github.com/diegosouzapw/OmniRoute/issues/1634
|
|
*/
|
|
|
|
import { copyFileSync, cpSync, existsSync, mkdirSync, readdirSync } from "node:fs";
|
|
import { dirname, join } from "node:path";
|
|
import { fileURLToPath } from "node:url";
|
|
|
|
import { PUBLISHED_BUILD_ARCH, PUBLISHED_BUILD_PLATFORM } from "./native-binary-compat.mjs";
|
|
import { hasStandaloneAppBundle, isTermux } from "./postinstallSupport.mjs";
|
|
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = dirname(__filename);
|
|
const ROOT = join(__dirname, "..", "..");
|
|
|
|
const appBinary = join(
|
|
ROOT,
|
|
"app",
|
|
"node_modules",
|
|
"better-sqlite3",
|
|
"build",
|
|
"Release",
|
|
"better_sqlite3.node"
|
|
);
|
|
const rootBinary = join(
|
|
ROOT,
|
|
"node_modules",
|
|
"better-sqlite3",
|
|
"build",
|
|
"Release",
|
|
"better_sqlite3.node"
|
|
);
|
|
|
|
async function fixBetterSqliteBinary() {
|
|
if (!existsSync(join(ROOT, "app", "node_modules", "better-sqlite3"))) {
|
|
return;
|
|
}
|
|
|
|
const platformMatch =
|
|
process.platform === PUBLISHED_BUILD_PLATFORM && process.arch === PUBLISHED_BUILD_ARCH;
|
|
|
|
if (platformMatch) {
|
|
try {
|
|
process.dlopen({ exports: {} }, appBinary);
|
|
return;
|
|
} catch (err) {
|
|
console.warn(` ⚠️ Bundled binary incompatible despite platform match: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
console.log(`\n 🔧 Fixing better-sqlite3 binary for ${process.platform}-${process.arch}...`);
|
|
|
|
if (existsSync(rootBinary)) {
|
|
try {
|
|
mkdirSync(dirname(appBinary), { recursive: true });
|
|
copyFileSync(rootBinary, appBinary);
|
|
} catch (err) {
|
|
console.warn(` ⚠️ Failed to copy binary: ${err.message}`);
|
|
}
|
|
|
|
try {
|
|
process.dlopen({ exports: {} }, appBinary);
|
|
console.log(" ✅ Native module fixed successfully!\n");
|
|
return;
|
|
} catch (err) {
|
|
console.warn(` ⚠️ Copied binary failed to load: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
console.log(" 📥 Attempting to download prebuilt binary via node-pre-gyp...");
|
|
try {
|
|
const { execSync } = await import("node:child_process");
|
|
const preGypBin = join(
|
|
ROOT,
|
|
"app",
|
|
"node_modules",
|
|
".bin",
|
|
process.platform === "win32" ? "node-pre-gyp.cmd" : "node-pre-gyp"
|
|
);
|
|
const preGypFallback = join(
|
|
ROOT,
|
|
"app",
|
|
"node_modules",
|
|
"@mapbox",
|
|
"node-pre-gyp",
|
|
"bin",
|
|
"node-pre-gyp"
|
|
);
|
|
const preGypCmd = existsSync(preGypBin) ? preGypBin : preGypFallback;
|
|
|
|
if (existsSync(preGypCmd)) {
|
|
execSync(`"${process.execPath}" "${preGypCmd}" install --fallback-to-build=false`, {
|
|
cwd: join(ROOT, "app", "node_modules", "better-sqlite3"),
|
|
stdio: "inherit",
|
|
timeout: 60_000,
|
|
});
|
|
mkdirSync(dirname(appBinary), { recursive: true });
|
|
|
|
try {
|
|
process.dlopen({ exports: {} }, appBinary);
|
|
console.log(" ✅ Prebuilt binary downloaded and loaded successfully!\n");
|
|
return;
|
|
} catch (loadErr) {
|
|
console.warn(` ⚠️ Downloaded binary failed to load: ${loadErr.message}`);
|
|
}
|
|
} else {
|
|
console.warn(" ⚠️ node-pre-gyp not found, skipping prebuilt download.");
|
|
}
|
|
} catch (err) {
|
|
console.warn(` ⚠️ node-pre-gyp download failed: ${err.message.split("\n")[0]}`);
|
|
}
|
|
|
|
console.log(" ⚠️ Attempting npm rebuild (requires build tools)...");
|
|
|
|
try {
|
|
const { execSync } = await import("node:child_process");
|
|
|
|
// On Android/Termux, rebuild from source with --build-from-source flag
|
|
const isAndroid = process.platform === "android" || isTermux();
|
|
const rebuildCmd = isAndroid
|
|
? "npm install better-sqlite3 --build-from-source --force"
|
|
: "npm rebuild better-sqlite3";
|
|
|
|
const env = { ...process.env };
|
|
if (isAndroid) {
|
|
env.GYP_DEFINES = "android_ndk_path=''";
|
|
}
|
|
|
|
execSync(rebuildCmd, {
|
|
cwd: join(ROOT, "app"),
|
|
stdio: "inherit",
|
|
timeout: isAndroid ? 600_000 : 300_000, // ARM compilation is slower
|
|
env,
|
|
});
|
|
|
|
process.dlopen({ exports: {} }, appBinary);
|
|
console.log(" ✅ Native module rebuilt successfully!\n");
|
|
return;
|
|
} catch (err) {
|
|
const isTimeout = err.killed || err.signal === "SIGTERM";
|
|
if (isTimeout) {
|
|
const secs = isAndroid ? 600 : 300;
|
|
console.warn(` ⚠️ npm rebuild timed out after ${secs}s.`);
|
|
} else {
|
|
console.warn(` ⚠️ npm rebuild failed: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
console.warn("\n ⚠️ Could not fix better-sqlite3 native module automatically.");
|
|
console.warn(" The server may not start correctly.");
|
|
console.warn(" Manual fix options:");
|
|
if (process.platform === "win32") {
|
|
console.warn(" Option A (easiest — no build tools needed):");
|
|
console.warn(` cd "${join(ROOT, "app", "node_modules", "better-sqlite3")}"`);
|
|
console.warn(" npx @mapbox/node-pre-gyp install --fallback-to-build=false");
|
|
console.warn(" Option B (requires Build Tools for Visual Studio):");
|
|
console.warn(` cd "${join(ROOT, "app")}" && npm rebuild better-sqlite3`);
|
|
console.warn(" Install from: https://visualstudio.microsoft.com/visual-cpp-build-tools/");
|
|
console.warn(" Also ensure Python is installed: https://python.org");
|
|
} else if (process.platform === "darwin") {
|
|
console.warn(` cd ${join(ROOT, "app")} && npm rebuild better-sqlite3`);
|
|
console.warn(" If build tools are missing: xcode-select --install");
|
|
} else {
|
|
console.warn(` cd ${join(ROOT, "app")} && npm rebuild better-sqlite3`);
|
|
}
|
|
console.warn("");
|
|
}
|
|
|
|
/**
|
|
* Fix wreq-js native binary for the standalone app directory.
|
|
*
|
|
* wreq-js ships platform-specific .node binaries under rust/.
|
|
* The standalone build may only contain Linux binaries from the CI.
|
|
* This copies the correct platform binary from the root install.
|
|
*
|
|
* Fixes: https://github.com/diegosouzapw/OmniRoute/issues/1634
|
|
*/
|
|
async function fixWreqJsBinary() {
|
|
// wreq-js native module is not loadable in Termux (libgcc path mismatch).
|
|
// The runtime already falls back gracefully when wreq-js is unavailable.
|
|
if (process.platform === "android" || isTermux()) {
|
|
console.log(
|
|
" [postinstall] wreq-js: skipped on Termux/Android " +
|
|
"(libgcc not available — OAuth TLS fingerprinting will use the fallback path)"
|
|
);
|
|
return;
|
|
}
|
|
|
|
const appWreqDir = join(ROOT, "app", "node_modules", "wreq-js", "rust");
|
|
const rootWreqDir = join(ROOT, "node_modules", "wreq-js", "rust");
|
|
|
|
if (!existsSync(join(ROOT, "app", "node_modules", "wreq-js"))) {
|
|
return;
|
|
}
|
|
|
|
const binaryName = `wreq-js.${process.platform}-${process.arch}.node`;
|
|
const appBinaryPath = join(appWreqDir, binaryName);
|
|
const rootBinaryPath = join(rootWreqDir, binaryName);
|
|
|
|
// Check if the platform binary already exists and loads
|
|
if (existsSync(appBinaryPath)) {
|
|
try {
|
|
process.dlopen({ exports: {} }, appBinaryPath);
|
|
return; // Already working
|
|
} catch (err) {
|
|
console.warn(` ⚠️ wreq-js binary exists but failed to load: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
console.log(`\n 🔧 Fixing wreq-js binary for ${process.platform}-${process.arch}...`);
|
|
|
|
// Strategy 1: Copy from root node_modules
|
|
if (existsSync(rootBinaryPath)) {
|
|
try {
|
|
mkdirSync(appWreqDir, { recursive: true });
|
|
copyFileSync(rootBinaryPath, appBinaryPath);
|
|
process.dlopen({ exports: {} }, appBinaryPath);
|
|
console.log(" ✅ wreq-js native module fixed successfully!\n");
|
|
return;
|
|
} catch (err) {
|
|
console.warn(` ⚠️ Copied wreq-js binary failed to load: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
// Strategy 2: Copy entire rust/ directory from root (gets all platform binaries)
|
|
if (existsSync(rootWreqDir)) {
|
|
try {
|
|
mkdirSync(appWreqDir, { recursive: true });
|
|
const files = readdirSync(rootWreqDir);
|
|
for (const file of files) {
|
|
if (file.endsWith(".node")) {
|
|
copyFileSync(join(rootWreqDir, file), join(appWreqDir, file));
|
|
}
|
|
}
|
|
if (existsSync(appBinaryPath)) {
|
|
process.dlopen({ exports: {} }, appBinaryPath);
|
|
console.log(" ✅ wreq-js native module fixed (full copy) successfully!\n");
|
|
return;
|
|
}
|
|
} catch (err) {
|
|
console.warn(` ⚠️ wreq-js full copy failed: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
// Strategy 3: Rebuild wreq-js inside app/
|
|
console.log(" 📥 Attempting npm rebuild wreq-js...");
|
|
try {
|
|
const { execSync } = await import("node:child_process");
|
|
execSync("npm rebuild wreq-js", {
|
|
cwd: join(ROOT, "app"),
|
|
stdio: "inherit",
|
|
timeout: 120_000,
|
|
});
|
|
if (existsSync(appBinaryPath)) {
|
|
process.dlopen({ exports: {} }, appBinaryPath);
|
|
console.log(" ✅ wreq-js native module rebuilt successfully!\n");
|
|
return;
|
|
}
|
|
} catch (err) {
|
|
console.warn(` ⚠️ wreq-js rebuild failed: ${err.message}`);
|
|
}
|
|
|
|
console.warn(
|
|
`\n ⚠️ Could not fix wreq-js native module for ${process.platform}-${process.arch}.`
|
|
);
|
|
console.warn(" OAuth-based providers (Codex, Cursor, etc.) may not work.");
|
|
console.warn(` Manual fix: cd ${join(ROOT, "app")} && npm install wreq-js --no-save\n`);
|
|
}
|
|
|
|
async function ensureSwcHelpers() {
|
|
if (!hasStandaloneAppBundle(ROOT)) {
|
|
return;
|
|
}
|
|
|
|
const swcHelpersApp = join(ROOT, "app", "node_modules", "@swc", "helpers");
|
|
const swcHelpersRoot = join(ROOT, "node_modules", "@swc", "helpers");
|
|
|
|
if (existsSync(swcHelpersApp)) {
|
|
return;
|
|
}
|
|
|
|
if (existsSync(swcHelpersRoot)) {
|
|
try {
|
|
const { cpSync } = await import("node:fs");
|
|
mkdirSync(join(ROOT, "app", "node_modules", "@swc"), { recursive: true });
|
|
cpSync(swcHelpersRoot, swcHelpersApp, { recursive: true });
|
|
console.log(" ✅ @swc/helpers copied to standalone app/node_modules.\n");
|
|
} catch (err) {
|
|
console.warn(` ⚠️ Could not copy @swc/helpers: ${err.message}`);
|
|
console.warn(
|
|
" Try manually: cp -r node_modules/@swc/helpers app/node_modules/@swc/helpers\n"
|
|
);
|
|
}
|
|
return;
|
|
}
|
|
|
|
console.warn(" ⚠️ @swc/helpers not found in root node_modules either.");
|
|
console.warn(" Try: npm install --save-exact @swc/helpers@0.5.19\n");
|
|
}
|
|
|
|
async function syncProjectEnv() {
|
|
try {
|
|
const { syncEnv } = await import("./sync-env.mjs");
|
|
syncEnv({ rootDir: ROOT });
|
|
} catch (err) {
|
|
console.warn(` ⚠️ .env sync skipped: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
await fixBetterSqliteBinary();
|
|
await fixWreqJsBinary();
|
|
await ensureSwcHelpers();
|
|
await syncProjectEnv();
|
|
|
|
// Warm up native runtimes (better-sqlite3 in ~/.omniroute/runtime/).
|
|
// Non-fatal: errors are caught inside postinstall.mjs.
|
|
try {
|
|
await import("../postinstall.mjs");
|
|
} catch {
|
|
// Silently skip — runtime warm-up is best-effort.
|
|
}
|