UNPKG

@gofynd/fdk-extension-javascript

Version:
257 lines (211 loc) 9.63 kB
# fdk-extension-javascript FDK Extension Helper Library #### Initial Setup ```javascript const bodyParser = require("body-parser"); const express = require("express"); const cookieParser = require("cookie-parser"); const { setupFdk } = require("fdk-extension-javascript/express"); const { RedisStorage } = require("fdk-extension-javascript/express/storage"); const Redis = require("ioredis"); const app = express(); app.use(cookieParser("ext.session")); app.use(bodyParser.json({ limit: "2mb" })); const redis = new Redis(); let extensionHandler = { auth: async function (data) { console.log("called auth callback"); }, uninstall: async function (data) { console.log("called uninstall callback"); }, }; let fdkClient = setupFdk({ api_key: "<API_KEY>", api_secret: "<API_SECRET>", base_url: baseUrl, // this is optional scopes: ["company/products"], // this is optional callbacks: extensionHandler, storage: new RedisStorage(redis), access_mode: "offline", }); app.use(fdkClient.fdkHandler); app.listen(8080); ``` #### How to call platform apis? To call platform api you need to have instance of `PlatformClient`. Instance holds methods for SDK classes. All routes registered under `platformApiRoutes` express router will have `platformClient` under request object which is instance of `PlatformClient`. > Here `platformApiRoutes` has middleware attached which allows passing such request which are called after launching extension under any company. ```javascript fdkClient.platformApiRoutes.get("/test/routes", async (req, res, next) => { try { let data = await req.platformClient.lead.getTickets(); res.json(data); } catch (error) { console.error(error); next(error); } }); app.use(fdkClient.platformApiRoutes); ``` #### How to call platform apis in background tasks? Background tasks running under some consumer or webhook or under any queue can get platform client via method `getPlatformClient`. It will return instance of `PlatformClient` as well. > Here FdkClient `access_mode` should be **offline**. Cause such client can only access PlatformClient in background task. ```javascript function backgroundHandler(companyId) { try { const platformClient = await fdkExtension.getPlatformClient(companyId); let data = await platformClient.lead.getTickets(); // Some business logic here res.json({ success: true }); } catch (err) { console.error(err); res.status(404).json({ success: false }); } } ``` #### How to call partner apis in background tasks? Background tasks running under some consumer or webhook or under any queue can get partner client via method `getPartnerClient`. It will return instance of `PartnerClient` as well. > Here FdkClient `access_mode` should be **offline**. Cause such client can only access PartnerClient in background task. ```javascript function backgroundHandler(organizationId) { try { const partnerClient = await fdkExtension.getPartnerClient(organizationId); let data = await partnerClient.webhook.responseTimeSummary({ extensionId: '<EXTENSION_ID>', startDate: '<START_DATE>', endDate: '<END_DATE>' }); // Some business logic here res.json({ success: true }); } catch (err) { console.error(err); res.status(404).json({ success: false }); } } ``` #### How to call partner apis? To call partner api you need to have instance of `PartnerClient`. Instance holds methods for SDK classes. All routes registered under `partnerApiRoutes` express router will have `partnerClient` under request object which is instance of `PartnerClient`. > Here `partnerApiRoutes` has middleware attached which allows passing such request which are called after launching extension under any company. ```javascript fdkClient.partnerApiRoutes.get("/test/routes", async (req, res, next) => { try { let data = await req.partnerClient.lead.getTickets(); res.json(data); } catch (error) { console.error(error); next(error); } }); app.use(fdkClient.partnerApiRoutes); ``` #### How to register for webhook events? Webhook events can be helpful to handle tasks when certan events occur on platform. You can subscribe to such events by passing `webhook_config` in setupFdk function. ```javascript let fdkClient = setupFdk({ api_key: "<API_KEY>", api_secret: "<API_SECRET>", base_url: baseUrl, // this is optional scopes: ["company/products"], // this is optional callbacks: { auth: async function (data) { console.log("called auth callback"); }, uninstall: async function (data) { console.log("called uninstall callback"); }, }, storage: new RedisStorage(redis), access_mode: "offline", webhook_config: { api_path: "/api/v1/webhooks", // required notification_email: "test@abc.com", // required subscribe_on_install: false, //optional. Default true subscribed_saleschannel: 'specific', //optional. Default all marketplace: true, // to receive marketplace saleschannel events. Only allowed when subscribed_saleschannel is set to specific event_map: { // required 'company/brand/create': { version: '1', handler: handleBrandCreate, provider: 'rest' // if not provided, Default is `rest` }, 'company/location/update': { version: '1', handler: handleLocationUpdate, }, 'application/coupon/create': { version: '1', topic: 'coupon_create_kafka_topic', provider: 'kafka' }, 'company/brand/update': { version: '1', topic: "company-brand-create", provider: 'pub_sub' }, 'extension/extension/install': { version: '1', queue: "extension-install", workflow_name: "extension", provider: 'temporal' }, 'company/location/create': { version: '1', queue: "company-location-create", provider: 'sqs' }, 'company/product-size/create': { version: '1', event_bridge_name: "company-product-size-create", provider: 'event_bridge' } } }, debug: true // optional. Enables debug logs if `true`. Default `false` }); ``` > By default all webhook events all subscribed for all companies whenever they are installed. To disable this behavior set `subscribe_on_install` to `false`. If `subscribe_on_install` is set to false, you need to manually enable webhook event subscription by calling `syncEvents` method of `webhookRegistry` There should be view on given api path to receive webhook call. It should be `POST` api path. Api view should call `processWebhook` method of `webhookRegistry` object available under `fdkClient` here. > Here `processWebhook` will do payload validation with signature and calls individual handlers for event passed with webhook config. ```javascript app.post('/api/v1/webhooks', async (req, res, next) => { try { await fdkClient.webhookRegistry.processWebhook(req); return res.status(200).json({"success": true}); } catch(err) { logger.error(err); return res.status(500).json({"success": false}); } }); ``` > Setting `subscribed_saleschannel` as "specific" means, you will have to manually subscribe saleschannel level event for individual saleschannel. Default value here is "all" and event will be subscribed for all sales channels. For enabling events manually use function `enableSalesChannelWebhook`. To disable receiving events for a saleschannel use function `disableSalesChannelWebhook`. #### Filters and reducers in webhook events A filter and reducer can be provided to refine the data delivered for each subscribed event. The Filter functionality allows selective delivery of data by specifying conditions based on JSONPath queries and logical operators. Reducer allow customization of the payload structure by specifying only the fields needed by the subscriber. The reducer extracts fields from the event’s data and restructures them as needed. ```javascript webhook_config: { api_path: "/v1.0/webhooks", notification_email: "rahultambe@gofynd.com", marketplace: true, subscribed_saleschannel: 'specific', event_map: { 'company/brand/update': { version: '1', handler: handleExtensionUninstall, filters: { query: "$.brand.uid", condition: "(uid) => uid === 238" }, reducer: { brand_name: "$.brand.name", logo_link: "$.brand.logo" } }] } } ``` ##### How webhook registery subscribes to webhooks on Fynd Platform? After webhook config is passed to setupFdk whenever extension is launched to any of companies where extension is installed or to be installed, webhook config data is used to create webhook subscriber on Fynd Platform for that company. > Any update to webhook config will not automatically update subscriber data on Fynd Platform for a company until extension is opened atleast once after the update. Other way to update webhook config manually for a company is to call `syncEvents` function of webhookRegistery. # [Custom storage class](/express/storage/README.md) The FDK Extension JavaScript library provides built-in support for SQLite, Redis and in-memory storage options as default choices for session data storage. However, if you require a different storage option, this readme will guide you through the process of implementing a custom storage class.