components.ts•18.3 kB
import path from "path";
import { Context } from "../../bundler/context.js";
import {
changeSpinner,
logFinishedStep,
logMessage,
} from "../../bundler/log.js";
import {
ProjectConfig,
configFromProjectConfig,
debugIsolateEndpointBundles,
getFunctionsDirectoryPath,
readProjectConfig,
pullConfig,
diffConfig,
} from "./config.js";
import {
finishPush,
reportPushCompleted,
startPush,
waitForSchema,
} from "./deploy2.js";
import { version } from "../version.js";
import { PushOptions, runNonComponentsPush } from "./push.js";
import { ensureHasConvexDependency, functionsDir } from "./utils/utils.js";
import {
bundleDefinitions,
bundleImplementations,
componentGraph,
} from "./components/definition/bundle.js";
import { isComponentDirectory } from "./components/definition/directoryStructure.js";
import {
doFinalComponentCodegen,
doInitialComponentCodegen,
CodegenOptions,
doInitCodegen,
doCodegen,
} from "./codegen.js";
import {
AppDefinitionConfig,
ComponentDefinitionConfig,
} from "./deployApi/definitionConfig.js";
import { typeCheckFunctionsInMode, TypeCheckMode } from "./typecheck.js";
import { withTmpDir } from "../../bundler/fs.js";
import { handleDebugBundlePath } from "./debugBundlePath.js";
import chalk from "chalk";
import { StartPushRequest, StartPushResponse } from "./deployApi/startPush.js";
import {
deploymentSelectionWithinProjectFromOptions,
loadSelectedDeploymentCredentials,
} from "./api.js";
import { FinishPushDiff } from "./deployApi/finishPush.js";
import { Reporter, Span } from "./tracing.js";
import {
DEFINITION_FILENAME_JS,
DEFINITION_FILENAME_TS,
} from "./components/constants.js";
import { DeploymentSelection } from "./deploymentSelection.js";
import { deploymentDashboardUrlPage } from "./dashboard.js";
import { formatIndex } from "./indexes.js";
async function findComponentRootPath(ctx: Context, functionsDir: string) {
// Default to `.ts` but fallback to `.js` if not present.
let componentRootPath = path.resolve(
path.join(functionsDir, DEFINITION_FILENAME_TS),
);
if (!ctx.fs.exists(componentRootPath)) {
componentRootPath = path.resolve(
path.join(functionsDir, DEFINITION_FILENAME_JS),
);
}
return componentRootPath;
}
export async function runCodegen(
ctx: Context,
deploymentSelection: DeploymentSelection,
options: CodegenOptions,
) {
// This also ensures the current directory is the project root.
await ensureHasConvexDependency(ctx, "codegen");
const { configPath, projectConfig } = await readProjectConfig(ctx);
const functionsDirectoryPath = functionsDir(configPath, projectConfig);
const componentRootPath = await findComponentRootPath(
ctx,
functionsDirectoryPath,
);
if (options.init) {
await doInitCodegen(ctx, functionsDirectoryPath, false, {
dryRun: options.dryRun,
debug: options.debug,
});
}
if (
(ctx.fs.exists(componentRootPath) ||
process.env.USE_LEGACY_PUSH === undefined) &&
!options.systemUdfs
) {
// Early exit for a better error message trying to use a preview key.
if (deploymentSelection.kind === "preview") {
return await ctx.crash({
exitCode: 1,
errorType: "invalid filesystem data",
printedMessage: `Codegen requires an existing deployment so doesn't support CONVEX_DEPLOY_KEY.\nGenerate code in dev and commit it to the repo instead.\nhttps://docs.convex.dev/understanding/best-practices/other-recommendations#check-generated-code-into-version-control`,
});
}
const selectionWithinProject =
deploymentSelectionWithinProjectFromOptions(options);
const credentials = await loadSelectedDeploymentCredentials(
ctx,
deploymentSelection,
selectionWithinProject,
);
await startComponentsPushAndCodegen(
ctx,
Span.noop(),
projectConfig,
configPath,
{
...options,
deploymentName: credentials.deploymentFields?.deploymentName ?? null,
url: credentials.url,
adminKey: credentials.adminKey,
generateCommonJSApi: options.commonjs,
verbose: options.dryRun,
codegen: true,
liveComponentSources: options.liveComponentSources,
typecheckComponents: false,
debugNodeApis: options.debugNodeApis,
},
);
} else {
if (options.typecheck !== "disable") {
logMessage(chalk.gray("Running TypeScript typecheck…"));
}
await doCodegen(ctx, functionsDirectoryPath, options.typecheck, {
dryRun: options.dryRun,
debug: options.debug,
generateCommonJSApi: options.commonjs,
});
}
}
export async function runPush(ctx: Context, options: PushOptions) {
const { configPath, projectConfig } = await readProjectConfig(ctx);
const convexDir = functionsDir(configPath, projectConfig);
const componentRootPath = await findComponentRootPath(ctx, convexDir);
if (
!ctx.fs.exists(componentRootPath) &&
process.env.USE_LEGACY_PUSH !== undefined
) {
await runNonComponentsPush(ctx, options, configPath, projectConfig);
} else {
await runComponentsPush(ctx, options, configPath, projectConfig);
}
}
async function startComponentsPushAndCodegen(
ctx: Context,
parentSpan: Span,
projectConfig: ProjectConfig,
configPath: string,
options: {
typecheck: TypeCheckMode;
typecheckComponents: boolean;
adminKey: string;
url: string;
deploymentName: string | null;
verbose: boolean;
debugBundlePath?: string | undefined;
dryRun: boolean;
generateCommonJSApi?: boolean;
debug: boolean;
writePushRequest?: string | undefined;
codegen: boolean;
liveComponentSources?: boolean;
debugNodeApis: boolean;
},
): Promise<StartPushResponse | null> {
const convexDir = await getFunctionsDirectoryPath(ctx);
// '.' means use the process current working directory, it's the default behavior.
// Spelling it out here to be explicit for a future where this code can run
// from other directories.
// In esbuild the working directory is used to print error messages and resolving
// relatives paths passed to it. It generally doesn't matter for resolving imports,
// imports are resolved from the file where they are written.
const absWorkingDir = path.resolve(".");
const isComponent = isComponentDirectory(ctx, convexDir, true);
if (isComponent.kind === "err") {
return await ctx.crash({
exitCode: 1,
errorType: "invalid filesystem data",
printedMessage: `Invalid component root directory (${isComponent.why}): ${convexDir}`,
});
}
const rootComponent = isComponent.component;
changeSpinner("Finding component definitions...");
// Create a list of relevant component directories. These are just for knowing
// while directories to bundle in bundleDefinitions and bundleImplementations.
// This produces a bundle in memory as a side effect but it's thrown away.
const { components, dependencyGraph } = await parentSpan.enterAsync(
"componentGraph",
() =>
componentGraph(
ctx,
absWorkingDir,
rootComponent,
!!options.liveComponentSources,
options.verbose,
),
);
if (options.codegen) {
changeSpinner("Generating server code...");
await parentSpan.enterAsync("doInitialComponentCodegen", () =>
withTmpDir(async (tmpDir) => {
await doInitialComponentCodegen(ctx, tmpDir, rootComponent, options);
for (const directory of components.values()) {
await doInitialComponentCodegen(ctx, tmpDir, directory, options);
}
}),
);
}
changeSpinner("Bundling component definitions...");
// This bundles everything but the actual function definitions
const {
appDefinitionSpecWithoutImpls,
componentDefinitionSpecsWithoutImpls,
} = await parentSpan.enterAsync("bundleDefinitions", () =>
bundleDefinitions(
ctx,
absWorkingDir,
dependencyGraph,
rootComponent,
// Note that this *includes* the root component.
[...components.values()],
!!options.liveComponentSources,
),
);
if (options.debugNodeApis) {
await debugIsolateEndpointBundles(ctx, projectConfig, configPath);
logFinishedStep(
"All non-'use node' entry points successfully bundled. Skipping rest of push.",
);
return null;
}
changeSpinner("Bundling component schemas and implementations...");
const { appImplementation, componentImplementations } =
await parentSpan.enterAsync("bundleImplementations", () =>
bundleImplementations(
ctx,
rootComponent,
[...components.values()],
projectConfig.node.externalPackages,
options.liveComponentSources ? ["@convex-dev/component-source"] : [],
options.verbose,
),
);
if (options.debugBundlePath) {
const { config: localConfig } = await configFromProjectConfig(
ctx,
projectConfig,
configPath,
options.verbose,
);
// TODO(ENG-6972): Actually write the bundles for components.
await handleDebugBundlePath(ctx, options.debugBundlePath, localConfig);
logMessage(
`Wrote bundle and metadata for modules in the root to ${options.debugBundlePath}. Skipping rest of push.`,
);
return null;
}
// We're just using the version this CLI is running with for now.
// This could be different than the version of `convex` the app runs with
// if the CLI is installed globally.
// TODO: This should be the version of the `convex` package used by each
// component, and may be different for each component.
const udfServerVersion = version;
const appDefinition: AppDefinitionConfig = {
...appDefinitionSpecWithoutImpls,
...appImplementation,
udfServerVersion,
};
const componentDefinitions: ComponentDefinitionConfig[] = [];
for (const componentDefinition of componentDefinitionSpecsWithoutImpls) {
const impl = componentImplementations.filter(
(impl) => impl.definitionPath === componentDefinition.definitionPath,
)[0];
if (!impl) {
return await ctx.crash({
exitCode: 1,
errorType: "fatal",
printedMessage: `missing! couldn't find ${componentDefinition.definitionPath} in ${componentImplementations.map((impl) => impl.definitionPath).toString()}`,
});
}
componentDefinitions.push({
...componentDefinition,
...impl,
udfServerVersion,
});
}
const startPushRequest = {
adminKey: options.adminKey,
dryRun: options.dryRun,
functions: projectConfig.functions,
appDefinition,
componentDefinitions,
nodeDependencies: appImplementation.externalNodeDependencies,
nodeVersion: projectConfig.node.nodeVersion,
};
if (options.writePushRequest) {
const pushRequestPath = path.resolve(options.writePushRequest);
ctx.fs.writeUtf8File(
`${pushRequestPath}.json`,
JSON.stringify(startPushRequest),
);
return null;
}
logStartPushSizes(parentSpan, startPushRequest);
changeSpinner("Uploading functions to Convex...");
const startPushResponse = await parentSpan.enterAsync("startPush", (span) =>
startPush(ctx, span, startPushRequest, options),
);
if (options.verbose) {
logMessage("startPush: " + JSON.stringify(startPushResponse, null, 2));
}
if (options.codegen) {
changeSpinner("Generating TypeScript bindings...");
await parentSpan.enterAsync("doFinalComponentCodegen", () =>
withTmpDir(async (tmpDir) => {
await doFinalComponentCodegen(
ctx,
tmpDir,
rootComponent,
rootComponent,
startPushResponse,
options,
);
for (const directory of components.values()) {
await doFinalComponentCodegen(
ctx,
tmpDir,
rootComponent,
directory,
startPushResponse,
options,
);
}
}),
);
}
changeSpinner("Running TypeScript...");
await parentSpan.enterAsync("typeCheckFunctionsInMode", async () => {
await typeCheckFunctionsInMode(ctx, options.typecheck, rootComponent.path);
if (options.typecheckComponents) {
for (const directory of components.values()) {
await typeCheckFunctionsInMode(ctx, options.typecheck, directory.path);
}
}
});
return startPushResponse;
}
function logStartPushSizes(span: Span, startPushRequest: StartPushRequest) {
let v8Size = 0;
let v8Count = 0;
let nodeSize = 0;
let nodeCount = 0;
for (const componentDefinition of startPushRequest.componentDefinitions) {
for (const module of componentDefinition.functions) {
if (module.environment === "isolate") {
v8Size += module.source.length + (module.sourceMap ?? "").length;
v8Count += 1;
} else if (module.environment === "node") {
nodeSize += module.source.length + (module.sourceMap ?? "").length;
nodeCount += 1;
}
}
}
span.setProperty("v8_size", v8Size.toString());
span.setProperty("v8_count", v8Count.toString());
span.setProperty("node_size", nodeSize.toString());
span.setProperty("node_count", nodeCount.toString());
}
export async function runComponentsPush(
ctx: Context,
options: PushOptions,
configPath: string,
projectConfig: ProjectConfig,
) {
const reporter = new Reporter();
const pushSpan = Span.root(reporter, "runComponentsPush");
pushSpan.setProperty("cli_version", version);
const verbose = options.verbose || options.dryRun;
await ensureHasConvexDependency(ctx, "push");
const startPushResponse = await pushSpan.enterAsync(
"startComponentsPushAndCodegen",
(span) =>
startComponentsPushAndCodegen(
ctx,
span,
projectConfig,
configPath,
options,
),
);
if (!startPushResponse) {
return;
}
await pushSpan.enterAsync("waitForSchema", (span) =>
waitForSchema(ctx, span, startPushResponse, options),
);
const remoteConfigWithModuleHashes = await pullConfig(
ctx,
undefined,
undefined,
options.url,
options.adminKey,
);
const { config: localConfig } = await configFromProjectConfig(
ctx,
projectConfig,
configPath,
options.verbose,
);
changeSpinner("Diffing local code and deployment state...");
const { diffString } = diffConfig(
remoteConfigWithModuleHashes,
localConfig,
false,
);
if (verbose) {
logFinishedStep(
`Remote config ${
options.dryRun ? "would" : "will"
} be overwritten with the following changes:\n ` +
diffString.replace(/\n/g, "\n "),
);
}
const finishPushResponse = await pushSpan.enterAsync("finishPush", (span) =>
finishPush(ctx, span, startPushResponse, options),
);
printDiff(startPushResponse, finishPushResponse, options);
pushSpan.end();
// Asynchronously report that the push completed.
if (!options.dryRun) {
void reportPushCompleted(ctx, options.adminKey, options.url, reporter);
}
}
function printDiff(
startPushResponse: StartPushResponse,
finishPushResponse: FinishPushDiff,
opts: { verbose: boolean; dryRun: boolean; deploymentName: string | null },
) {
if (opts.verbose) {
const diffString = JSON.stringify(finishPushResponse, null, 2);
logMessage(diffString);
return;
}
const indexDiffs = startPushResponse.schemaChange.indexDiffs;
const { componentDiffs } = finishPushResponse;
// Print out index diffs for the root component.
let rootDiff = indexDiffs?.[""] || componentDiffs[""]?.indexDiff;
if (rootDiff) {
if (rootDiff.removed_indexes.length > 0) {
let msg = `${opts.dryRun ? "Would delete" : "Deleted"} table indexes:\n`;
for (const index of rootDiff.removed_indexes) {
msg += ` [-] ${formatIndex(index)}\n`;
}
msg = msg.slice(0, -1); // strip last new line
logFinishedStep(msg);
}
const addedEnabled = rootDiff.added_indexes.filter((i) => !i.staged);
if (addedEnabled.length > 0) {
let msg = `${opts.dryRun ? "Would add" : "Added"} table indexes:\n`;
for (const index of addedEnabled) {
msg += ` [+] ${formatIndex(index)}\n`;
}
msg = msg.slice(0, -1); // strip last new line
logFinishedStep(msg);
}
const addedStaged = rootDiff.added_indexes.filter((i) => i.staged);
if (addedStaged.length > 0) {
let msg = `${opts.dryRun ? "Would add" : "Added"} staged table indexes:\n`;
for (const index of addedStaged) {
const table = index.name.split(".")[0];
const progressLink = deploymentDashboardUrlPage(
opts.deploymentName,
`/data?table=${table}&showIndexes=true`,
);
msg += ` [+] ${formatIndex(index)}\n`;
msg += ` See progress: ${progressLink}\n`;
}
msg = msg.slice(0, -1); // strip last new line
logFinishedStep(msg);
}
if (rootDiff.enabled_indexes && rootDiff.enabled_indexes.length > 0) {
let msg = opts.dryRun
? `These indexes would be enabled:\n`
: `These indexes are now enabled:\n`;
for (const index of rootDiff.enabled_indexes) {
msg += ` [*] ${formatIndex(index)}\n`;
}
msg = msg.slice(0, -1); // strip last new line
logFinishedStep(msg);
}
if (rootDiff.disabled_indexes && rootDiff.disabled_indexes.length > 0) {
let msg = opts.dryRun
? `These indexes would be staged:\n`
: `These indexes are now staged:\n`;
for (const index of rootDiff.disabled_indexes) {
msg += ` [*] ${formatIndex(index)}\n`;
}
msg = msg.slice(0, -1); // strip last new line
logFinishedStep(msg);
}
}
// Only show component level diffs for other components.
for (const [componentPath, componentDiff] of Object.entries(componentDiffs)) {
if (componentPath === "") {
continue;
}
if (componentDiff.diffType.type === "create") {
logFinishedStep(`Installed component ${componentPath}.`);
}
if (componentDiff.diffType.type === "unmount") {
logFinishedStep(`Unmounted component ${componentPath}.`);
}
if (componentDiff.diffType.type === "remount") {
logFinishedStep(`Remounted component ${componentPath}.`);
}
}
}