mirror of
https://github.com/formbricks/formbricks.git
synced 2026-01-26 02:58:48 -06:00
* added input validation using zod * changed console.log to console.error * fix formatting issues --------- Co-authored-by: Matthias Nannt <mail@matthiasnannt.com>
122 lines
3.2 KiB
TypeScript
122 lines
3.2 KiB
TypeScript
import "server-only";
|
|
import z from "zod";
|
|
import { prisma } from "@formbricks/database";
|
|
import { TApiKey, TApiKeyCreateInput, ZApiKeyCreateInput } from "@formbricks/types/v1/apiKeys";
|
|
import { Prisma } from "@prisma/client";
|
|
import { getHash } from "../crypto";
|
|
import { createHash, randomBytes } from "crypto";
|
|
import { DatabaseError, InvalidInputError, ResourceNotFoundError } from "@formbricks/types/v1/errors";
|
|
import { cache } from "react";
|
|
import { validateInputs } from "../utils/validate";
|
|
import { ZId } from "@formbricks/types/v1/environment";
|
|
|
|
export const getApiKey = async (apiKey: string): Promise<TApiKey | null> => {
|
|
validateInputs([apiKey, z.string()]);
|
|
if (!apiKey) {
|
|
throw new InvalidInputError("API key cannot be null or undefined.");
|
|
}
|
|
|
|
try {
|
|
const apiKeyData = await prisma.apiKey.findUnique({
|
|
where: {
|
|
hashedKey: getHash(apiKey),
|
|
},
|
|
});
|
|
|
|
if (!apiKeyData) {
|
|
throw new ResourceNotFoundError("API Key", apiKey);
|
|
}
|
|
|
|
return apiKeyData;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError("Database operation failed");
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
export const getApiKeys = cache(async (environmentId: string): Promise<TApiKey[]> => {
|
|
validateInputs([environmentId, ZId]);
|
|
try {
|
|
const apiKeys = await prisma.apiKey.findMany({
|
|
where: {
|
|
environmentId,
|
|
},
|
|
});
|
|
|
|
return apiKeys;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError("Database operation failed");
|
|
}
|
|
throw error;
|
|
}
|
|
});
|
|
|
|
export const hashApiKey = (key: string): string => createHash("sha256").update(key).digest("hex");
|
|
|
|
export async function createApiKey(environmentId: string, apiKeyData: TApiKeyCreateInput): Promise<TApiKey> {
|
|
validateInputs([environmentId, ZId], [apiKeyData, ZApiKeyCreateInput]);
|
|
try {
|
|
const key = randomBytes(16).toString("hex");
|
|
const hashedKey = hashApiKey(key);
|
|
|
|
const result = await prisma.apiKey.create({
|
|
data: {
|
|
...apiKeyData,
|
|
hashedKey,
|
|
environment: { connect: { id: environmentId } },
|
|
},
|
|
});
|
|
|
|
return { ...result, apiKey: key };
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError("Database operation failed");
|
|
}
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
export const getApiKeyFromKey = async (apiKey: string): Promise<TApiKey | null> => {
|
|
validateInputs([apiKey, z.string()]);
|
|
if (!apiKey) {
|
|
throw new InvalidInputError("API key cannot be null or undefined.");
|
|
}
|
|
|
|
try {
|
|
const apiKeyData = await prisma.apiKey.findUnique({
|
|
where: {
|
|
hashedKey: getHash(apiKey),
|
|
},
|
|
});
|
|
|
|
return apiKeyData;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError("Database operation failed");
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
export const deleteApiKey = async (id: string): Promise<void> => {
|
|
validateInputs([id, ZId]);
|
|
try {
|
|
await prisma.apiKey.delete({
|
|
where: {
|
|
id: id,
|
|
},
|
|
});
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError("Database operation failed");
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
};
|