saleor-apps-redis_apl/apps/taxes/scripts/migrations/app-webhook-migrator.ts
Adrian Pilarczyk 416c92fb6c
feat: change the flow from OrderCreated to OrderConfirmed (#826)
* refactor: ♻️ extract order-metadata-manager

* feat: 🚧 add basic boilerplate

* feat:  add readExternalIdFromOrderMetadata

* Revert "feat:  add readExternalIdFromOrderMetadata"

This reverts commit a78d9d4597672f8605cf998a9f784aebaab27de1.

* feat:  add order-cancelled avatax adapter

* test:  add tests for AvataxOrderCancelledPayloadTransformer

* refactor: avataxId instead of externalId

* refactor: ♻️ split up webhook response

* build: ⬆️ upgrade avatax

* refactor: ♻️ extend logging in webhook response errors

* fix: 🐛 split privateMetadata and publicMetadata

* fix: 🐛 use "DEFAULT" value of companyCode for commit to work

* fix: ⚗️ fix voidTransaction type

* refactor: 🚚 order_created -> order_confirmed

* fix: 🐛 change voidReason

* build: 👷 add changeset

* refactor: 🔥 order_fulfilled webhook

* feat: Avatax metadata tax calculation date (#843)

* feat:  add metadata tax calculation date

* build: 👷 add changeset

* feat: Avatax metadata document code (#844)

* feat:  provide document code through metadata field

* build: 👷 add changeset

* refactor: ♻️ fallback to default company code for migration

* refactor: ♻️ patch order-created files and add deprecation note

* Revert "refactor: 🔥 order_fulfilled webhook"

This reverts commit fd098642735ae9d62e3a876088226bd0f108afd6.

* refactor: ♻️ patch order-fulfilled files and add deprecation note

* fix: 🐛 bring back deprecated webhooks to manifest

* feat: ⚗️ add AppWebhookMigrator (#850)

* refactor: 🚚 order_created -> order_confirmed

* refactor: 🔥 order_fulfilled webhook

* feat: ⚗️ add AppWebhookMigrator

* feat:  add mode to migrator

* feat:  add draft of run-report and migrateWebhook method

* refactor: ♻️ address feedback

* feat:  add tests and new structure

* refactor: 🔥 util

* feat:  add enable/disable webhook rollback flow

* refactor: ♻️ modify the taxes-migration flow

* refactor: ♻️ generalize document code & date resolver

* chore: 🗃️ add run-migration

* chore: 💡 update comments about migration flow

* fix: 🐛 slice document code

* refactor: ♻️ try/catch at the top level

* chore: 💡 add comments

* Update shiny-meals-wait.md

* Update soft-steaks-know.md

* Update soft-steaks-know.md

* fix:  fix test

* feat:  change createTransaction to createOrAdjustTransaction

this feature grants idempotency of the transaction flow

* feat:  add number field to OrderConfirmed payload

* chore: 💡 add deprecation comment to metadata method

* docs: 📝 add todo comment to refactor sumPayloadLines

* feat:  add resolveStringOrThrow and use it for email

* fix: 🐛 add missing number to mock
2023-08-10 13:08:20 +02:00

186 lines
5.4 KiB
TypeScript

import { SaleorAsyncWebhook, SaleorSyncWebhook } from "@saleor/app-sdk/handlers/next";
import { WebhookEventTypeAsyncEnum, WebhookEventTypeSyncEnum } from "../../generated/graphql";
import { AppWebhookRepository } from "./app-webhook-repository";
import { AuthData } from "@saleor/app-sdk/APL";
import { createGraphQLClient } from "@saleor/apps-shared";
type AppWebhookMigratorOptions = {
mode: "report" | "migrate";
};
export class AppWebhookMigrator {
private appWebhookRepository: AppWebhookRepository;
private appId: string;
private apiUrl: string;
private mode: AppWebhookMigratorOptions["mode"];
constructor(
{
appWebhookRepository,
apiUrl,
appId,
}: {
apiUrl: string;
appId: string;
appWebhookRepository: AppWebhookRepository;
},
{ mode }: AppWebhookMigratorOptions
) {
this.appWebhookRepository = appWebhookRepository;
this.appId = appId;
this.apiUrl = apiUrl;
this.mode = mode;
}
private registerWebhookFromHandler(webhookHandler: SaleorSyncWebhook | SaleorAsyncWebhook) {
const manifest = webhookHandler.getWebhookManifest(this.apiUrl);
if (!manifest.query) {
throw new Error("Webhook query is required");
}
if (!manifest.name) {
throw new Error("Webhook name is required");
}
return this.appWebhookRepository.create({
appId: this.appId,
name: manifest.name,
query: manifest.query,
targetUrl: manifest.targetUrl,
asyncEvents: (manifest.asyncEvents ?? []) as WebhookEventTypeAsyncEnum[],
syncEvents: (manifest.syncEvents ?? []) as WebhookEventTypeSyncEnum[],
isActive: manifest.isActive ?? true,
});
}
private async deleteWebhookById(webhookId: string) {
console.log(`Webhook ${webhookId} will be deleted`);
if (this.mode === "migrate") {
await this.appWebhookRepository.delete(webhookId);
console.log(`Webhook ${webhookId} deleted`);
}
}
private async disableWebhookById(webhookId: string) {
console.log(`Webhook ${webhookId} will be disabled`);
if (this.mode === "migrate") {
await this.appWebhookRepository.disable(webhookId);
console.log(`Webhook ${webhookId} disabled`);
}
}
/**
* @returns all webhooks for the app
* @throws error if fetching webhooks fails
*/
async getAppWebhooks() {
const webhooks = await this.appWebhookRepository.getAll();
console.log(`Webhooks for app ${this.appId}: `, webhooks);
return webhooks;
}
private async disableFirstWebhookByName(webhookName: string) {
const webhooks = await this.getAppWebhooks();
const webhook = webhooks.find((webhook) => webhook.name === webhookName);
if (!webhook) {
console.log(`Webhook ${webhookName} not found`);
return;
}
await this.disableWebhookById(webhook.id);
}
/**
* Deletes first app webhook that matches the name.
* @param webhookName - name of the webhook to delete
*/
async DANGEROUS_DELETE_APP_WEBHOOK_BY_NAME(webhookName: string) {
const webhooks = await this.getAppWebhooks();
const webhook = webhooks.find((webhook) => webhook.name === webhookName);
if (!webhook) {
console.log(`Webhook ${webhookName} not found`);
return;
}
await this.deleteWebhookById(webhook.id);
}
/**
* Registers a webhook if it doesn't exist based on a handler.
* @param webhookHandler - The handler of the webhook we want to register.
* @example registerWebhookIfItDoesntExist(orderConfirmedAsyncWebhook)
*/
async registerWebhookIfItDoesntExist(webhookHandler: SaleorSyncWebhook | SaleorAsyncWebhook) {
const webhooks = await this.getAppWebhooks();
const webhookExists = webhooks.some((webhook) => webhook.name === webhookHandler.name);
if (webhookExists) {
console.log(`Webhook ${webhookHandler.name} already exists`);
return;
}
console.log(`Webhook ${webhookHandler.name} will be registered`);
if (this.mode === "migrate") {
await this.registerWebhookFromHandler(webhookHandler);
console.log(`Webhook ${webhookHandler.name} registered`);
}
}
/**
* Rolls back webhook migration by deleting the new webhook and enabling the old one.
* @param prevWebhookName - The name of the webhook we wanted to migrate from.
* @param nextWebhookHandler - The handler of the webhook we wanted to migrate to.
* @example rollbackWebhookMigrations("OrderCreated", orderConfirmedAsyncWebhook)
*/
async rollbackWebhookMigrations(
prevWebhookName: string,
nextWebhookHandler: SaleorSyncWebhook | SaleorAsyncWebhook
) {
const webhooks = await this.appWebhookRepository.getAll();
const webhooksToRemove = webhooks.filter((webhook) => webhook.name === nextWebhookHandler.name);
const webhooksToEnable = webhooks.filter((webhook) => webhook.name === prevWebhookName);
for (const webhook of webhooksToRemove) {
await this.deleteWebhookById(webhook.id);
}
for (const webhook of webhooksToEnable) {
await this.appWebhookRepository.enable(webhook.id);
}
}
}
export function createAppWebhookMigrator(env: AuthData, options: AppWebhookMigratorOptions) {
const client = createGraphQLClient({
saleorApiUrl: env.saleorApiUrl,
token: env.token,
});
const appWebhookRepository = new AppWebhookRepository(client);
return new AppWebhookMigrator(
{
apiUrl: env.saleorApiUrl,
appId: env.appId,
appWebhookRepository,
},
options
);
}