key.ts 2.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. import { z } from "zod"
  2. import { fn } from "./util/fn"
  3. import { Actor } from "./actor"
  4. import { and, Database, eq, isNull, sql } from "./drizzle"
  5. import { Identifier } from "./identifier"
  6. import { KeyTable } from "./schema/key.sql"
  7. import { UserTable } from "./schema/user.sql"
  8. import { AuthTable } from "./schema/auth.sql"
  9. export namespace Key {
  10. export const list = fn(z.void(), async () => {
  11. const keys = await Database.use((tx) =>
  12. tx
  13. .select({
  14. id: KeyTable.id,
  15. name: KeyTable.name,
  16. key: KeyTable.key,
  17. timeUsed: KeyTable.timeUsed,
  18. userID: KeyTable.userID,
  19. email: AuthTable.subject,
  20. })
  21. .from(KeyTable)
  22. .innerJoin(UserTable, and(eq(KeyTable.userID, UserTable.id), eq(KeyTable.workspaceID, UserTable.workspaceID)))
  23. .innerJoin(AuthTable, and(eq(UserTable.accountID, AuthTable.accountID), eq(AuthTable.provider, "email")))
  24. .where(
  25. and(
  26. eq(KeyTable.workspaceID, Actor.workspace()),
  27. isNull(KeyTable.timeDeleted),
  28. ...(Actor.userRole() === "admin" ? [] : [eq(KeyTable.userID, Actor.userID())]),
  29. ),
  30. )
  31. .orderBy(sql`${KeyTable.name} DESC`),
  32. )
  33. // only return value for user's keys
  34. return keys.map((key) => ({
  35. ...key,
  36. key: key.userID === Actor.userID() ? key.key : undefined,
  37. keyDisplay: `${key.key.slice(0, 7)}...${key.key.slice(-4)}`,
  38. }))
  39. })
  40. export const create = fn(
  41. z.object({
  42. userID: z.string(),
  43. name: z.string().min(1).max(255),
  44. }),
  45. async (input) => {
  46. const { name } = input
  47. // Generate secret key: sk- + 64 random characters (upper, lower, numbers)
  48. const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"
  49. let secretKey = "sk-"
  50. const array = new Uint32Array(64)
  51. crypto.getRandomValues(array)
  52. for (let i = 0, l = array.length; i < l; i++) {
  53. secretKey += chars[array[i] % chars.length]
  54. }
  55. const keyID = Identifier.create("key")
  56. await Database.use((tx) =>
  57. tx.insert(KeyTable).values({
  58. id: keyID,
  59. workspaceID: Actor.workspace(),
  60. userID: input.userID,
  61. name,
  62. key: secretKey,
  63. timeUsed: null,
  64. }),
  65. )
  66. return keyID
  67. },
  68. )
  69. export const remove = fn(z.object({ id: z.string() }), async (input) => {
  70. // only admin can remove other user's keys
  71. await Database.use((tx) =>
  72. tx
  73. .update(KeyTable)
  74. .set({
  75. timeDeleted: sql`now()`,
  76. })
  77. .where(
  78. and(
  79. eq(KeyTable.id, input.id),
  80. eq(KeyTable.workspaceID, Actor.workspace()),
  81. ...(Actor.userRole() === "admin" ? [] : [eq(KeyTable.userID, Actor.userID())]),
  82. ),
  83. ),
  84. )
  85. })
  86. }