Creating Connectors
Thanks for your interest in adding a new connector to Khizab! Please take a moment to review this document before starting work on a new connector.
Overview
This guide details how to create new connectors and upstream them back into Khizab. By following these steps, you will understand the development process, workflow, and requirements for new connectors. Not all connectors will be accepted into Khizab for a variety of reasons outlined in this document.
In addition, for connector requests to be accepted, the team creating the connector must sponsor Khizab. It takes time and effort to maintain third-party connectors. Khizab is an OSS project that depends on sponsors and grants to continue our work. Please get in touch via [email protected] if you have questions about sponsoring.
Please ask first before starting work on a new connector.
To avoid having your pull request declined after investing time and effort into a new connector, we ask that contributors create a Connector Request before starting work. This ensures the connector solves for an important or general use-case of interest to Khizab users and is well supported by the Khizab and connector teams.
1. Follow the contributing guide
Check out the Contributing Guide to get your local development environment set up and learn more about the contributing workflow.
2. Create a new file for the connector
Create a new file in packages/connector/src
named after the connector you want to add.
For example, if you want to add Foo, you would create a file named foo.ts
. File names should be camel-cased and as short as possible.
3. Create the connector object.
Import createConnector
from @khizab/core
and export a new function that accepts a parameters object and returns the createConnector
result. This is the base of all connectors. The name of the connector name should be the same as the file name.
4. Add the missing properties to the object
Now that the base of the connector is set up, you should see a type error that looks something like this:
ts
Type '{}' is missing the following properties from type '{ readonly icon?: string | undefined; readonly id: string; readonly name: string; readonly type: string; setup?: (() => Promise<void>) | undefined; connect: (parameters?: { isReconnecting?: boolean | undefined; } | undefined) => Promise<...>; ... 6 more ...; signMessage: (message: SignMessagePayload) => Promise<...>...': id, name, type, connect, and 7 more.2740Type '{}' is missing the following properties from type '{ readonly icon?: string | undefined; readonly id: string; readonly name: string; readonly type: string; setup?: (() => Promise<void>) | undefined; connect: (parameters?: { isReconnecting?: boolean | undefined; } | undefined) => Promise<...>; ... 6 more ...; signMessage: (message: SignMessagePayload) => Promise<...>...': id, name, type, connect, and 7 more.createConnector ((config ) =>({}) );
ts
Type '{}' is missing the following properties from type '{ readonly icon?: string | undefined; readonly id: string; readonly name: string; readonly type: string; setup?: (() => Promise<void>) | undefined; connect: (parameters?: { isReconnecting?: boolean | undefined; } | undefined) => Promise<...>; ... 6 more ...; signMessage: (message: SignMessagePayload) => Promise<...>...': id, name, type, connect, and 7 more.2740Type '{}' is missing the following properties from type '{ readonly icon?: string | undefined; readonly id: string; readonly name: string; readonly type: string; setup?: (() => Promise<void>) | undefined; connect: (parameters?: { isReconnecting?: boolean | undefined; } | undefined) => Promise<...>; ... 6 more ...; signMessage: (message: SignMessagePayload) => Promise<...>...': id, name, type, connect, and 7 more.createConnector ((config ) =>({}) );
The type error tells you what properties are missing from createConnector
's return type. Add them all in!
Properties
icon
: Optional icon URL for the connector.id
: The ID for the connector. This should be camel-cased and as short as possible. Example:fooBarBaz
.name
: Human-readable name for the connector. Example:'Foo Bar Baz'
.
Methods
connect
: Function for connecting the connector.disconnect
: Function for disconnecting the connector.getAccounts
: Function that returns the connected accounts for the connector.getChainId
: Function that returns the connected chain ID for the connector.getProvider
: Function that returns the underlying provider interface for internal use throughout the connector.setup
: Optional function for running when the connector is first created.switchChain
: Optional function for switching the connector's active chain.
Events
onAccountsChanged
: Function for subscribing to account changes internally in the connector.onChainChanged
: Function for subscribing to chain changes internally in the connector.onConnect
: Function for subscribing to connection events internally in the connector.onDisconnect
: Function for subscribing to disconnection events internally in the connector.onMessage
: Optional function for subscribing to messages internally in the connector.
Parameters
createConnector
also has the following config properties you can use within the connector:
chains
: List of chains configured by the user.emitter
: Emitter for emitting events. Used to sync connector state with KhizabConfig
. The following events are available:change
: Emitted when the connected accounts or chain changes.connect
: Emitted when the connector connects.disconnect
: Emitted when the connector disconnects.error
: Emitted when the connector receives an error.message
: Emitted when the connector receives a message.
storage
: Optional storage configured by the user. Defaults to wrapper around localStorage.
TIP
If you plan to use a third-party SDK, it should have minimal dependencies (limit bundle size, supply chain attacks, etc.) and use the most permissive license possible (ideally MIT). Any third-party packages, should also have "sideEffects": false
in their package.json
file for maximum tree-shakability support.
5. Export the connector
Export the connector from packages/connector/src/exports/index.ts
in alphabetic order.
export { fooBarBaz } from "./fooBarBaz.js";
export { fooBarBaz } from "./fooBarBaz.js";
6. Try out the connector and add tests
While building a connector, it can be useful to try it out with Khizab. You can use the development playgrounds for testing your changes.
Ideally, you should also be able to add tests for the connector in a connectorName.test.ts
file. This isn't always easy so at a minimum please create a test file with instructions for how to test the connector manually. The test file should include actual tests or "instruction tests" for the following:
- How to connect the connector.
- How to disconnect the connector.
- How to switch the connector's active chain (if applicable).
Remember to include all info required to test the connector, like software to install (browser extension, mobile app, etc.), smart contracts to interact with/deploy, etc.
Finally, you should also update the test file in packages/connectors/src/exports/index.test.ts
to include the new connector. You can do this manually or by running:
pnpm test:update packages/connectors/src/exports/index.test.ts
pnpm test:update packages/connectors/src/exports/index.test.ts
7. Add your team to CODEOWNERS
It is critical that connectors are updated in a timely manner and actively maintained so that users of Khizab can rely on them in production settings.
The Khizab core team will provide as much assistance as possible to keep connectors up-to-date with breaking changes from Khizab, but it is your responsibility to ensure that any dependencies and issues/discussions related to the connector are handled in a timely manner. If issues are not resolved in a timely manner, the connector may be removed from Khizab.
In support of this goal, add at least one member of your team to the CODEOWNERS file so that you get notified of pull requests, issues, etc. related to the connector. You can add your team like this:
/packages/connectors/src/fooBarBaz @tmm @jxom
/packages/connectors/src/fooBarBaz @tmm @jxom
For more info about GitHub code owners, check out the GitHub Documentation.
8. Document the connector
The connector should be documented. Follow the step on writing documentation to get set up with running the docs site locally and add the required pages.
9. Create a changeset
Now that the connector works and has tests, it's time to create a changeset to prepare for release. Run the following to create a changeset:
pnpm changeset
pnpm changeset
The changeset should be a patch
applied to the @khizab/connectors
repository with the description Added [ConnectorName]
, For example, Added Foo Bar Baz
.
10. Create a pull request
The connector is ready to go! Create a pull request and the connector should make it into a future release of Khizab after some review.