FDK Extension Helper Library
-
Create Maven project and add the dependency in the pom.xml
<dependency> <groupId>com.github.gofynd</groupId> <artifactId>fdk-extension-java</artifactId> <version><LATEST_VERSION></version> </dependency>
-
Add the Jitpack Repo to your root pom.xml:
<repositories> <repository> <id>jitpack.io</id> <url>https://jitpack.io</url> </repository> </repositories>
-
Add Extension specific Configuration properties in application.yml file
redis : host : 'redis://127.0.0.1:6379' ext : api_key : <API_KEY> api_secret : <API_SECRET> scope : 'company/saleschannel' base_url : 'https://test.extension.com' access_mode : 'offline' fdk-extension: version: '0.5.0'
-
Create Main Application class and Initialise the Extension using the properties.
@SpringBootApplication
@ComponentScan(basePackages = {"com.fynd.**","com.sdk.**"})
public class EmailExtensionApplication {
private String CACHE_PREFIX_KEY = "inv_email";
@Autowired
@Qualifier("jedispoolbean")
JedisPool jedis;
@Autowired
ExtensionProperties extensionProperties;
ExtensionCallback callbacks = new ExtensionCallback((request) -> {
Session fdkSession = (Session) request.getAttribute("session");
System.out.println("In Auth callback");
if(request.getParameter("application_id") != null){
return extensionProperties.getBaseUrl() + "/company/" + fdkSession.getCompanyId() + "/application/" + request.getParameter("application_id");
}
else {
return extensionProperties.getBaseUrl() + "/company/" + fdkSession.getCompanyId();
}
}, (context) -> {
System.out.println("In install callback");
return extensionProperties.getBaseUrl();
}, (fdkSession) -> {
System.out.println("In uninstall callback");
return extensionProperties.getBaseUrl();
}, (fdkSession) -> {
System.out.println("In auto-install callback");
return extensionProperties.getBaseUrl();
});
public static void main(String[] args) {
SpringApplication.run(SampleApplication.class, args);
}
@Bean
public com.fynd.extension.model.Extension getExtension() {
Extension extension = new Extension();
return extension.initialize(extensionProperties,
new RedisStorage(jedis, CACHE_PREFIX_KEY), //BaseStorage is the parent class, any child class can be used here - REDIS / Memory
callbacks);
}
}
- Define Redis Service which is used to save the intermediate Sessions for each CompanyID
@Service
public class RedisService {
@Value("${redis.host}")
private String redisHost;
JedisPool jedisPool;
@Bean(name = "jedispoolbean")
JedisPool getJedis() throws URISyntaxException {
JedisPoolConfig poolConfig = new JedisPoolConfig();
poolConfig.setMaxTotal(1100);
poolConfig.setMaxIdle(16);
poolConfig.setMinIdle(16);
poolConfig.setTestOnBorrow(true);
poolConfig.setTestOnReturn(true);
poolConfig.setTestWhileIdle(true);
poolConfig.setJmxEnabled(false);
poolConfig.setMinEvictableIdleTimeMillis(Duration.ofSeconds(60)
.toMillis());
poolConfig.setTimeBetweenEvictionRunsMillis(Duration.ofSeconds(30)
.toMillis());
poolConfig.setNumTestsPerEvictionRun(3);
poolConfig.setBlockWhenExhausted(true);
jedisPool = new JedisPool(poolConfig, redisHost); //Use SSL if necessary in envs
return jedisPool;
}
@PreDestroy
public void destroy() {
System.out.println("Closing the jedis pool connection");
if (jedisPool != null)
jedisPool.close();
jedisPool = null;
}
}
To call platform api you need to have instance of PlatformClient
. Instance holds methods for SDK classes.
extend BasePlatformController
class to create controller which will add PlatformClient
in request.
@RestController
@RequestMapping("/api/v1")
@Slf4j
public class PlatformController extends BasePlatformController {
@GetMapping(value = "/applications", produces = "application/json")
public ConfigurationPlatformModels.ApplicationsResponse getApplications(HttpServletRequest request) {
try {
PlatformClient platformClient = (PlatformClient) request.getAttribute("platformClient");
ConfigurationPlatformModels.ApplicationsResponse applications
= platformClient.configuration.getApplications(1, 100, "");
return applications;
} catch (Exception e) {
System.out.println(e.getMessage());
throw new RuntimeException(e);
}
}
}
To obtain the PlatformClient
for offline access mode in your Java extension, use the provided ExtensionService
class.
This example demonstrates how to retrieve applications using the obtained PlatformClient
@RestController
@RequestMapping("/api/v1")
public class ExampleOfflineAccessMode {
@Autowired
ExtensionService extensionService;
@GetMapping(value = "/applications", produces = "application/json")
public ConfigurationPlatformModels.ApplicationsResponse getApplications() {
try {
String companyId = "1";
PlatformClient platformClient = extensionService.getPlatformClient(companyId);
return platformClient.configuration.getApplications(1, 100, "");
} catch (Exception e) {
System.out.println(e.getMessage());
throw new RuntimeException(e);
}
}
}
To obtain the PartnerClient
for offline access mode in your Java extension, use the provided ExtensionService
class.
This example demonstrates how to retrieve response time using the obtained PartnerClient
@RestController
@RequestMapping("/api/v1")
public class ExampleOfflineAccessMode {
@Autowired
ExtensionService extensionService;
@GetMapping(value = "/response-time", produces = "application/json")
public WebhookPartnerModels.ResponseTimeTs getResponseTime() {
try {
String organizationId = "64b0e718f56860951c34d735";
PartnerClient partnerClient = extensionService.getPartnerClient("64b0e718f56860951c34d735");
WebhookPartnerModels.ResponseTimeTs responseTime = partnerClient.webhook.responseTimeSummary(
"66c5b90b2d58ee212f3891f8",
"2024-11-18T11:38:44+05:30",
"2024-11-18T14:38:44+05:30"
);
return responseTime;
} catch (Exception e) {
System.out.println(e.getMessage());
throw new RuntimeException(e);
}
}
}
To call partner api you need to have instance of PartnerClient
. Instance holds methods for SDK classes.
extend BasePartnerController
class to create controller which will add PartnerClient
in request.
@RestController
@RequestMapping("/api/v1")
@Slf4j
public class PartnerController extends BasePartnerController {
@GetMapping(value = "/orgThemes", produces = "application/json")
public ThemePartnerModels.MarketplaceThemeSchema getOrgThemes(HttpServletRequest request) {
try {
PartnerClient partnerClient = (PartnerClient) request.getAttribute("partnerClient");
ThemePartnerModels.MarketplaceThemeSchema orgThemes = partnerClient.theme.getOrganizationThemes("published", null, null);
return orgThemes;
} catch (Exception e) {
System.out.println(e.getMessage());
throw new RuntimeException(e);
}
}
}
Webhook events can be helpful to handle tasks when certain events occur on platform. You can subscribe to such events by passing webhook in Extension Configuration Property
- Add the Configuration property in application yaml File
ext :
api_key : <API_KEY>
api_secret : <API_SECRET>
scope : ""
base_url : "https://test.extension.com"
access_mode : "offline"
webhook:
api_path: "/webhook" #<POST API URL>
notification_email: <EMAIL_ID>
subscribe_on_install: false, #optional. Default true
subscribed_saleschannel: 'all' #Can be 'SPECIFIC'/'EMPTY'
event_map:
- name: 'product/create'
handler: productCreateHandler #Make sure this matches the Component Bean name
category: 'company'
version: '1'
provider: 'rest' # If not provided, Default is 'rest'
-
name: 'product/delete'
handler: productDeleteHandler
category: 'company'
version: '1'
provider: 'kafka'
- Create Handlers for each event which is mentioned in the Event Map (as specified above)
@Component("productCreateHandler")
public class ProductCreateHandler implements com.fynd.extension.middleware.EventHandler {
@Override
public void handle(String eventName, Object body, String companyId, String applicationId) {
// Code to handle webhook event
}
}
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 callingsyncEvents
method ofwebhookRegistry
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.
@CrossOrigin(origins = "*")
@RestController
@RequestMapping()
@Slf4j
public class WebhookController {
@Autowired
WebhookService webhookService;
@PostMapping(path = "/webhook")
public Map<String, Boolean> receiveWebhookEvents(HttpServletRequest httpServletRequest) {
try {
webhookService.processWebhook(httpServletRequest);
return Collections.singletonMap("success", true);
} catch (Exception e) {
log.error("Exception occurred", e);
return Collections.singletonMap("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
.
After webhook config is passed to initialize 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 webhookRegistry.
Custom storage classes expand data storage options beyond default choices like Redis and in-memory storage. You would required to create a custom storage class by extending the base storage class provided by fdk extension java library and implementing required methods as per your chosen database.
import com.fynd.extension.storage.BaseStorage;
public class MyCustomStorage extends BaseStorage {
private StorgeClient storgeClient;
// StorgeClient is connection variable to your storage.
public MyCustomStorage(StorgeClient storgeClient, String prefixKey) {
super(prefixKey);
this.storgeClient = storgeClient;
}
@Override
public String get(String key) {
try (StorgeClient storgeClient = storgeClient.getResource()) {
return storgeClient.get(super.prefixKey + key);
}
}
// All of the below methods need to be implemented as per your chosen databse
public String set(String key, String value) {
// Implementation of a set method
}
public Object del(String key) {
// Implementation of a del method
}
public String setex(String key, int ttl, String value) {
// Implementation of a setex method
}
public String hget(String key, String hashKey) {
// Implementation of a hget method
}
public Object hset(String key, String hashKey, String value) {
// Implementation of a hset method
}
public Map<String, Object> hgetall(String key) {
// Implementation of a hgetall method
}
}