UNPKG

convex

Version:

Client for the Convex Cloud

272 lines (268 loc) 9.29 kB
"use strict"; import { Command, Option } from "@commander-js/extra-typings"; import { logFailure, logFinishedStep, logMessage, oneoffContext } from "../bundler/context.js"; import { checkAuthorization, performLogin } from "./lib/login.js"; import { loadUuidForAnonymousUser } from "./lib/localDeployment/filePaths.js"; import { handleLinkToProject, listExistingAnonymousDeployments } from "./lib/localDeployment/anonymous.js"; import { DASHBOARD_HOST, deploymentDashboardUrlPage, teamDashboardUrl } from "./lib/dashboard.js"; import { promptSearch, promptYesNo } from "./lib/utils/prompts.js"; import { bigBrainAPI, validateOrSelectTeam } from "./lib/utils/utils.js"; import { selectProject, updateEnvAndConfigForDeploymentSelection } from "./configure.js"; import { getDeploymentSelection, shouldAllowAnonymousDevelopment } from "./lib/deploymentSelection.js"; import { removeAnonymousPrefix } from "./lib/deployment.js"; export const login = new Command("login").description("Login to Convex").allowExcessArguments(false).option( "--device-name <name>", "Provide a name for the device being authorized" ).option( "-f, --force", "Proceed with login even if a valid access token already exists for this device" ).option( "--no-open", "Don't automatically open the login link in the default browser" ).addOption( new Option( "--login-flow <mode>", `How to log in; defaults to guessing based on the environment.` ).choices(["paste", "auto", "poll"]).default("auto") ).addOption(new Option("--link-deployments").hideHelp()).addOption(new Option("--override-auth-url <url>").hideHelp()).addOption(new Option("--override-auth-client <id>").hideHelp()).addOption(new Option("--override-auth-username <username>").hideHelp()).addOption(new Option("--override-auth-password <password>").hideHelp()).addOption(new Option("--override-access-token <token>").hideHelp()).addOption(new Option("--accept-opt-ins").hideHelp()).addOption(new Option("--dump-access-token").hideHelp()).addOption(new Option("--check-login").hideHelp()).action(async (options, cmd) => { const ctx = await oneoffContext({ url: void 0, adminKey: void 0, envFile: void 0 }); if (!options.force && await checkAuthorization(ctx, !!options.acceptOptIns)) { logFinishedStep( ctx, "This device has previously been authorized and is ready for use with Convex." ); await handleLinkingDeployments(ctx, { interactive: !!options.linkDeployments }); return; } if (!options.force && options.checkLogin) { const isLoggedIn = await checkAuthorization(ctx, !!options.acceptOptIns); if (!isLoggedIn) { return ctx.crash({ exitCode: 1, errorType: "fatal", errForSentry: "You are not logged in.", printedMessage: "You are not logged in." }); } } if (!!options.overrideAuthUsername !== !!options.overrideAuthPassword) { cmd.error( "If overriding credentials, both username and password must be provided" ); } const uuid = loadUuidForAnonymousUser(ctx); await performLogin(ctx, { ...options, anonymousId: uuid }); await handleLinkingDeployments(ctx, { interactive: !!options.linkDeployments }); }); async function handleLinkingDeployments(ctx, args) { if (!shouldAllowAnonymousDevelopment()) { return; } const anonymousDeployments = await listExistingAnonymousDeployments(ctx); if (anonymousDeployments.length === 0) { if (args.interactive) { logMessage( ctx, "It doesn't look like you have any deployments to link. You can run `npx convex dev` to set up a new project or select an existing one." ); } return; } if (!args.interactive) { const message = getMessage( anonymousDeployments.map((d) => d.deploymentName) ); const createProjects = await promptYesNo(ctx, { message, default: true }); if (!createProjects) { logMessage( ctx, "Not linking your existing deployments. If you want to link them later, run `npx convex login --link-deployments`." ); logMessage( ctx, `Visit ${DASHBOARD_HOST} or run \`npx convex dev\` to get started with your new account.` ); return; } const { teamSlug } = await validateOrSelectTeam( ctx, void 0, "Choose a team for your deployments:" ); const projectsRemaining = await getProjectsRemaining(ctx, teamSlug); if (anonymousDeployments.length > projectsRemaining) { logFailure( ctx, `You have ${anonymousDeployments.length} deployments to link, but only have ${projectsRemaining} projects remaining. If you'd like to choose which ones to link, run this command with the --link-deployments flag.` ); return; } const deploymentSelection2 = await getDeploymentSelection(ctx, { url: void 0, adminKey: void 0, envFile: void 0 }); const configuredDeployment2 = deploymentSelection2.kind === "anonymous" ? deploymentSelection2.deploymentName : null; let dashboardUrl = teamDashboardUrl(teamSlug); for (const deployment of anonymousDeployments) { const linkedDeployment = await handleLinkToProject(ctx, { deploymentName: deployment.deploymentName, teamSlug, projectSlug: null }); logFinishedStep( ctx, `Added ${deployment.deploymentName} to project ${linkedDeployment.projectSlug}` ); if (deployment.deploymentName === configuredDeployment2) { await updateEnvAndConfigForDeploymentSelection( ctx, { url: linkedDeployment.deploymentUrl, deploymentName: linkedDeployment.deploymentName, teamSlug, projectSlug: linkedDeployment.projectSlug, deploymentType: "local" }, configuredDeployment2 ); dashboardUrl = deploymentDashboardUrlPage( linkedDeployment.deploymentName, "" ); } } logFinishedStep( ctx, `Sucessfully linked your deployments! Visit ${dashboardUrl} to get started.` ); return; } const deploymentSelection = await getDeploymentSelection(ctx, { url: void 0, adminKey: void 0, envFile: void 0 }); const configuredDeployment = deploymentSelection.kind === "anonymous" ? deploymentSelection.deploymentName : null; while (true) { logMessage( ctx, getDeploymentListMessage( anonymousDeployments.map((d) => d.deploymentName) ) ); const updatedAnonymousDeployments = await listExistingAnonymousDeployments(ctx); const deploymentToLink = await promptSearch(ctx, { message: "Which deployment would you like to link to your account?", choices: updatedAnonymousDeployments.map((d) => ({ name: d.deploymentName, value: d.deploymentName })) }); const { teamSlug } = await validateOrSelectTeam( ctx, void 0, "Choose a team for your deployment:" ); const { projectSlug } = await selectProject(ctx, "ask", { team: teamSlug, devDeployment: "local", defaultProjectName: removeAnonymousPrefix(deploymentToLink) }); const linkedDeployment = await handleLinkToProject(ctx, { deploymentName: deploymentToLink, teamSlug, projectSlug }); logFinishedStep( ctx, `Added ${deploymentToLink} to project ${linkedDeployment.projectSlug}` ); if (deploymentToLink === configuredDeployment) { await updateEnvAndConfigForDeploymentSelection( ctx, { url: linkedDeployment.deploymentUrl, deploymentName: linkedDeployment.deploymentName, teamSlug, projectSlug: linkedDeployment.projectSlug, deploymentType: "local" }, configuredDeployment ); } const shouldContinue = await promptYesNo(ctx, { message: "Would you like to link another deployment?", default: true }); if (!shouldContinue) { break; } } } async function getProjectsRemaining(ctx, teamSlug) { const response = await bigBrainAPI({ ctx, method: "GET", url: `/api/teams/${teamSlug}/projects_remaining` }); return response.projectsRemaining; } function getDeploymentListMessage(anonymousDeploymentNames) { let message = `You have ${anonymousDeploymentNames.length} existing deployments.`; message += ` Deployments:`; for (const deploymentName of anonymousDeploymentNames) { message += ` - ${deploymentName}`; } return message; } function getMessage(anonymousDeploymentNames) { if (anonymousDeploymentNames.length === 1) { return `Would you like to link your existing deployment to your account? ("${anonymousDeploymentNames[0]}")`; } let message = `You have ${anonymousDeploymentNames.length} existing deployments. Would you like to link them to your account?`; message += ` Deployments:`; for (const deploymentName of anonymousDeploymentNames) { message += ` - ${deploymentName}`; } message += ` You can alternatively run \`npx convex login --link-deployments\` to interactively choose which deployments to add.`; return message; } //# sourceMappingURL=login.js.map