Notifi Frontend Client
The Notifi Frontend Client SDK is a JavaScript (Typescript) library that allows you to integrate the Notifi Network into your decentralized application (dApp). Dapp will be able to access Notifi services using client methods.
@notifi/notifi-frontend-client package is not framework specific, it means you can use it with any JavaScript framework like React, Angular, Vue, svelte, etc.
Note: if you are using React, Notifi has a React SDK that is built on top of this SDK, @notifi-network/notifi-react
📥 Installation
npm i @notifi-network/notifi-frontend-client
🪝 Initialization
Instantiate the client object at your dapp to interact with the Notifi core services by providing your notifi Notifi tenant credentials and the user's wallet information.
const tenantId = '<notifi-tenent-id>';
const env = 'Production';
const walletBlockchain = '<the-blockchain-of-your-choice>';
const client = instantiateFrontendClient({
walletBlockchain,
walletPublicKey: '<user-wallet-public-key>',
tenantId,
env,
});
const newUserState = await client.initialize();
Reference:
- You can register a new tenant to get tenantId (dappAddress) at Notifi Admin Panel for free.
- The interface of input argument differs depending on the blockchain, see the type doc here
🔏 Authorization
There are two authentication categories to onboard the users with Notifi:
On-chain method: Arbitrary signature authorization
Authorize users using logIn client method by providing a signMessage callback function.
const signMessage: AptosSignMessageFunction = async (message, nonce) => {
return signature;
};
const loginResult = await client.logIn({
walletBlockchain: `the-blockchain-of-your-choice`,
signMessage,
});
NOTE:
On-chain method: Transaction signature authorization
Authorize users by using beginLoginViaTransaction method and completeLoginViaTransaction client method. The beginLoginViaTransaction method will return a nonce that should be appended to the transaction data before signing the transaction. The completeLoginViaTransaction method will take the transaction signature and the nonce to complete the login process. The following examples show how to authorize a user using the transaction signature method on EVM (Ethereum virtual machine) compatible blockchains.
const smartContract = new ethers.Contract(
`<contract-address>`,
`<contract-abi>`,
provider.getSigner(),
);
const amount = ethers.utils.parseUnits('0.1', 'ether');
const calldata = smartContract.interface.encodeFunctionData('example', [
amount,
]);
const { nonce } = await notifiClient.beginLoginViaTransaction({
walletBlockchain,
walletAddress,
});
console.info({ notifiNonce: nonce });
const txParams = {
to: smartContract.address,
data: calldata + nonce.replace('0x', ''),
};
const { hash } = await provider.getSigner().sendTransaction(txParams);
const notifiLoginResult = await notifiClient.completeLoginViaTransaction({
walletAddress,
walletAddress,
transactionSignature: hash,
walletBlockchain,
});
IMPORTANT NOTE:
- Browser extension constraints: Some browser extensions (Metamask ...) do not allow appending additional info to the calldata of particular smart contract method (example: Token contract (ERC20, ERC721 ...etc)). Browser extension (Metamask) will throw
Message: Invalid token value; should be exactly xx hex digits long error in this case.
Off-chain method: OpenID Connect (OIDC) Authorization
Authorize users by using logIn client method by providing a signIn callback function.
const signIn: OidcSignInFunction = async () => {
const jwt = '<the-oidc-id-token-here>';
return {
oidcProvider: 'GOOGLE',
jwt,
};
};
const loginResult = await client.logIn({
walletBlockchain: 'OFF_CHAIN',
signIn,
});
NOTE
- Respective OIDC
login arguments
- Notifi supported OIDC providers: OidcProvider Enum document
- To enable OIDC login, it requires additional setup to integrate your OIDC provider with Notifi tenant using Notifi Admin Portal check on the Notifi Documentation (WIP: coming soon)
Fetch User Data
The fetchFusionData client method is used to fetch the user's data, including their:
targetGroup: contact informations
alerts: subscribed alerts
NOTE:
📭 Destinations (Target & TargetGroup)
When a user logs in for the first time, a default target group named Default is automatically created for them. Users can manage their target groups using the alterTargetGroup client method.
Supported Destination Types
Notifi currently supports the following destination types:
- Email
- SMS
- Telegram
- Discord
- More to come (additional types will be added in the future)
Target Group Management
The alterTargetGroup method allows users to modify their target groups. It supports three operations:
- Ensure (
{ type: 'ensure'; name: string }):
- Adds a target to the target group if it doesn't already exist.
- If the target doesn't exist, it creates a new target using the provided
name.
- Remove (
{ type: 'remove' }):
- Removes the target from the target group.
- The target remains in Notifi's backend and can be re-added later without re-verification.
- Delete (
{ type: 'delete'; id: string }):
- Removes the target from the target group and deletes it from Notifi's backend.
- The target will need to be re-verified if added again in the future.
Example Usage
Below is an example of how to use the alterTargetGroup method:
const targetGroup = await client.alterTargetGroup({
name: 'Default',
email: {
type: 'ensure',
name: '<user-email>',
},
discord: {
type: 'remove',
},
telegram: {
type: 'delete',
id: '<telegram-id>',
},
});
Key Notes
- Reference Documentation: For detailed type documentation, refer to this link (WIP).
- Remove vs Delete:
- Remove: Targets are only removed from the target group but remain in Notifi's backend. They can be re-added using the
ensure method without re-verification.
- Delete: Targets are removed from the target group and permanently deleted from Notifi's backend. Re-adding them will require re-verification.
- Unspecified Targets: If a target type is not included in the
alterTargetGroup call, it will be removed from the target group.
🪢 Topics & Alerts
There are two key concepts of Notifi's notification source: Alert and Topic.
Topic is the event allowed to be subscribed (Tenant specific).
Alert is the subscription of the Topic (User specific).
Topic is the prototype of the Alert.
Topics
We can get the available topics of the tenant by calling fetchTenantConfig client method.
const id = '<ui-card-config-of-the-tenant>';
const tenantConfig = await client.fetchTenantConfig({
id,
type: 'SUBSCRIPTION_CARD',
});
const topics = tenantConfig.fusionEventTopics;
NOTE:
Alerts
We can subscribe users to a topic by creating an alert object by using the ensureFusionAlerts client method.
The following methods are used to get and delete alerts:
The full example of getting, deleting, and creating an alert is shown below:
import { resolveObjectArrayRef } from '@notifi-network/notifi-frontend-client';
const topic = topics[0];
const fusionEventDescriptor = topic.fusionEventDescriptor;
const fusionEventMetadataJson = fusionEventDescriptor.metadata;
const fusionEventId = fusionEventDescriptor.id;
if (fusionEventMetadataJson && fusionEventId) {
const fusionEventMetadata = getFusionEventMetadata(topic);
const filters = fusionEventMetadata?.filters?.filter(isAlertFilter);
const fusionFilterOptionsInput: FusionFilterOptions['input'] = {};
if (filters && filters.length > 0) {
const userInputParams = filters[0].userInputParams;
const userInputOptions: UserInputOptions = {};
userInputParams.forEach((userInputParm) => {
userInputOptions[userInputParm.name] = userInputParm.defaultValue;
});
fusionFilterOptionsInput[filters[0].name] = userInputOptions;
}
const filterOptions: FusionFilterOptions = {
version: 1,
input: fusionFilterOptionsInput,
};
const subscriptionValueOrRef =
fusionEventMetadata?.uiConfigOverride?.subscriptionValueOrRef;
const subscriptionValue = subscriptionValueOrRef
? resolveObjectArrayRef('', subscriptionValueOrRef, inputs)
: null;
const alertToCreate = fusionEventId;
const createAlertInputs = [
{
fusionEventId: fusionEventId,
name: alertToCreate,
targetGroupId,
subscriptionValue: subscriptionValue?.[0]?.value,
filterOptions: JSON.stringify(filterOptions),
},
];
}
const existingAlerts = await getAlerts()) ?? [];
const existingAlertNames = alerts.map((alert) => alert.name);
if (existingAlertNames.includes(alerToCreate)) {
const id = existingAlerts[alerToCreate].id;
await client.deleteAlert({ id });
}
await frontendClient.ensureFusionAlerts({ alerts: createAlertInputs });
const getFusionEventMetadata = (
topic: FusionEventTopic,
): FusionEventMetadata | null => {
const parsedMetadata = JSON.parse(
topic.fusionEventDescriptor.metadata ?? '{}',
);
if (isFusionEventMetadata(parsedMetadata)) {
return parsedMetadata;
}
return null;
};
const isFusionEventMetadata = (
metadata: unknown,
): metadata is FusionEventMetadata => {
const metadataToVerify = metadata as any;
if (typeof metadataToVerify !== 'object' || !metadataToVerify) {
return false;
}
return (
'filters' in metadataToVerify && Array.isArray(metadataToVerify.filters)
);
};
Related Information:
- inputs: it is a concept which is used when a dynamic topic subscription value needs to be adopted. The input is a key-value pair object. The key is the value associated with the Create a new topic section in Admin Portal. The value is the dynamic value that needs to be passed into the topic subscription. The value needs to be an array of InputObject interface. See the doc here
🔔 Notification History
To get notification history, use the getFusionNotificationHistory()
const { nodes, pageInfo } = await client.getFusionNotificationHistory({
first,
after,
});
const notifications = nodes.map((node) => {
return {
icon: node.detail.icon,
topicName: node.detail.sourceName,
message: node.detail.message,
timestamp: node.createdDate,
};
});
⚠️ Listening to events
Using addEventListener method allows to monitor the stateChanged event.
Listening to event
const id = client.addEventListener('stateChanged', eventHandler, errorHandler);
const eventHandler = (event) => {
if (event.__typename === 'TargetStateChangedEvent') {
console.info('User target state changed', event);
}
if (event.__typename === 'NotificationHistoryStateChangedEvent') {
console.info('New notification coming', event);
}
};
const errorHandler = (error: Error) => {
console.error('Error occurred', error);
};
Removing the event listener
client.removeEventListener('stateChanged', id);
Contributing & Testing
We welcome and appreciate your contributions! Please review our contribution guidelines before getting started.
To ensure a smooth contribution process, please run npm run test below to confirm that all tests pass before submitting your changes
If your contribution includes a new feature, you may also want to create a dedicated test case within the __tests__/NotifiFrontendClient.test.ts file to validate its functionality.
📝 More examples
-
notifi-dapp-example: A dapp example built with @notifi/notifi-frontend-client and @notifi/notifi-react SDKs.
-
notifi-react-example-v2: React contexts & components library built with @notifi/notifi-frontend-client.