
* 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
186 lines
5.4 KiB
TypeScript
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
|
|
);
|
|
}
|