-
Notifications
You must be signed in to change notification settings - Fork 10
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #30 from Darlington02/develop
feat: connector base
- Loading branch information
Showing
10 changed files
with
2,148 additions
and
465 deletions.
There are no files selected for viewing
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 = `` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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] | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 | ||
} |
Oops, something went wrong.