* 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>
		
			
				
	
	
		
			76 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			76 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
import { Webhook } from "@prisma/client";
 | 
						|
import { compile } from "handlebars";
 | 
						|
 | 
						|
import type { CalendarEvent } from "@calcom/types/Calendar";
 | 
						|
 | 
						|
type ContentType = "application/json" | "application/x-www-form-urlencoded";
 | 
						|
 | 
						|
function applyTemplate(template: string, data: CalendarEvent, contentType: ContentType) {
 | 
						|
  const compiled = compile(template)(data);
 | 
						|
  if (contentType === "application/json") {
 | 
						|
    return JSON.stringify(jsonParse(compiled));
 | 
						|
  }
 | 
						|
  return compiled;
 | 
						|
}
 | 
						|
 | 
						|
function jsonParse(jsonString: string) {
 | 
						|
  try {
 | 
						|
    return JSON.parse(jsonString);
 | 
						|
  } catch (e) {
 | 
						|
    // don't do anything.
 | 
						|
  }
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
const sendPayload = async (
 | 
						|
  triggerEvent: string,
 | 
						|
  createdAt: string,
 | 
						|
  webhook: Pick<Webhook, "subscriberUrl" | "appId" | "payloadTemplate">,
 | 
						|
  data: CalendarEvent & {
 | 
						|
    metadata?: { [key: string]: string };
 | 
						|
    rescheduleUid?: string;
 | 
						|
  }
 | 
						|
) => {
 | 
						|
  const { subscriberUrl, appId, payloadTemplate: template } = webhook;
 | 
						|
  if (!subscriberUrl || !data) {
 | 
						|
    throw new Error("Missing required elements to send webhook payload.");
 | 
						|
  }
 | 
						|
 | 
						|
  const contentType =
 | 
						|
    !template || jsonParse(template) ? "application/json" : "application/x-www-form-urlencoded";
 | 
						|
 | 
						|
  data.description = data.description || data.additionalNotes;
 | 
						|
 | 
						|
  let body;
 | 
						|
 | 
						|
  /* Zapier id is hardcoded in the DB, we send the raw data for this case  */
 | 
						|
  if (appId === "zapier") {
 | 
						|
    body = JSON.stringify(data);
 | 
						|
  } else if (template) {
 | 
						|
    body = applyTemplate(template, data, contentType);
 | 
						|
  } else {
 | 
						|
    body = JSON.stringify({
 | 
						|
      triggerEvent: triggerEvent,
 | 
						|
      createdAt: createdAt,
 | 
						|
      payload: data,
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  const response = await fetch(subscriberUrl, {
 | 
						|
    method: "POST",
 | 
						|
    headers: {
 | 
						|
      "Content-Type": contentType,
 | 
						|
    },
 | 
						|
    body,
 | 
						|
  });
 | 
						|
 | 
						|
  const text = await response.text();
 | 
						|
 | 
						|
  return {
 | 
						|
    ok: response.ok,
 | 
						|
    status: response.status,
 | 
						|
    message: text,
 | 
						|
  };
 | 
						|
};
 | 
						|
 | 
						|
export default sendPayload;
 |