
* Refactored Schedule component * Merge branch 'main' into feature/availability-page-revamp * wip * Turned value into number, many other TS tweaks * NodeJS 16x works 100% on my local, but out of scope for this already massive PR * Fixed TS errors in viewer.tsx and schedule/index.ts * Reverted next.config.js * Fixed minor remnant from moving types to @lib/types * schema comment * some changes to form handling * add comments * Turned ConfigType into number; which seems to be the value preferred by tRPC * Fixed localized time display during onboarding * Update components/ui/form/Schedule.tsx Co-authored-by: Alex Johansson <alexander@n1s.se> * Added showToast to indicate save success * Converted number to Date, and also always establish time based on current date * prevent height flickering of availability by removing mb-2 of input field * availabilty: re-added mb-2 but added min-height * Quite a few bugs discovered, but this seems functional Co-authored-by: KATT <alexander@n1s.se> Co-authored-by: Bailey Pumfleet <pumfleet@hey.com> Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com> Co-authored-by: Peer Richelsen <peeroke@gmail.com>
119 lines
3.3 KiB
TypeScript
119 lines
3.3 KiB
TypeScript
import { Prisma } from "@prisma/client";
|
|
import _ from "lodash";
|
|
|
|
import { validJson } from "@lib/jsonUtils";
|
|
|
|
const credentialData = Prisma.validator<Prisma.CredentialArgs>()({
|
|
select: { id: true, type: true },
|
|
});
|
|
|
|
type CredentialData = Prisma.CredentialGetPayload<typeof credentialData>;
|
|
|
|
export type Integration = {
|
|
installed: boolean;
|
|
type:
|
|
| "google_calendar"
|
|
| "office365_calendar"
|
|
| "zoom_video"
|
|
| "daily_video"
|
|
| "caldav_calendar"
|
|
| "apple_calendar"
|
|
| "stripe_payment";
|
|
title: string;
|
|
imageSrc: string;
|
|
description: string;
|
|
variant: "calendar" | "conferencing" | "payment";
|
|
};
|
|
|
|
export const ALL_INTEGRATIONS = [
|
|
{
|
|
installed: !!(process.env.GOOGLE_API_CREDENTIALS && validJson(process.env.GOOGLE_API_CREDENTIALS)),
|
|
type: "google_calendar",
|
|
title: "Google Calendar",
|
|
imageSrc: "integrations/google-calendar.svg",
|
|
description: "For personal and business calendars",
|
|
variant: "calendar",
|
|
},
|
|
{
|
|
installed: !!(process.env.MS_GRAPH_CLIENT_ID && process.env.MS_GRAPH_CLIENT_SECRET),
|
|
type: "office365_calendar",
|
|
title: "Office 365 / Outlook.com Calendar",
|
|
imageSrc: "integrations/outlook.svg",
|
|
description: "For personal and business calendars",
|
|
variant: "calendar",
|
|
},
|
|
{
|
|
installed: !!(process.env.ZOOM_CLIENT_ID && process.env.ZOOM_CLIENT_SECRET),
|
|
type: "zoom_video",
|
|
title: "Zoom",
|
|
imageSrc: "integrations/zoom.svg",
|
|
description: "Video Conferencing",
|
|
variant: "conferencing",
|
|
},
|
|
{
|
|
installed: !!process.env.DAILY_API_KEY,
|
|
type: "daily_video",
|
|
title: "Daily.co Video",
|
|
imageSrc: "integrations/daily.svg",
|
|
description: "Video Conferencing",
|
|
variant: "conferencing",
|
|
},
|
|
{
|
|
installed: true,
|
|
type: "caldav_calendar",
|
|
title: "CalDav Server",
|
|
imageSrc: "integrations/caldav.svg",
|
|
description: "For personal and business calendars",
|
|
variant: "calendar",
|
|
},
|
|
{
|
|
installed: true,
|
|
type: "apple_calendar",
|
|
title: "Apple Calendar",
|
|
imageSrc: "integrations/apple-calendar.svg",
|
|
description: "For personal and business calendars",
|
|
variant: "calendar",
|
|
},
|
|
{
|
|
installed: !!(
|
|
process.env.STRIPE_CLIENT_ID &&
|
|
process.env.NEXT_PUBLIC_STRIPE_PUBLIC_KEY &&
|
|
process.env.STRIPE_PRIVATE_KEY
|
|
),
|
|
type: "stripe_payment",
|
|
title: "Stripe",
|
|
imageSrc: "integrations/stripe.svg",
|
|
description: "Collect payments",
|
|
variant: "payment",
|
|
},
|
|
] as Integration[];
|
|
|
|
function getIntegrations(userCredentials: CredentialData[]) {
|
|
const integrations = ALL_INTEGRATIONS.map((integration) => {
|
|
const credentials = userCredentials
|
|
.filter((credential) => credential.type === integration.type)
|
|
.map((credential) => _.pick(credential, ["id", "type"])); // ensure we don't leak `key` to frontend
|
|
|
|
const credential: typeof credentials[number] | null = credentials[0] || null;
|
|
return {
|
|
...integration,
|
|
/**
|
|
* @deprecated use `credentials`
|
|
*/
|
|
credential,
|
|
credentials,
|
|
};
|
|
});
|
|
|
|
return integrations;
|
|
}
|
|
|
|
export type IntegrationMeta = ReturnType<typeof getIntegrations>;
|
|
|
|
export function hasIntegration(integrations: IntegrationMeta, type: string): boolean {
|
|
return !!integrations.find(
|
|
(i) => i.type === type && !!i.installed && (type === "daily_video" || i.credentials.length > 0)
|
|
);
|
|
}
|
|
|
|
export default getIntegrations;
|