Skip to content

Commit

Permalink
Merge pull request #30 from Darlington02/develop
Browse files Browse the repository at this point in the history
feat: connector base
  • Loading branch information
Darlington02 authored Feb 17, 2024
2 parents 4b49e5b + 9786a8f commit 17c4dc8
Show file tree
Hide file tree
Showing 10 changed files with 2,148 additions and 465 deletions.
2,162 changes: 1,713 additions & 449 deletions connector/package-lock.json

Large diffs are not rendered by default.

14 changes: 13 additions & 1 deletion connector/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -8,14 +8,27 @@
"files": [
"dist"
],
"types": "./dist/index.d.ts",
"main": "./dist/tokenbound-connector.umd.cjs",
"module": "./dist/tokenbound-connector.js",
"exports": {
".": {
"import": "./dist/tokenbound-connector.js",
"require": "./dist/tokenbound-connector.umd.cjs",
"types": "./dist/index.d.ts"
}
},
"dependencies": {
"@testing-library/jest-dom": "^5.17.0",
"@testing-library/react": "^13.4.0",
"@testing-library/user-event": "^13.5.0",
"eventemitter3": "^5.0.1",
"get-starknet-core": "^3.2.0",
"react": "^18.2.0",
"react-dom": "^18.2.0",
"react-scripts": "5.0.1",
"starknet": "^5.24.3",
"starknetkit": "^1.1.3",
"web-vitals": "^2.1.4"
},
"scripts": {
Expand All @@ -41,7 +54,6 @@
"@types/testing-library__jest-dom": "^5.14.9",
"eslint": "^8.50.0",
"eslint-config-prettier": "^9.0.0",
"starknet": "^6.0.0",
"typescript": "^5.2.2",
"vite": "^4.4.9",
"vite-plugin-dts": "^3.5.1",
Expand Down
21 changes: 21 additions & 0 deletions connector/src/connector/constants.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
export class ConnectorNotConnectedError extends Error {
name = "ConnectorNotConnectedError";
message = "Connector not connected";
}

export class ConnectorNotFoundError extends Error {
name = "ConnectorNotFoundError";
message = "Connector not found";
}

export class UserRejectedRequestError extends Error {
name = "UserRejectedRequestError";
message = "User rejected request";
}

export class UserNotConnectedError extends Error {
name = "UserNotConnectedError";
message = "User not connected";
}

export const tokenbound_icon = ``
22 changes: 22 additions & 0 deletions connector/src/connector/helpers/RPCNodes.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
export type PublicRpcNode = {
mainnet: string
testnet: string
}

// Public RPC nodes
export const BLAST_RPC_NODE: PublicRpcNode = {
mainnet: "https://starknet-mainnet.public.blastapi.io",
testnet: "https://starknet-testnet.public.blastapi.io",
} as const

export const LAVA_RPC_NODE: PublicRpcNode = {
mainnet: "https://rpc.starknet.lava.build",
testnet: "https://rpc.starknet-testnet.lava.build",
} as const

export const PUBLIC_RPC_NODES = [BLAST_RPC_NODE, LAVA_RPC_NODE] as const

export function getRandomPublicRPCNode() {
const randomIndex = Math.floor(Math.random() * PUBLIC_RPC_NODES.length)
return PUBLIC_RPC_NODES[randomIndex]
}
11 changes: 11 additions & 0 deletions connector/src/connector/helpers/mapChainToNodeUrl.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
import { getRandomPublicRPCNode } from "./RPCNodes"

export function mapChainToNodeUrl(chainId: string): string {
const PublicRpcNode = getRandomPublicRPCNode()
if(chainId == "SN_MAIN") {
return PublicRpcNode.mainnet
}
else {
return PublicRpcNode.testnet
}
}
221 changes: 219 additions & 2 deletions connector/src/connector/index.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,220 @@
// connector base, important links:
import { connect } from "starknetkit"
import { InjectedConnector } from "starknetkit/injected"
import { WebWalletConnector } from "starknetkit/webwallet"
import { ArgentMobileConnector } from "starknetkit/argentMobile"

// https://github.com/argentlabs/starknetkit/blob/develop/src/connectors/webwallet/index.ts
import type {
AccountChangeEventHandler,
StarknetWindowObject,
} from "get-starknet-core"

import type {
AccountInterface,
ProviderInterface,
} from "starknet"

import {
Connector,
type ConnectorIcons
} from "./types/connector"

import {
ConnectorNotConnectedError,
ConnectorNotFoundError,
UserNotConnectedError,
UserRejectedRequestError,
} from "./constants"

import { tokenbound_icon } from "./constants"
import { getTokenboundStarknetWindowObject } from "./windowObject/TBAStarknetWindowObject"

let _wallet: StarknetWindowObject | null = null

interface TokenboundConnectorOptions {
tokenboundAddress: string,
parentAccountId: string,
provider?: ProviderInterface
}

export class TokenboundConnector extends Connector {
private _wallet: StarknetWindowObject | null = null
private _options: TokenboundConnectorOptions

constructor(options: TokenboundConnectorOptions) {
super()
this._options = options
}

available(): boolean {
return true
}

async ready(): Promise<boolean> {
if(!_wallet) {
this._wallet = null
return false
}

this._wallet = _wallet
return this._wallet.isPreauthorized()
}

get id(): string {
this._wallet = _wallet
return this._wallet?.id || "TBA"
}

get name(): string {
this._wallet = _wallet
return this._wallet?.name || "Tokenbound Account"
}

get icon(): ConnectorIcons {
return {
light: tokenbound_icon,
dark: tokenbound_icon
}
}

get wallet(): StarknetWindowObject {
if(!this._wallet) {
throw new ConnectorNotConnectedError()
}
return this._wallet
}

get subtitle(): string {
return "Powered by Starknet Africa"
}

async connect(): Promise<StarknetWindowObject> {
await this.ensureWallet()

if(!this._wallet) {
throw new ConnectorNotFoundError()
}

try {
await this._wallet.enable({ starknetVersion: "v4" })
}
catch {
throw new UserRejectedRequestError()
}

if(!this._wallet.isConnected) {
throw new UserRejectedRequestError()
}

return this._wallet
}

async disconnect(): Promise<void> {
if(!this.available() && !this._wallet) {
throw new ConnectorNotFoundError()
}

if(!this._wallet?.isConnected) {
throw new UserNotConnectedError()
}

_wallet = null
this._wallet = _wallet
}

async account(): Promise<AccountInterface> {
this._wallet = _wallet

if(!this._wallet || !this._wallet.account) {
throw new ConnectorNotConnectedError()
}

return this._wallet.account as unknown as AccountInterface
}

async chainId(): Promise<bigint> {
if (!this._wallet || !this.wallet.account || !this._wallet.provider) {
throw new ConnectorNotConnectedError()
}

const chainIdHex = await this._wallet.provider.getChainId()
const chainId = BigInt(chainIdHex)
return chainId
}

async initEventListener(accountChangeCb: AccountChangeEventHandler) {
this._wallet = _wallet
if (!this._wallet) {
throw new ConnectorNotConnectedError()
}

this._wallet.on("accountsChanged", accountChangeCb)
}

async removeEventListener(accountChangeCb: AccountChangeEventHandler) {
this._wallet = _wallet
if (!this._wallet) {
throw new ConnectorNotConnectedError()
}

this._wallet.off("accountsChanged", accountChangeCb)

_wallet = null
this._wallet = null
}

private async connectParentAccount(id: string): Promise<AccountInterface> {
let parentWallet

if(id == "argentMobile") {
const { wallet } = await connect({
connectors: [
new ArgentMobileConnector()
]
})
parentWallet = wallet
}
else if(id == "argentWebWallet") {
const { wallet } = await connect({
connectors: [
new WebWalletConnector({
url: "https://web.argent.xyz"
})
]
})
parentWallet = wallet
}
else {
const { wallet } = await connect({
connectors: [
new InjectedConnector({
options: {id: id}
})
]
})
parentWallet = wallet
}

return parentWallet?.account as AccountInterface
}

private async ensureWallet(): Promise<void> {
const tokenboundAddress = this._options.tokenboundAddress
const parentAccount = await this.connectParentAccount(this._options.parentAccountId)
const provider = this._options.provider

const wallet = await getTokenboundStarknetWindowObject(
{
id: "TBA",
icon: tokenbound_icon as string,
name: "Tokenbound Account",
version: "1.0.0"
},
tokenboundAddress,
parentAccount,
provider
)

_wallet = wallet ?? null
this._wallet = _wallet
}
}
53 changes: 53 additions & 0 deletions connector/src/connector/types/connector.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
import EventEmitter from "eventemitter3"
import type { StarknetWindowObject } from "get-starknet-core"
import { AccountInterface } from "starknet"

/** Connector icons, as base64 encoded svg. */
export type ConnectorIcons = {
/** Dark-mode icon. */
dark?: string
/** Light-mode icon. */
light?: string
}

/** Connector data. */
export type ConnectorData = {
/** Connector account. */
account?: string
/** Connector network. */
chainId?: bigint
}

/** Connector events. */
export interface ConnectorEvents {
/** Emitted when account or network changes. */
change(data: ConnectorData): void
/** Emitted when connection is established. */
connect(data: ConnectorData): void
/** Emitted when connection is lost. */
disconnect(): void
}

export abstract class Connector extends EventEmitter<ConnectorEvents> {
/** Unique connector id. */
abstract get id(): string
/** Connector name. */
abstract get name(): string
/** Connector icons. */
abstract get icon(): ConnectorIcons

/** Whether connector is available for use */
abstract available(): boolean
/** Whether connector is already authorized */
abstract ready(): Promise<boolean>
/** Connect wallet. */
abstract connect(): Promise<StarknetWindowObject>
/** Disconnect wallet. */
abstract disconnect(): Promise<void>
/** Get current account. */
abstract account(): Promise<AccountInterface>
/** Get current chain id. */
abstract chainId(): Promise<bigint>
/** Connector StarknetWindowObject */
abstract get wallet(): StarknetWindowObject
}
Loading

0 comments on commit 17c4dc8

Please sign in to comment.