Skip to content

Commit

Permalink
fix: cache deno imports on deploy
Browse files Browse the repository at this point in the history
  • Loading branch information
luckasRanarison committed Nov 26, 2024
1 parent 9671d60 commit f536dd9
Show file tree
Hide file tree
Showing 5 changed files with 113 additions and 43 deletions.
22 changes: 5 additions & 17 deletions src/typegate/src/runtimes/deno/deno.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import { DenoMessenger } from "./deno_messenger.ts";
import type { Task } from "./shared_types.ts";
import { path } from "compress/deps.ts";
import { globalConfig as config } from "../../config.ts";
import { createArtifactMeta } from "../utils/deno.ts";

const predefinedFuncs: Record<string, Resolver<Record<string, unknown>>> = {
identity: ({ _, ...args }) => args,
Expand Down Expand Up @@ -81,23 +82,10 @@ export class DenoRuntime extends Runtime {
} else if (mat.name === "module") {
const matData = mat.data;
const entryPoint = artifacts[matData.entryPoint as string];
const deps = (matData.deps as string[]).map((dep) => artifacts[dep]);

const moduleMeta = {
typegraphName: typegraphName,
relativePath: entryPoint.path,
hash: entryPoint.hash,
sizeInBytes: entryPoint.size,
};

const depMetas = deps.map((dep) => {
return {
typegraphName: typegraphName,
relativePath: dep.path,
hash: dep.hash,
sizeInBytes: dep.size,
};
});
const depMetas = (matData.deps as string[]).map((dep) =>
createArtifactMeta(typegraphName, artifacts[dep]),
);
const moduleMeta = createArtifactMeta(typegraphName, entryPoint);

// Note:
// Worker destruction seems to have no effect on the import cache? (deinit() => stop(worker))
Expand Down
53 changes: 53 additions & 0 deletions src/typegate/src/runtimes/deno/hooks.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
import { getLogger } from "../../log.ts";
import { PushFailure, PushHandler } from "../../typegate/hooks.ts";
import { createArtifactMeta } from "../utils/deno.ts";

const logger = getLogger("typegate");

export class DenoFailure extends Error {
failure: PushFailure;

constructor(message: string) {
super(message);
this.failure = { reason: "DenoImportError", message };
}
}

export const cacheModules: PushHandler = async (
typegraph,
_secretManager,
_response,
artifactStore,
) => {
const { title } = typegraph.types[0];
const { artifacts } = typegraph.meta;

for (const mat of typegraph.materializers) {
if (mat.name === "module") {
const matData = mat.data;
const entryPoint = artifacts[matData.entryPoint as string];
const moduleMeta = createArtifactMeta(title, entryPoint);
const depMetas = (matData.deps as string[]).map((dep) =>
createArtifactMeta(title, artifacts[dep]),
);
const entryModulePath = await artifactStore.getLocalPath(
moduleMeta,
depMetas,
);

logger.info("Caching deno imports");

try {
await import(entryModulePath);
} catch (error) {
console.error(error.stack);

throw new DenoFailure(
`An error occured when trying to import '${entryPoint.path}'`,
);
}
}
}

return typegraph;
};
17 changes: 17 additions & 0 deletions src/typegate/src/runtimes/utils/deno.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
// Copyright Metatype OÜ, licensed under the Mozilla Public License Version 2.0.
// SPDX-License-Identifier: MPL-2.0

import { ArtifactMeta } from "../../typegate/artifacts/mod.ts";
import { Artifact } from "../../typegraph/types.ts";

export function createArtifactMeta(
typegraphName: string,
artifact: Artifact,
): ArtifactMeta {
return {
typegraphName,
hash: artifact.hash,
sizeInBytes: artifact.size,
relativePath: artifact.path,
};
}
37 changes: 23 additions & 14 deletions src/typegate/src/typegate/hooks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,27 +3,35 @@

import type { MessageEntry, Migrations } from "../typegate/register.ts";
import type { SecretManager, TypeGraphDS } from "../typegraph/mod.ts";
import { ArtifactStore } from "./artifacts/mod.ts";

const Message = {
INFO: "info",
WARNING: "warning",
ERROR: "error",
} as const;

export type PushFailure = {
reason: "DatabaseResetRequired";
message: string;
runtimeName: string;
} | {
reason: "NullConstraintViolation";
message: string;
runtimeName: string;
column: string;
table: string;
} | {
reason: "Unknown";
message: string;
};
export type PushFailure =
| {
reason: "DatabaseResetRequired";
message: string;
runtimeName: string;
}
| {
reason: "NullConstraintViolation";
message: string;
runtimeName: string;
column: string;
table: string;
}
| {
reason: "DenoImportError";
message: string;
}
| {
reason: "Unknown";
message: string;
};

export class PushResponse {
tgName?: string;
Expand Down Expand Up @@ -74,5 +82,6 @@ export interface PushHandler {
tg: TypeGraphDS,
secretManager: SecretManager,
response: PushResponse,
artifactStore: ArtifactStore,
): Promise<TypeGraphDS>;
}
27 changes: 15 additions & 12 deletions src/typegate/src/typegate/mod.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import { type PushHandler, PushResponse } from "../typegate/hooks.ts";
import { upgradeTypegraph } from "../typegraph/versions.ts";
import { parseGraphQLTypeGraph } from "../transports/graphql/typegraph.ts";
import * as PrismaHooks from "../runtimes/prisma/hooks/mod.ts";
import * as DenoHooks from "../runtimes/deno/hooks.ts";
import {
type RuntimeResolver,
SecretManager,
Expand All @@ -31,9 +32,8 @@ import { resolveIdentifier } from "../services/middlewares.ts";
import { handleGraphQL } from "../services/graphql_service.ts";
import { getLogger } from "../log.ts";
import { MigrationFailure } from "../runtimes/prisma/hooks/run_migrations.ts";
import introspectionJson from "../typegraphs/introspection.json" with {
type: "json",
};
import { DenoFailure } from "../runtimes/deno/hooks.ts";
import introspectionJson from "../typegraphs/introspection.json" with { type: "json" };
import { ArtifactService } from "../services/artifact_service.ts";
import type { ArtifactStore } from "./artifacts/mod.ts";
// TODO move from tests (MET-497)
Expand Down Expand Up @@ -170,6 +170,7 @@ export class Typegate implements AsyncDisposable {
this.#onPush((tg) => Promise.resolve(parseGraphQLTypeGraph(tg)));
this.#onPush(PrismaHooks.generateSchema);
this.#onPush(PrismaHooks.runMigrations);
this.#onPush(DenoHooks.cacheModules);
this.#artifactService = new ArtifactService(artifactStore);
}

Expand All @@ -192,13 +193,15 @@ export class Typegate implements AsyncDisposable {

for (const handler of this.#onPushHooks) {
try {
res = await handler(res, secretManager, response);
res = await handler(res, secretManager, response, this.artifactStore);
} catch (e) {
logger.error(`Error in onPush hook: ${e}`);
// FIXME: MigrationFailur err message parser doesn't support all errors like
// can't reach database errs
if (e instanceof MigrationFailure && e.errors[0]) {
response.setFailure(e.errors[0]);
} else if (e instanceof DenoFailure) {
response.setFailure(e.failure);
} else {
response.setFailure({
reason: "Unknown",
Expand Down Expand Up @@ -399,14 +402,14 @@ export class Typegate implements AsyncDisposable {

const introspection = enableIntrospection
? await TypeGraph.init(
this,
introspectionDef,
new SecretManager(introspectionDef, {}),
{
typegraph: TypeGraphRuntime.init(tgDS, [], {}),
},
null,
)
this,
introspectionDef,
new SecretManager(introspectionDef, {}),
{
typegraph: TypeGraphRuntime.init(tgDS, [], {}),
},
null,
)
: null;

const tg = await TypeGraph.init(
Expand Down

0 comments on commit f536dd9

Please sign in to comment.