Skip to content

Commit

Permalink
Resolved changes
Browse files Browse the repository at this point in the history
  • Loading branch information
harshsbhat committed Oct 4, 2024
1 parent 29df0b5 commit 971abe2
Show file tree
Hide file tree
Showing 6 changed files with 172 additions and 174 deletions.
173 changes: 0 additions & 173 deletions apps/api/src/routes/v1_keys_getWhoAmI.ts

This file was deleted.

Empty file.
Empty file.
Empty file.
171 changes: 171 additions & 0 deletions apps/api/src/routes/v1_keys_whoAmI.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,171 @@
import type { App } from "@/pkg/hono/app";
import { createRoute, z } from "@hono/zod-openapi";

import { rootKeyAuth } from "@/pkg/auth/root_key";
import { UnkeyApiError, openApiErrorResponses } from "@/pkg/errors";
import { sha256 } from "@unkey/hash";
import { buildUnkeyQuery } from "@unkey/rbac";

const route = createRoute({
tags: ["keys"],
operationId: "whoAmI",
method: "post",
path: "/v1/keys.whoAmI",
security: [{ bearerAuth: [] }],
request: {
body: {
required: true,
content: {
"application/json": {
schema: z.object({
key: z.string().min(1).openapi({
description: "The actual key to fetch",
example: "sk_123",
}),
}),
},
},
},
},

responses: {
200: {
description: "The configuration for a single key",
content: {
"application/json": {
schema: z.object({
id: z.string().openapi({
description: "The ID of the key",
example: "key_123",
}),
name: z.string().optional().openapi({
description: "The name of the key",
example: "API Key 1",
}),
remaining: z.number().int().optional().openapi({
description: "The remaining number of requests for the key",
example: 1000,
}),
identity: z
.object({
id: z.string().openapi({
description: "The identity ID associated with the key",
example: "id_123",
}),
externalId: z.string().openapi({
description: "The external identity ID associated with the key",
example: "ext123",
}),
})
.optional()
.openapi({
description: "The identity object associated with the key",
}),
meta: z
.record(z.unknown())
.optional()
.openapi({
description: "Metadata associated with the key",
example: { role: "admin", plan: "premium" },
}),
createdAt: z.number().int().openapi({
description: "The timestamp in milliseconds when the key was created",
example: 1620000000000,
}),
enabled: z.boolean().openapi({
description: "Whether the key is enabled",
example: true,
}),
environment: z.string().optional().openapi({
description: "The environment the key is associated with",
example: "production",
}),
}),
},
},
},
...openApiErrorResponses,
},
});

export type Route = typeof route;
export type V1KeysWhoAmIResponse = z.infer<
(typeof route.responses)[200]["content"]["application/json"]["schema"]
>;

export const registerV1KeysWhoAmi = (app: App) =>
app.openapi(route, async (c) => {
const { key: secret } = c.req.valid("json");
const { cache, db } = c.get("services");
const hash = await sha256(secret);
const { val: data, err } = await cache.keyByHash.swr(hash, async () => {
const dbRes = await db.readonly.query.keys.findFirst({
where: (table, { eq, and, isNull }) => and(eq(table.hash, hash), isNull(table.deletedAt)),
with: {
keyAuth: {
with: {
api: true,
},
},
identity: true,
},
});

if (!dbRes) {
return null;
}

return {
key: {
...dbRes,
},
api: dbRes.keyAuth.api,
identity: dbRes.identity,
} as any; // this was neccessary so that we don't need to return the workspace and other types defined in keyByHash
});

if (err) {
throw new UnkeyApiError({
code: "INTERNAL_SERVER_ERROR",
message: `unable to load key: ${err.message}`,
});
}
if (!data) {
throw new UnkeyApiError({
code: "NOT_FOUND",
message: "Key not found",
});
}
const { api, key } = data;
const auth = await rootKeyAuth(
c,
buildUnkeyQuery(({ or }) => or("*", "api.*.read_key", `api.${api.id}.read_key`)),
);

if (key.workspaceId !== auth.authorizedWorkspaceId) {
throw new UnkeyApiError({
code: "NOT_FOUND",
message: "Key not found",
});
}
let meta = key.meta ? JSON.parse(key.meta) : undefined;
if (!meta || Object.keys(meta).length === 0) {
meta = undefined;
}

return c.json({
id: key.id,
name: key.name ?? undefined,
remaining: key.remaining ?? undefined,
identity: data.identity
? {
id: data.identity.id,
externalId: data.identity.externalId,
}
: undefined,
meta: key.meta ? JSON.parse(key.meta) : undefined,
createdAt: key.createdAt.getTime(),
enabled: key.enabled,
environment: key.environment ?? undefined,
});
});
2 changes: 1 addition & 1 deletion apps/api/src/worker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,11 @@ import { registerV1ApisListKeys } from "./routes/v1_apis_listKeys";
import { registerV1KeysCreateKey } from "./routes/v1_keys_createKey";
import { registerV1KeysDeleteKey } from "./routes/v1_keys_deleteKey";
import { registerV1KeysGetKey } from "./routes/v1_keys_getKey";
import { registerV1KeysWhoAmi } from "./routes/v1_keys_getWhoAmI";
import { registerV1KeysGetVerifications } from "./routes/v1_keys_getVerifications";
import { registerV1KeysUpdate } from "./routes/v1_keys_updateKey";
import { registerV1KeysUpdateRemaining } from "./routes/v1_keys_updateRemaining";
import { registerV1KeysVerifyKey } from "./routes/v1_keys_verifyKey";
import { registerV1KeysWhoAmi } from "./routes/v1_keys_whoAmI";
import { registerV1Liveness } from "./routes/v1_liveness";
import { registerV1RatelimitLimit } from "./routes/v1_ratelimit_limit";

Expand Down

0 comments on commit 971abe2

Please sign in to comment.