
* create basic app structure * add zapierSubscription model to prisma.schema * change column name triggerEvent to lower case * add zapier functionality + enpoints + adjust prisma.schema * add subscriptionType + refactor code * add app store information * create setup page to generate api key * clean code * add copy functionality in setup page * clean code * add apiKeyType and delte key when uninstalled or new key generated * clean code * use Promise.all * only approve zapier api key * clean code * fix findValidApiKey for api keys that don't expire * fix migrations * clean code * small fixes * add i18n * add README.md file * add setup guide to README.md * fix yarn.lock * Renames zapierother to zapier * Typo * Updates package name * Rename fixes * Adds zapier to the App Store seeder * Adds missing zapier to apiHandlers * Adds credential relationship to App * Rename fixes * Allows tailwind to pick up custom app-store components * Consolidates zapier_setup_instructions * Webhook fixes * Uses app relationship instead of custom type * Refactors sendPayload to accept webhook object Instead of individual parameters * refactoring * Removes unused zapier check * Update cancel.ts * Refactoring * Removes example comments * Update InstallAppButton.tsx * Type fixes * E2E fixes * Deletes all user zapier webhooks on integration removal Co-authored-by: CarinaWolli <wollencarina@gmail.com> Co-authored-by: zomars <zomars@me.com>
125 lines
3 KiB
TypeScript
125 lines
3 KiB
TypeScript
import { v4 } from "uuid";
|
|
import { z } from "zod";
|
|
|
|
import { generateUniqueAPIKey } from "@calcom/ee/lib/api/apiKeys";
|
|
|
|
import { createProtectedRouter } from "@server/createRouter";
|
|
|
|
export const apiKeysRouter = createProtectedRouter()
|
|
.query("list", {
|
|
async resolve({ ctx }) {
|
|
return await ctx.prisma.apiKey.findMany({
|
|
where: {
|
|
userId: ctx.user.id,
|
|
NOT: {
|
|
appId: "zapier",
|
|
},
|
|
},
|
|
orderBy: { createdAt: "desc" },
|
|
});
|
|
},
|
|
})
|
|
.query("findKeyOfType", {
|
|
input: z.object({
|
|
appId: z.string().optional().nullable(),
|
|
}),
|
|
async resolve({ ctx, input }) {
|
|
return await ctx.prisma.apiKey.findFirst({
|
|
where: {
|
|
AND: [
|
|
{
|
|
userId: ctx.user.id,
|
|
},
|
|
{
|
|
appId: input.appId,
|
|
},
|
|
],
|
|
},
|
|
});
|
|
},
|
|
})
|
|
.mutation("create", {
|
|
input: z.object({
|
|
note: z.string().optional().nullish(),
|
|
expiresAt: z.date().optional().nullable(),
|
|
neverExpires: z.boolean().optional(),
|
|
appId: z.string().optional().nullable(),
|
|
}),
|
|
async resolve({ ctx, input }) {
|
|
const [hashedApiKey, apiKey] = generateUniqueAPIKey();
|
|
// Here we snap never expires before deleting it so it's not passed to prisma create call.
|
|
const neverExpires = input.neverExpires;
|
|
delete input.neverExpires;
|
|
await ctx.prisma.apiKey.create({
|
|
data: {
|
|
id: v4(),
|
|
userId: ctx.user.id,
|
|
...input,
|
|
// And here we pass a null to expiresAt if never expires is true. otherwise just pass expiresAt from input
|
|
expiresAt: neverExpires ? null : input.expiresAt,
|
|
hashedKey: hashedApiKey,
|
|
},
|
|
});
|
|
const prefixedApiKey = `${process.env.API_KEY_PREFIX ?? "cal_"}${apiKey}`;
|
|
return prefixedApiKey;
|
|
},
|
|
})
|
|
.mutation("edit", {
|
|
input: z.object({
|
|
id: z.string(),
|
|
note: z.string().optional().nullish(),
|
|
expiresAt: z.date().optional(),
|
|
}),
|
|
async resolve({ ctx, input }) {
|
|
const { id, ...data } = input;
|
|
const {
|
|
apiKeys: [updatedApiKey],
|
|
} = await ctx.prisma.user.update({
|
|
where: {
|
|
id: ctx.user.id,
|
|
},
|
|
data: {
|
|
apiKeys: {
|
|
update: {
|
|
where: {
|
|
id,
|
|
},
|
|
data,
|
|
},
|
|
},
|
|
},
|
|
select: {
|
|
apiKeys: {
|
|
where: {
|
|
id,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
return updatedApiKey;
|
|
},
|
|
})
|
|
.mutation("delete", {
|
|
input: z.object({
|
|
id: z.string(),
|
|
eventTypeId: z.number().optional(),
|
|
}),
|
|
async resolve({ ctx, input }) {
|
|
const { id } = input;
|
|
await ctx.prisma.user.update({
|
|
where: {
|
|
id: ctx.user.id,
|
|
},
|
|
data: {
|
|
apiKeys: {
|
|
delete: {
|
|
id,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
return {
|
|
id,
|
|
};
|
|
},
|
|
});
|