UNPKG

convex

Version:

Client for the Convex Cloud

331 lines (330 loc) 10.3 kB
"use strict"; import { logFinishedStep, logVerbose, showSpinner } from "../../../bundler/log.js"; import { bigBrainPause, bigBrainRecordActivity, bigBrainStart } from "./bigBrain.js"; import { loadDeploymentConfig, loadDeploymentConfigFromDir, loadProjectLocalConfig, legacyDeploymentStateDir, rootDeploymentStateDir, saveDeploymentConfig } from "./filePaths.js"; import { ensureBackendRunning, ensureBackendStopped, localDeploymentUrl, runLocalBackend, withRunningBackend } from "./run.js"; import { handlePotentialUpgrade } from "./upgrade.js"; import { promptSearch } from "../utils/prompts.js"; import { LocalDeploymentError, printLocalDeploymentOnError } from "./errors.js"; import { chooseLocalBackendPorts, printLocalDeploymentWelcomeMessage, isOffline, LOCAL_BACKEND_INSTANCE_SECRET } from "./utils.js"; import { ensureBackendBinaryDownloaded } from "./download.js"; import { defaultEnvBackend } from "../defaultEnv.js"; import { deploymentEnvBackend } from "../env.js"; import { getProjectDetails } from "../deploymentSelection.js"; export async function handleLocalDeployment(ctx, options) { if (await isOffline()) { return handleOffline(ctx, options); } const existingDeploymentForProject = await getExistingDeployment(ctx, { projectSlug: options.projectSlug, teamSlug: options.teamSlug }); const isFirstTime = existingDeploymentForProject === null; if (isFirstTime) { printLocalDeploymentWelcomeMessage(); } ctx.registerCleanup(async (_exitCode, err) => { if (err instanceof LocalDeploymentError) { printLocalDeploymentOnError(); } }); if (existingDeploymentForProject !== null) { logVerbose(`Found existing deployment for project ${options.projectSlug}`); await ensureBackendStopped(ctx, { ports: { cloud: existingDeploymentForProject.config.ports.cloud }, maxTimeSecs: 5, deploymentName: existingDeploymentForProject.deploymentName, allowOtherDeployments: true }); } const { binaryPath, version } = await ensureBackendBinaryDownloaded( ctx, options.backendVersion === void 0 ? { kind: "latest", allowedVersion: existingDeploymentForProject?.config.backendVersion } : { kind: "version", version: options.backendVersion } ); const { cloudPort, sitePort } = await chooseLocalBackendPorts(ctx, { requestedPorts: options.ports, suggestedPorts: existingDeploymentForProject?.config.ports }); const { deploymentName, adminKey } = await bigBrainStart(ctx, { port: cloudPort, projectSlug: options.projectSlug, teamSlug: options.teamSlug, instanceName: existingDeploymentForProject?.deploymentName ?? null }); const onActivity = async (isOffline2, _wasOffline) => { await ensureBackendRunning(ctx, { cloudPort, deploymentName, maxTimeSecs: 5 }); if (isOffline2) { return; } await bigBrainRecordActivity(ctx, { instanceName: deploymentName }); }; const { cleanupHandle } = await handlePotentialUpgrade(ctx, { deploymentKind: "local", deploymentName, oldVersion: existingDeploymentForProject?.config.backendVersion ?? null, newBinaryPath: binaryPath, newVersion: version, ports: { cloud: cloudPort, site: sitePort }, adminKey, instanceSecret: LOCAL_BACKEND_INSTANCE_SECRET, forceUpgrade: options.forceUpgrade }); if (isFirstTime) { await importDefaultEnvVars(ctx, { teamSlug: options.teamSlug, projectSlug: options.projectSlug, deploymentName, deploymentUrl: localDeploymentUrl(cloudPort), adminKey }); } let activityTimeout = null; const scheduleActivityPing = () => { activityTimeout = setTimeout(async () => { try { await bigBrainRecordActivity(ctx, { instanceName: deploymentName }); } catch { } scheduleActivityPing(); }, 6e4); }; scheduleActivityPing(); const cleanupFunc = ctx.removeCleanup(cleanupHandle); ctx.registerCleanup(async (exitCode, err) => { if (activityTimeout !== null) { clearTimeout(activityTimeout); } if (cleanupFunc !== null) { await cleanupFunc(exitCode, err); } await bigBrainPause(ctx, { projectSlug: options.projectSlug, teamSlug: options.teamSlug }); }); return { adminKey, deploymentName, deploymentUrl: localDeploymentUrl(cloudPort), onActivity }; } export async function loadLocalDeploymentCredentials(ctx, deploymentName) { const config = loadDeploymentConfig(ctx, "local", deploymentName); if (config === null) { return ctx.crash({ exitCode: 1, errorType: "fatal", printedMessage: "Failed to load deployment config - try running `npx convex dev --configure`" }); } return { deploymentName, deploymentUrl: localDeploymentUrl(config.ports.cloud), adminKey: config.adminKey }; } async function handleOffline(ctx, options) { const { deploymentName, config } = await chooseFromExistingLocalDeployments(ctx); const { binaryPath } = await ensureBackendBinaryDownloaded(ctx, { kind: "version", version: config.backendVersion }); const { cloudPort, sitePort } = await chooseLocalBackendPorts(ctx, { requestedPorts: options.ports // FIXME: This doesn’t try to reuse the ports already assigned in the config. // Please update this if we ever support offline mode (currently this is dead code). }); saveDeploymentConfig(ctx, "local", deploymentName, config); await runLocalBackend(ctx, { binaryPath, ports: { cloud: cloudPort, site: sitePort }, deploymentName, deploymentKind: "local", instanceSecret: LOCAL_BACKEND_INSTANCE_SECRET, isLatestVersion: false }); return { adminKey: config.adminKey, deploymentName, deploymentUrl: localDeploymentUrl(cloudPort), onActivity: async (isOffline2, wasOffline) => { await ensureBackendRunning(ctx, { cloudPort, deploymentName, maxTimeSecs: 5 }); if (isOffline2) { return; } if (wasOffline) { await bigBrainStart(ctx, { port: cloudPort, projectSlug: options.projectSlug, teamSlug: options.teamSlug, instanceName: deploymentName }); } await bigBrainRecordActivity(ctx, { instanceName: deploymentName }); } }; } async function getExistingDeployment(ctx, options) { const { projectSlug, teamSlug } = options; const projectLocal = loadProjectLocalConfig(ctx); if (projectLocal !== null) { const expectedPrefix = `local-${teamSlug.replace(/-/g, "_")}-${projectSlug.replace(/-/g, "_")}`; if (projectLocal.deploymentName.startsWith(expectedPrefix)) { return projectLocal; } logVerbose( `Project-local deployment ${projectLocal.deploymentName} doesn't match expected prefix ${expectedPrefix}` ); } const prefix = `local-${teamSlug.replace(/-/g, "_")}-${projectSlug.replace(/-/g, "_")}`; const legacyDeployments = await getLegacyLocalDeployments(ctx); const existingDeploymentForProject = legacyDeployments.find( (d) => d.deploymentName.startsWith(prefix) ); if (existingDeploymentForProject === void 0) { return null; } return { deploymentName: existingDeploymentForProject.deploymentName, config: existingDeploymentForProject.config }; } async function getLegacyLocalDeployments(ctx) { const dir = rootDeploymentStateDir("local"); if (!ctx.fs.exists(dir)) { return []; } const deploymentNames = ctx.fs.listDir(dir).map((d) => d.name).filter((d) => d.startsWith("local-")); return deploymentNames.flatMap((deploymentName) => { const legacyDir = legacyDeploymentStateDir("local", deploymentName); const config = loadDeploymentConfigFromDir(ctx, legacyDir); if (config !== null) { return [{ deploymentName, config }]; } return []; }); } async function getLocalDeployments(ctx) { const deployments = []; const projectLocal = loadProjectLocalConfig(ctx); if (projectLocal !== null && projectLocal.deploymentName.startsWith("local-")) { deployments.push(projectLocal); } const legacyDeployments = await getLegacyLocalDeployments(ctx); for (const legacy of legacyDeployments) { if (!deployments.some((d) => d.deploymentName === legacy.deploymentName)) { deployments.push(legacy); } } return deployments; } async function chooseFromExistingLocalDeployments(ctx) { const localDeployments = await getLocalDeployments(ctx); if (localDeployments.length === 0) { return ctx.crash({ exitCode: 1, errorType: "fatal", printedMessage: "No local deployments found. Please run `npx convex dev` while online first." }); } if (localDeployments.length === 1) { logVerbose( `Auto-selecting the only local deployment: ${localDeployments[0].deploymentName}` ); return localDeployments[0]; } return promptSearch(ctx, { message: "Choose from an existing local deployment:", choices: localDeployments.map((d) => ({ name: d.deploymentName, value: d })) }); } export async function importDefaultEnvVars(ctx, { teamSlug, projectSlug, deploymentName, deploymentUrl, adminKey }) { showSpinner("Importing default env vars..."); const project = await getProjectDetails(ctx, { kind: "teamAndProjectSlugs", teamSlug, projectSlug }); const defaults = await defaultEnvBackend(ctx, project.id, "dev").list(); if (defaults.length === 0) { logFinishedStep("No default env vars to import."); return; } const deployment = { deploymentUrl, deploymentFields: { deploymentName, deploymentType: "local", projectSlug, teamSlug } }; await withRunningBackend({ ctx, deployment, action: async () => { await deploymentEnvBackend(ctx, { deploymentUrl, adminKey }).update( defaults.map((v) => ({ name: v.name, value: v.value })) ); logFinishedStep( `Imported ${defaults.length} environment ${defaults.length === 1 ? "variable" : "variables"} from default environment variables: ${defaults.map((v) => v.name).join(", ")}` ); } }); } //# sourceMappingURL=localDeployment.js.map