🚧 Alpha
An easy-to-use SDK for building frontend interfaces that interact with a community server and any related smart contracts that we use. This SDK is frontend framework agnostic and includes types, API calls, and state management.
Welcome to the official SDK for our platform. This SDK provides a simple and efficient way to integrate your frontend application with our backend services. It's designed for speeding up development of client applications by providing state management out of the box. You can simply make requests and have your UI listen for and display the data.
Whether you're building a small side project or a large-scale commercial application, this SDK has everything you need to get started. It includes:
- Types: Comprehensive type definitions for all of our API responses and request bodies.
- API Calls: Pre-configured API calls to interact with our backend services.
- State Management: Built-in state management to help you manage the state of your application.
We hope you find this SDK useful, and we're excited to see what you build with it!
To install the SDK, run the following command in your terminal:
npm install --save @citizenwallet/sdk
A voucher is actually simply a random private key which is used to generate a Smart Account that we top up and send to someone with some metadata.
You are effectively creating an account, topping it up, describing what it is and sending that over to the other person.
const communityAlias = 'bread'; // config.community.alias
const voucherName = 'Voucher for X tokens';
const creatorAddress = '0x59D17ec3d96C52d4539eed43f33492679ae4aCf7'; // since the user who is redeeming will only see a transaction from the voucher to them, this allows displaying the original creator on the UI.
const signer = ethers.Wallet.createRandom(); // generate a random account which will be used for the voucher.
const voucher = await createVoucher(
communityAlias,
voucherName,
creatorAddress,
signer
);
Example:
{
"voucherLink": "https://app.citizenwallet.xyz/#/?voucher=H4sIAEFF7WYAAw3JyQ3AMAgEwIoiLeYy5YDBJaT-5D3vemZEhHHS83INkWm5G4uo6MJkN4f_jFiEuhZnI3sHyJkaH_18VYRDAAAA¶ms=H4sIAEFF7WYAAw3LsRLCIAwA0L_p4qIkEDIwSC3_kUI4vdNyR6nn59vlbU_eL9nD2lXKlE9H6-H6s_y4kWYo7GZrClpg1YJQAZCNIxZFmStNknM7tnEWMItjghRNrQ6i95YpMSzI0Zv7SmhIgFOZNvloGLqPy7cd-an9D-Zqgw6DAAAA",
"voucherAccountAddress": "0x32E6973FB2ff63B88597F93E49B82Ab7427a39Fd"
}
const parsed = parseVoucher(voucher.voucherLink);
Example:
{
"voucher": {
"alias": "bread",
"creator": "0x59D17ec3d96C52d4539eed43f33492679ae4aCf7",
"account": "0x32E6973FB2ff63B88597F93E49B82Ab7427a39Fd",
"name": "test voucher"
},
"signer": {
"provider": null,
"address": "0x60fab84316061E5c7C2eD09a2c4Be454B6B1fC69"
}
}
signer = type ethers.Wallet
Every community and their currency has a configuration in a json format. These can be retrieved in order to interact with the community in question.
We have created methods that you can call with state management, so that you can focus on building your user interface.
import { useConfig } from '@citizenwallet/sdk';
function MyComponent() {
const [subscribe, actions] = useConfig();
useEffect(() => {
actions.getConfigs();
}, [actions])
const configs = subscribe(state => state.configs); // the component will re-render whenever .configs changes
const loading = subscribe(state => state.loading); // same here with loading
return (
<div>
{loading && !configs?.length ? 'loading...' : configs.map(c => <div>c.community.name</div>)}
</div>
)
}
function MyComponent({ slug }) {
const [subscribe, actions] = useConfig();
useEffect(() => {
actions.getConfig(slug); // fetch for a slug and any time the slug changes
}, [actions, slug])
const config = subscribe(state => state.config); // the component will re-render whenever .config changes
const loading = subscribe(state => state.loading); // same here with loading
return (
<div>
{loading && !configs?.length ? 'loading...' : <div>config.community.name</div>}
</div>
)
}
You can interact with the ConfigService directly as well.
import { ConfigService } from '@citizenwallet/sdk';
const configService = new ConfigServce();
const configs = await configService.get(); // all configs
const configService = new ConfigServce();
const config = await configService.getBySlug('my-slug'); // a single config
You can point the service at another base url, if you have your own place where a communities.json
file is stored.
import { useConfig } from '@citizenwallet/sdk';
const [subscribe, actions] = useConfig('https://your-custom-base-url.com');
You can do the same with the ConfigService.
import { ConfigService, ApiService } from '@citizenwallet/sdk';
const configService = new ConfigService(new ApiService('https://your-custom-base-url.com'));
An IOU is a smart contract which manages IOU redemption for a single ERC 20 token. It will hash, verify signatures and keep track of redemptions.
Import the SDK into your project:
import { useSimpleFaucetContract } from '@citizenwallet/sdk';
Then, you can use the SDK's functions to interact with the backend. For example:
const [subscribe, actions] = useSimpleFaucetContract(address, signer);
actions.getHash(...);
actions.redeem(...);
const loading = subscribe(state => state.loading);
const hash = subscribe(state => state.hash);
const hash = actions.store.getState().hash;
function Component() {
const [subscribe, actions] = useSimpleFaucetContract(
address, // contract address
rpcSigner, // ethers JsonRpcSigner
sender, // the smart contract account belonging to the signer
config, // the community config
);
useEffect(() => {
actions.fetchMetadata();
actions.redeem();
}, [actions]);
const loading = subscribe(state => state.loading);
const metadataLoading = subscribe(state => state.metadataLoading);
const tokenAddress = subscribe(state => state.tokenAddress);
const redeemAmount = subscribe(state => state.amount);
return (
<div>
{metadataLoading? : 'loading meta data' : `token: ${tokenAddress} redeem amount: ${redeemAmount}`}
{loading ? 'loading...' : 'redeemed!'}
</div>
)
}
If you want to make direct calls to the contract and handle everything else yourself.
Import the SDK into your project:
import { SimpleFaucetContractService } from '@citizenwallet/sdk';
Then, you can use the SDK's functions to interact with the backend. For example:
const contract = SimpleFaucetContractService(
address, // contract address
rpcSigner, // ethers JsonRpcSigner
sender, // the smart contract account belonging to the signer
config, // the community config
);
await contract.redeem();
An IOU is a smart contract which manages IOU redemption for a single ERC 20 token. It will hash, verify signatures and keep track of redemptions.
Import the SDK into your project:
import { useIOUContract } from '@citizenwallet/sdk';
Then, you can use the SDK's functions to interact with the backend. For example:
const [subscribe, actions] = useIOUContract(address, signer);
actions.getHash(...);
actions.redeem(...);
const loading = subscribe(state => state.loading);
const hash = subscribe(state => state.hash);
const hash = actions.store.getState().hash;
function Component() {
const [subscribe, actions] = useIOUContract(address, signer);
useEffect(() => {
actions.getHash(...);
}, [actions]);
const loading = subscribe(state => state.loading);
const hash = subscribe(state => state.hash);
return (
<div>{loading ? 'loading...' : hash}</div>
)
}
If you want to make direct calls to the contract and handle everything else yourself.
Import the SDK into your project:
import { IOUContractService } from '@citizenwallet/sdk';
Then, you can use the SDK's functions to interact with the backend. For example:
const contract = IOUContractService(address, signer);
const hash = await contract.getHash(...);
await contract.redeem(...);
To build the SDK, run the following command in your terminal:
npm run build
This will compile the TypeScript code to JavaScript.
To automatically recompile the SDK when a file changes, run the following command in your terminal:
npm run watch
We welcome contributions! Please see our contributing guidelines for more details.
This project is licensed under the MIT license.