import type { NextApiRequest, NextApiResponse } from "next"; import { stringify } from "querystring"; import { WEBAPP_URL } from "@calcom/lib/constants"; import { getSafeRedirectUrl } from "@calcom/lib/getSafeRedirectUrl"; import logger from "@calcom/lib/logger"; import { defaultHandler, defaultResponder } from "@calcom/lib/server"; import getAppKeysFromSlug from "../../_utils/getAppKeysFromSlug"; import getInstalledAppPath from "../../_utils/getInstalledAppPath"; import createOAuthAppCredential from "../../_utils/oauth/createOAuthAppCredential"; import { decodeOAuthState } from "../../_utils/oauth/decodeOAuthState"; import config from "../config.json"; import type { ZohoAuthCredentials } from "../types/ZohoCalendar"; import { appKeysSchema as zohoKeysSchema } from "../zod"; const log = logger.getChildLogger({ prefix: [`[[zohocalendar/api/callback]`] }); const OAUTH_BASE_URL = "https://accounts.zoho.com/oauth/v2"; async function getHandler(req: NextApiRequest, res: NextApiResponse) { const { code } = req.query; const state = decodeOAuthState(req); if (code && typeof code !== "string") { res.status(400).json({ message: "`code` must be a string" }); return; } if (!req.session?.user?.id) { return res.status(401).json({ message: "You must be logged in to do this" }); } const appKeys = await getAppKeysFromSlug(config.slug); const { client_id, client_secret } = zohoKeysSchema.parse(appKeys); const params = { client_id, grant_type: "authorization_code", client_secret, redirect_uri: `${WEBAPP_URL}/api/integrations/${config.slug}/callback`, code, }; const query = stringify(params); const response = await fetch(`${OAUTH_BASE_URL}/token?${query}`, { method: "POST", headers: { "Content-Type": "application/json; charset=utf-8", }, }); const responseBody = await response.json(); if (!response.ok || responseBody.error) { log.error("get access_token failed", responseBody); return res.redirect(`/apps/installed?error=${JSON.stringify(responseBody)}`); } const key: ZohoAuthCredentials = { access_token: responseBody.access_token, refresh_token: responseBody.refresh_token, expires_in: Math.round(+new Date() / 1000 + responseBody.expires_in), }; await createOAuthAppCredential({ appId: config.slug, type: config.type }, key, req); res.redirect( getSafeRedirectUrl(state?.returnTo) ?? getInstalledAppPath({ variant: config.variant, slug: config.slug }) ); } export default defaultHandler({ GET: Promise.resolve({ default: defaultResponder(getHandler) }), });