3.7+
FDK Extension Helper Python Library
22.9.0+ Sanic
Initial Setup
import os
import sys
import aioredis
from sanic import Sanic, Blueprint
from sanic import response
sys.path.insert(0, os.path.abspath(os.path.join(os.path.dirname(__file__), '..')))
from fdk_extension import setup_fdk
from fdk_extension.storage.redis_storage import RedisStorage
from examples.extension_handlers import extension_handler
app = Sanic("test")
redis_client = aioredis.from_url("redis://localhost")
base_url = "http://0.0.0.0:8000"
fdk_extension_client = setup_fdk({
"api_key": "<API_KEY>",
"api_secret": "<API_SECRET>",
"base_url": base_url,
"callbacks": extension_handler,
"storage": RedisStorage(redis_client, prefix_key="extension-example"), # add your redis key prefix here
"access_mode": "offline",
"cluster": "https://api.fyndx0.de", # this is optional by default it points to prod.
})
app.blueprint(fdk_extension_client.fdk_route)
platform_bp = Blueprint("platform blueprint")
platform_bp.add_route(test_route_handler, "/test/routes")
application_bp = Blueprint("application blueprint")
application_bp.add_route(test_route_handler, "/1234")
app.add_route(webhook_handler, "/webhook")
fdk_extension_client.platform_api_routes.append(platform_bp)
fdk_extension_client.application_proxy_routes.append(application_bp)
app.blueprint(fdk_extension_client.platform_api_routes)
app.blueprint(fdk_extension_client.application_proxy_routes)
# debug logs enabled with debug = True
if __name__ == "__main__":
app.run(host="0.0.0.0", port=8000, debug=True)
To call platform api you need to have instance of PlatformClient
. Instance holds methods for SDK classes. All routes registered under platform_api_routes
blueprint will have platform_client
under request context object which is instance of PlatformClient
.
Here
platform_api_routes
has middleware attached which allows passing such request which are called after launching extension under any company.
async def test_route_handler(request):
try:
platform_client = request.conn_info.ctx.platform_client
data = await platform_client.lead.getTickets()
return response.json({"data": data["json"]})
except Exception as e:
return response.json({"error_message": str(e)}, 500)
fdk_extension_client.platform_api_routes.add_route(test_route_handler, "/test/routes")
app.blueprint(fdk_extension_client.platform_api_routes)
Background tasks running under some consumer or webhook or under any queue can get platform client via method get_platform_client
. 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.
async def background_handler(request):
try:
company_id = request.args.get("companyId")
client = await fdk_extension_client.get_platform_client(company_id)
return response.json({"success": True})
except Exception as e:
return response.json({"error_message": str(e)}, 500)
Background tasks running under some consumer or webhook or under any queue can get partner client via method get_partner_client
. It will return instance of PartnerClient
as well.
Here FdkClient
access_mode
should be offline. Cause such client can only access PlatformClient in background task.
async def background_handler(request):
try:
organization_id = request.args.get("organizationId")
client = await fdk_extension_client.get_partner_client(organization_id)
return response.json({"success": True})
except Exception as e:
return response.json({"error_message": str(e)}, 500)
To call partner api you need to have instance of PartnerClient
. Instance holds methods for SDK classes. All routes registered under partner_api_routes
blueprint will have partner_client
under request context object which is instance of PartnerClient
.
Here
partner_api_routes
has middleware attached which allows passing such request which are called after launching extension under any company.
async def test_route_handler(request):
try:
partner_client = request.conn_info.ctx.partner_client
data = await partner_client.lead.getTickets()
return response.json({"data": data["json"]})
except Exception as e:
return response.json({"error_message": str(e)}, 500)
fdk_extension_client.partner_api_routes.add_route(test_route_handler, "/test/routes")
app.blueprint(fdk_extension_client.partner_api_routes)
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.
fdk_extension_client = setup_fdk({
"api_key": "<API_KEY>",
"api_secret": "<API_SECRET>",
"base_url": base_url,
"callbacks": extension_handler,
"storage": RedisStorage(redis_client, prefix_key="extension-example"), # add your redis key prefix here
"access_mode": "offline",
"cluster": "https://api.fyndx0.de", # this is optional by default it points to prod.
"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
"event_map": { # required
'company/brand/create': {
"version": '1',
"handler": handleBrandCreate
},
'company/location/update': {
"version": '1',
"handler": handleLocationUpdate
},
'application/coupon/create': {
"version": '1',
"handler": handleCouponCreate
}
}
}
})
By default all webhook events all subscribed for all companies whenever they are installed. To disable this behavior set
subscribe_on_install
tofalse
. Ifsubscribe_on_install
is set to false, you need to manually enable webhook event subscription by callingsync_events
method ofwebhookRegistry
There should be view on given api path to receive webhook call. It should be POST
api path. Api view should call process_webhook
method of webhookRegistry
object available under fdk_client
here.
Here
process_webhook
will do payload validation with signature and calls individual handlers for event passed with webhook config.
async def webhook_handler(request):
try:
await fdk_extension_client.webhook_registry.process_webhook(request)
return response.json({"success": True})
except Exception as e:
logger.exception(e)
return response.json({"error_message": str(e), "success": False}, 500)
app.add_route(webhook_handler, "/webhook", methods=["POST"])
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 functionenable_sales_channel_webhook
. To disable receiving events for a saleschannel use functiondisable_sales_channel_webhook
.
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 sync_events
function of webhookRegistery.