├── .npmrc
├── packages
├── dappauth
│ ├── .gitignore
│ ├── babel.config.cjs
│ ├── src
│ │ ├── ABIs
│ │ │ └── ERC1271.js
│ │ ├── utils
│ │ │ └── index.js
│ │ └── index.js
│ ├── CHANGELOG.md
│ ├── test
│ │ └── unit
│ │ │ ├── provider-mock.js
│ │ │ ├── jest.config.cjs
│ │ │ ├── test-utils.js
│ │ │ ├── contract-mock.js
│ │ │ └── index.test.js
│ ├── README.md
│ ├── LICENSE
│ ├── package.json
│ └── rollup.config.js
├── blocto-sdk
│ ├── .env.example
│ ├── src
│ │ ├── lib
│ │ │ ├── storage
│ │ │ │ ├── index.ts
│ │ │ │ ├── memoryStorage.ts
│ │ │ │ └── storage.ts
│ │ │ ├── isEmail.ts
│ │ │ ├── constant.ts
│ │ │ ├── invariant.ts
│ │ │ ├── utf8toHex.ts
│ │ │ ├── addSelfRemovableHandler.ts
│ │ │ ├── responseSessionGuard.ts
│ │ │ ├── frame.ts
│ │ │ ├── getEvmSupport.ts
│ │ │ └── isValidTransaction.ts
│ │ ├── providers
│ │ │ ├── types
│ │ │ │ ├── blocto.d.ts
│ │ │ │ ├── aptos.d.ts
│ │ │ │ └── ethereum.d.ts
│ │ │ └── blocto.ts
│ │ ├── main.ts
│ │ ├── index.d.ts
│ │ ├── __tests__
│ │ │ ├── isValidTransaction.test.js
│ │ │ ├── ethereum.test.ts
│ │ │ ├── fixtures
│ │ │ │ └── getEvmSupport.ts
│ │ │ └── index.test.js
│ │ └── constants.ts
│ ├── babel.config.json
│ ├── .eslintrc.cjs
│ ├── dev-cert
│ │ └── index.js
│ ├── jest.config.cjs
│ ├── rollup.config.dev.js
│ ├── .gitignore
│ ├── package.json
│ ├── README.md
│ ├── rollup.config.js
│ ├── example
│ │ └── index.html
│ └── tsconfig.json
├── tsconfig
│ ├── package.json
│ ├── react-library.json
│ ├── nextjs.json
│ └── base.json
└── eslint-config-custom
│ ├── index.js
│ └── package.json
├── .husky
└── commit-msg
├── .prettierrc
├── .github
├── pr-labeler.yml
├── dependabot.yml
├── pull_request_template.md
└── workflows
│ ├── pr-labeler.yml
│ ├── create-asana-attachment.yaml
│ ├── test.yml
│ ├── publish.yml
│ └── release-branch.yml
├── adapters
├── wagmi-connector
│ ├── src
│ │ ├── index.ts
│ │ ├── connector.ts
│ │ └── connector.test.ts
│ ├── .gitignore
│ ├── vite.config.ts
│ ├── tsconfig.json
│ ├── package.json
│ └── README.md
├── web3modal-connector
│ ├── src
│ │ ├── index.ts
│ │ └── web3ModalConfig.ts
│ ├── .gitignore
│ ├── README.md
│ ├── vite.config.ts
│ ├── tsconfig.json
│ ├── package.json
│ └── CHANGELOG.md
├── aptos-wallet-adapter-plugin
│ ├── jest.config.js
│ ├── README.md
│ ├── tsconfig.json
│ ├── package.json
│ ├── src
│ │ ├── __tests__
│ │ │ └── index.test.ts
│ │ └── index.ts
│ └── CHANGELOG.md
├── connectkit-connector
│ ├── src
│ │ ├── index.ts
│ │ ├── constants
│ │ │ └── index.tsx
│ │ └── components
│ │ │ └── BloctoLogo.tsx
│ ├── .gitignore
│ ├── vite.config.ts
│ ├── tsconfig.json
│ ├── README.md
│ ├── package.json
│ └── CHANGELOG.md
├── rainbowkit-connector
│ ├── jest.config.js
│ ├── tsconfig.json
│ ├── package.json
│ ├── README.md
│ └── src
│ │ ├── index.ts
│ │ └── __tests__
│ │ └── index.test.ts
└── web3-react-connector
│ ├── README.md
│ ├── tsconfig.json
│ ├── package.json
│ └── src
│ └── index.ts
├── .eslintrc.js
├── .yarnrc
├── turbo.json
├── .changeset
├── config.json
└── README.md
├── commitlint.config.js
├── LICENSE
├── package.json
├── .gitignore
├── CONTRIBUTING.md
└── README.md
/.npmrc:
--------------------------------------------------------------------------------
1 | auto-install-peers = true
2 |
--------------------------------------------------------------------------------
/packages/dappauth/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules/
2 | .nyc_output/
3 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/.env.example:
--------------------------------------------------------------------------------
1 | SERVER=http://localhost:8702
2 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/storage/index.ts:
--------------------------------------------------------------------------------
1 | export * from './storage';
2 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/babel.config.json:
--------------------------------------------------------------------------------
1 | {
2 | "presets": [["@babel/preset-env"]]
3 | }
4 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/.eslintrc.cjs:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | root: true,
3 | extends: ['custom'],
4 | };
5 |
--------------------------------------------------------------------------------
/.husky/commit-msg:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env sh
2 | . "$(dirname -- "$0")/_/husky.sh"
3 |
4 | npx --no -- commitlint --edit ${1}
5 |
--------------------------------------------------------------------------------
/.prettierrc:
--------------------------------------------------------------------------------
1 | {
2 | "semi": true,
3 | "singleQuote": true,
4 | "trailingComma": "es5",
5 | "printWidth": 80
6 | }
7 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/isEmail.ts:
--------------------------------------------------------------------------------
1 | export const isEmail = (value: string): boolean => /\S+@\S+\.\S+/.test(value);
2 |
--------------------------------------------------------------------------------
/.github/pr-labeler.yml:
--------------------------------------------------------------------------------
1 | Feature: ['feature/*', 'feat/*']
2 | Refactor: refactor/*
3 | BugFix: ['fix/*', 'bugfix/*']
4 | HotFix: ['hotfix/*']
--------------------------------------------------------------------------------
/adapters/wagmi-connector/src/index.ts:
--------------------------------------------------------------------------------
1 | export { blocto, createConnector } from './connector';
2 | export type * from './connector';
3 |
--------------------------------------------------------------------------------
/.eslintrc.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | root: true,
3 | // This tells ESLint to load the config from the package `eslint-config-custom`
4 | extends: ['custom'],
5 | };
6 |
--------------------------------------------------------------------------------
/adapters/web3modal-connector/src/index.ts:
--------------------------------------------------------------------------------
1 | export { BloctoConnector } from '@blocto/wagmi-connector';
2 | export { default as BloctoWeb3ModalConfig } from './web3ModalConfig';
3 |
--------------------------------------------------------------------------------
/.yarnrc:
--------------------------------------------------------------------------------
1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
2 | # yarn lockfile v1
3 |
4 |
5 | lastUpdateCheck 1686022997016
6 | yarn-path ".yarn/releases/yarn-1.19.0.js"
7 |
--------------------------------------------------------------------------------
/adapters/aptos-wallet-adapter-plugin/jest.config.js:
--------------------------------------------------------------------------------
1 | /** @type {import('ts-jest').JestConfigWithTsJest} */
2 | module.exports = {
3 | preset: 'ts-jest',
4 | testEnvironment: 'node',
5 | };
--------------------------------------------------------------------------------
/adapters/connectkit-connector/src/index.ts:
--------------------------------------------------------------------------------
1 | export { BloctoConnector } from '@blocto/wagmi-connector';
2 | export type * from '@blocto/wagmi-connector';
3 | export { BLOCTO_CONFIG } from './constants';
--------------------------------------------------------------------------------
/.github/dependabot.yml:
--------------------------------------------------------------------------------
1 | version: 2
2 | updates:
3 | - package-ecosystem: 'yarn'
4 | directory: '/'
5 | schedule:
6 | interval: 'daily'
7 | ignore:
8 | - dependency-name: 'web3'
9 |
--------------------------------------------------------------------------------
/packages/tsconfig/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "tsconfig",
3 | "version": "0.0.0",
4 | "private": true,
5 | "license": "MIT",
6 | "publishConfig": {
7 | "access": "public"
8 | }
9 | }
10 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/dev-cert/index.js:
--------------------------------------------------------------------------------
1 | const fs = require('fs');
2 |
3 | module.exports = {
4 | cert: fs.readFileSync(`${__dirname}/localhost.pem`),
5 | key: fs.readFileSync(`${__dirname}/localhost-key.pem`),
6 | };
7 |
--------------------------------------------------------------------------------
/adapters/rainbowkit-connector/jest.config.js:
--------------------------------------------------------------------------------
1 | /** @type {import('ts-jest').JestConfigWithTsJest} */
2 | module.exports = {
3 | preset: 'ts-jest',
4 | testEnvironment: 'node',
5 | extensionsToTreatAsEsm: [".ts", ".tsx"]
6 | };
--------------------------------------------------------------------------------
/packages/dappauth/babel.config.cjs:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | presets: [
3 | [
4 | '@babel/preset-env',
5 | {
6 | targets: {
7 | node: true,
8 | },
9 | },
10 | ],
11 | ],
12 | };
13 |
--------------------------------------------------------------------------------
/turbo.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "https://turbo.build/schema.json",
3 | "pipeline": {
4 | "build": {
5 | "dependsOn": ["^build"]
6 | },
7 | "test": {
8 | "dependsOn": ["build"]
9 | },
10 | "lint": {}
11 | }
12 | }
13 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/constant.ts:
--------------------------------------------------------------------------------
1 | export enum ERROR_MESSAGE {
2 | INVALID_TRANSACTION = 'Invalid transaction',
3 | INVALID_TRANSACTIONS = 'Invalid transactions',
4 | INVALID_TRANSACTION_VALUE = 'Transaction params "value" should be hex-encoded string',
5 | }
6 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/invariant.ts:
--------------------------------------------------------------------------------
1 | // instead invariant from package, since all error will throw on production
2 | function invariant(condition: unknown, format: string): void {
3 | if (!condition) {
4 | throw new Error(format);
5 | }
6 | }
7 | export default invariant;
8 |
--------------------------------------------------------------------------------
/packages/tsconfig/react-library.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "https://json.schemastore.org/tsconfig",
3 | "display": "React Library",
4 | "extends": "./base.json",
5 | "compilerOptions": {
6 | "jsx": "react-jsx",
7 | "lib": ["ES2015"],
8 | "module": "ESNext",
9 | "target": "es6"
10 | }
11 | }
12 |
--------------------------------------------------------------------------------
/.changeset/config.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "https://unpkg.com/@changesets/config@2.3.0/schema.json",
3 | "changelog": "@changesets/cli/changelog",
4 | "commit": false,
5 | "fixed": [],
6 | "linked": [],
7 | "access": "public",
8 | "baseBranch": "main",
9 | "updateInternalDependencies": "patch",
10 | "ignore": ["eslint-config-custom", "tsconfig"]
11 | }
12 |
--------------------------------------------------------------------------------
/adapters/wagmi-connector/.gitignore:
--------------------------------------------------------------------------------
1 | # Logs
2 | logs
3 | *.log
4 | npm-debug.log*
5 | yarn-debug.log*
6 | yarn-error.log*
7 | pnpm-debug.log*
8 | lerna-debug.log*
9 |
10 | node_modules
11 | dist
12 | dist-ssr
13 | *.local
14 |
15 | # Editor directories and files
16 | .vscode/*
17 | !.vscode/extensions.json
18 | .idea
19 | .DS_Store
20 | *.suo
21 | *.ntvs*
22 | *.njsproj
23 | *.sln
24 | *.sw?
25 |
--------------------------------------------------------------------------------
/adapters/web3modal-connector/.gitignore:
--------------------------------------------------------------------------------
1 | # Logs
2 | logs
3 | *.log
4 | npm-debug.log*
5 | yarn-debug.log*
6 | yarn-error.log*
7 | pnpm-debug.log*
8 | lerna-debug.log*
9 |
10 | node_modules
11 | dist
12 | dist-ssr
13 | *.local
14 |
15 | # Editor directories and files
16 | .vscode/*
17 | !.vscode/extensions.json
18 | .idea
19 | .DS_Store
20 | *.suo
21 | *.ntvs*
22 | *.njsproj
23 | *.sln
24 | *.sw?
25 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/utf8toHex.ts:
--------------------------------------------------------------------------------
1 | import { Buffer } from 'buffer';
2 | export type ValidInputTypes = Uint8Array | bigint | string | number | boolean;
3 |
4 | export const isHexString = (hex: ValidInputTypes) =>
5 | typeof hex === 'string' && /^0x[0-9A-Fa-f]*$/.test(hex);
6 |
7 | export const utf8ToHex = (str: string) => {
8 | return Buffer.from(str, 'utf8').toString('hex');
9 | };
10 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/providers/types/blocto.d.ts:
--------------------------------------------------------------------------------
1 | import { EIP1193Provider } from 'eip1193-provider';
2 |
3 | declare interface BloctoProviderInterface extends EIP1193Provider {
4 | isBlocto: boolean;
5 | isConnecting: boolean;
6 | appId: string;
7 | eventListeners: {
8 | [key: string]: Array<(arg?: any) => void>;
9 | };
10 | }
11 |
12 | export default BloctoProviderInterface;
13 |
--------------------------------------------------------------------------------
/adapters/connectkit-connector/.gitignore:
--------------------------------------------------------------------------------
1 | # Logs
2 | logs
3 | *.log
4 | npm-debug.log*
5 | yarn-debug.log*
6 | yarn-error.log*
7 | pnpm-debug.log*
8 | lerna-debug.log*
9 |
10 | node_modules
11 | dist
12 | dist-ssr
13 | *.local
14 |
15 | # Editor directories and files
16 | .vscode/*
17 | !.vscode/extensions.json
18 | .idea
19 | .DS_Store
20 | *.suo
21 | *.ntvs*
22 | *.njsproj
23 | *.sln
24 | *.sw?
25 |
--------------------------------------------------------------------------------
/adapters/web3-react-connector/README.md:
--------------------------------------------------------------------------------
1 | # web3-react-connector
2 |
3 | Blocto SDK connector for web3-react. For usage, please refer to [Blocto SDK web3-react connector](https://docs.blocto.app/blocto-sdk/evm-sdk/web3-react-connector)
4 |
5 | ## Development
6 |
7 | ### Install Dependencies
8 |
9 | ```bash
10 | yarn add @blocto/web3-react-connector
11 | ```
12 |
13 | ### Build
14 |
15 | ```bash
16 | yarn build
17 | ```
18 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/jest.config.cjs:
--------------------------------------------------------------------------------
1 | /** @type {import('ts-jest').JestConfigWithTsJest} */
2 | module.exports = {
3 | preset: 'ts-jest',
4 | testEnvironment: 'jsdom',
5 | transform: {
6 | '^.+\\.(ts|tsx)?$': 'ts-jest',
7 | '^.+\\.(js|jsx)$': 'babel-jest',
8 | },
9 | testPathIgnorePatterns: ['node_modules/(?!variables/.*)', '/fixtures/', '\\.d\\.ts$'],
10 | automock: false,
11 | resetMocks: false,
12 | };
13 |
--------------------------------------------------------------------------------
/.github/pull_request_template.md:
--------------------------------------------------------------------------------
1 | ## Summary
2 |
3 |
4 |
5 | ## Related Links
6 |
7 |
8 |
9 | **Asana**:
10 |
11 | ## Checklist
12 |
13 | - [ ] Pasted Asana link.
14 | - [ ] Tagged labels.
15 |
16 | ## Prerequisite/Related Pull Requests
17 |
18 |
19 |
--------------------------------------------------------------------------------
/packages/eslint-config-custom/index.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | extends: [
3 | 'turbo',
4 | 'prettier',
5 | 'eslint:recommended',
6 | 'plugin:@typescript-eslint/recommended',
7 | ],
8 | parser: '@typescript-eslint/parser',
9 | plugins: ['@typescript-eslint'],
10 | root: true,
11 | rules: {
12 | '@typescript-eslint/no-explicit-any': 'off',
13 | },
14 | env: {
15 | node: true,
16 | es6: true,
17 | browser: true,
18 | jest: true,
19 | },
20 | };
21 |
--------------------------------------------------------------------------------
/adapters/web3modal-connector/README.md:
--------------------------------------------------------------------------------
1 | # web3Modal-connector
2 |
3 | The Blocto configuration settings for web3Modal
4 |
5 | For usage, please refer to [Integrate with Web3Modal](https://docs.blocto.app/blocto-sdk/javascript-sdk/evm-sdk/integrate-with-web3modal)
6 |
7 | ## Usage
8 |
9 | ```
10 | yarn add @blocto/web3modal-connector
11 | ```
12 |
13 | ## Development
14 |
15 | ### Install Dependencies
16 |
17 | ```bash
18 | yarn
19 | ```
20 |
21 | ### Build
22 |
23 | ```
24 | yarn build
25 | ```
26 |
--------------------------------------------------------------------------------
/.github/workflows/pr-labeler.yml:
--------------------------------------------------------------------------------
1 | # https://github.com/TimonVS/pr-labeler-action
2 |
3 | name: PR Labeler
4 | on:
5 | pull_request:
6 | types: [opened, ready_for_review]
7 |
8 | jobs:
9 | pr-labeler:
10 | runs-on: ubuntu-latest
11 | steps:
12 | - uses: TimonVS/pr-labeler-action@v3
13 | with:
14 | configuration-path: .github/pr-labeler.yml # optional, .github/pr-labeler.yml is the default value
15 | env:
16 | GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
17 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/addSelfRemovableHandler.ts:
--------------------------------------------------------------------------------
1 | interface RemovableEventHandler {
2 | (e: Event, callback: () => void): void;
3 | }
4 |
5 | export default (
6 | eventType: string,
7 | handler: RemovableEventHandler,
8 | target: any = window
9 | ): void => {
10 | function listener(e: Event) {
11 | const removeEventListener = () =>
12 | target.removeEventListener(eventType, listener);
13 | handler(e, removeEventListener);
14 | }
15 | target.addEventListener(eventType, listener);
16 | };
17 |
--------------------------------------------------------------------------------
/packages/eslint-config-custom/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "eslint-config-custom",
3 | "version": "0.0.0",
4 | "main": "index.js",
5 | "license": "MIT",
6 | "private": true,
7 | "dependencies": {
8 | "@typescript-eslint/eslint-plugin": "^5.59.9",
9 | "@typescript-eslint/parser": "^5.59.9",
10 | "eslint": "^8.42.0",
11 | "eslint-config-prettier": "^8.8.0",
12 | "eslint-config-turbo": "1.10.1",
13 | "typescript": "^5.1.3"
14 | },
15 | "publishConfig": {
16 | "access": "public"
17 | }
18 | }
19 |
--------------------------------------------------------------------------------
/packages/dappauth/src/ABIs/ERC1271.js:
--------------------------------------------------------------------------------
1 | export default [
2 | {
3 | constant: true,
4 | inputs: [
5 | {
6 | name: 'hash',
7 | type: 'bytes32',
8 | },
9 | {
10 | name: '_signature',
11 | type: 'bytes',
12 | },
13 | ],
14 | name: 'isValidSignature',
15 | outputs: [
16 | {
17 | name: 'magicValue',
18 | type: 'bytes4',
19 | },
20 | ],
21 | payable: false,
22 | stateMutability: 'view',
23 | type: 'function',
24 | },
25 | ];
26 |
--------------------------------------------------------------------------------
/.changeset/README.md:
--------------------------------------------------------------------------------
1 | # Changesets
2 |
3 | Hello and welcome! This folder has been automatically generated by `@changesets/cli`, a build tool that works
4 | with multi-package repos, or single-package repos to help you version and publish your code. You can
5 | find the full documentation for it [in our repository](https://github.com/changesets/changesets)
6 |
7 | We have a quick list of common questions to get you started engaging with this project in
8 | [our documentation](https://github.com/changesets/changesets/blob/main/docs/common-questions.md)
9 |
--------------------------------------------------------------------------------
/adapters/aptos-wallet-adapter-plugin/README.md:
--------------------------------------------------------------------------------
1 | # Blocto Plugin for Aptos Wallet Adapter
2 |
3 | A Blocto wallet plugin to be used with [Aptos Wallet Adapter](https://github.com/aptos-labs/aptos-wallet-adapter).
4 |
5 | ## Usage
6 |
7 | ```shell
8 | yarn add @blocto/aptos-wallet-adapter-plugin
9 | ```
10 |
11 | or
12 |
13 | ```shell
14 | npm i @blocto/aptos-wallet-adapter-plugin
15 | ```
16 |
17 | and follow the instructions [adapter for dapps](https://github.com/aptos-labs/aptos-wallet-adapter/tree/main/packages/wallet-adapter-react)
18 |
--------------------------------------------------------------------------------
/adapters/wagmi-connector/vite.config.ts:
--------------------------------------------------------------------------------
1 | // vite.config.ts
2 | import { resolve } from 'path';
3 | import { defineConfig } from 'vite';
4 | import dts from 'vite-plugin-dts';
5 | // https://vitejs.dev/guide/build.html#library-mode
6 | export default defineConfig({
7 | build: {
8 | lib: {
9 | entry: resolve(__dirname, 'src/index.ts'),
10 | name: 'index',
11 | fileName: 'index',
12 | },
13 | rollupOptions: {
14 | treeshake: {
15 | moduleSideEffects: false,
16 | },
17 | },
18 | },
19 | plugins: [dts()],
20 | });
21 |
--------------------------------------------------------------------------------
/adapters/connectkit-connector/vite.config.ts:
--------------------------------------------------------------------------------
1 | // vite.config.ts
2 | import { resolve } from 'path';
3 | import { defineConfig } from 'vite';
4 | import dts from 'vite-plugin-dts';
5 | // https://vitejs.dev/guide/build.html#library-mode
6 | export default defineConfig({
7 | build: {
8 | lib: {
9 | entry: resolve(__dirname, 'src/index.ts'),
10 | name: 'index',
11 | fileName: 'index',
12 | },
13 | rollupOptions: {
14 | treeshake: {
15 | moduleSideEffects: false,
16 | },
17 | },
18 | },
19 | plugins: [dts()],
20 | });
21 |
--------------------------------------------------------------------------------
/adapters/web3modal-connector/vite.config.ts:
--------------------------------------------------------------------------------
1 | // vite.config.ts
2 | import { resolve } from 'path';
3 | import { defineConfig } from 'vite';
4 | import dts from 'vite-plugin-dts';
5 | // https://vitejs.dev/guide/build.html#library-mode
6 | export default defineConfig({
7 | build: {
8 | lib: {
9 | entry: resolve(__dirname, 'src/index.ts'),
10 | name: 'index',
11 | fileName: 'index',
12 | },
13 | rollupOptions: {
14 | treeshake: {
15 | moduleSideEffects: false,
16 | },
17 | },
18 | },
19 | plugins: [dts()],
20 | });
21 |
--------------------------------------------------------------------------------
/packages/tsconfig/nextjs.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "https://json.schemastore.org/tsconfig",
3 | "display": "Next.js",
4 | "extends": "./base.json",
5 | "compilerOptions": {
6 | "allowJs": true,
7 | "declaration": false,
8 | "declarationMap": false,
9 | "incremental": true,
10 | "jsx": "preserve",
11 | "lib": ["dom", "dom.iterable", "esnext"],
12 | "module": "esnext",
13 | "noEmit": true,
14 | "resolveJsonModule": true,
15 | "strict": false,
16 | "target": "es5"
17 | },
18 | "include": ["src", "next-env.d.ts"],
19 | "exclude": ["node_modules"]
20 | }
21 |
--------------------------------------------------------------------------------
/packages/tsconfig/base.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "https://json.schemastore.org/tsconfig",
3 | "display": "Default",
4 | "compilerOptions": {
5 | "composite": false,
6 | "declaration": true,
7 | "declarationMap": true,
8 | "esModuleInterop": true,
9 | "forceConsistentCasingInFileNames": true,
10 | "inlineSources": false,
11 | "isolatedModules": true,
12 | "moduleResolution": "node",
13 | "noUnusedLocals": false,
14 | "noUnusedParameters": false,
15 | "preserveWatchOutput": true,
16 | "skipLibCheck": true,
17 | "strict": true
18 | },
19 | "exclude": ["node_modules"]
20 | }
21 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/main.ts:
--------------------------------------------------------------------------------
1 | import EthereumProvider from './providers/ethereum';
2 | import AptosProvider from './providers/aptos';
3 | import { BloctoSDKConfig } from './index.d';
4 |
5 | export default class BloctoSDK {
6 | ethereum?: EthereumProvider;
7 | aptos?: AptosProvider;
8 |
9 | constructor({ appId, ethereum, aptos }: BloctoSDKConfig) {
10 | if (ethereum) {
11 | this.ethereum = new EthereumProvider({
12 | ...ethereum,
13 | appId,
14 | });
15 | }
16 | if (aptos) {
17 | this.aptos = new AptosProvider({
18 | ...aptos,
19 | appId,
20 | });
21 | }
22 | }
23 | }
24 |
--------------------------------------------------------------------------------
/adapters/connectkit-connector/src/constants/index.tsx:
--------------------------------------------------------------------------------
1 | import { supportedConnectors } from 'connectkit';
2 | import { BloctoLogo } from '../components/BloctoLogo';
3 |
4 | type Spc = typeof supportedConnectors;
5 |
6 | const BLOCTO_CONFIG: Spc[number] = {
7 | id: 'blocto',
8 | name: 'Blocto',
9 | shortName: 'Blocto',
10 | logos: {
11 | default: ,
12 | },
13 | appUrls: {
14 | ios: 'https://apps.apple.com/app/blocto/id1481181682',
15 | android: 'https://play.google.com/store/apps/details?id=com.portto.blocto',
16 | },
17 | scannable: false,
18 | extensionIsInstalled: () => true,
19 | };
20 |
21 | export { BLOCTO_CONFIG };
22 |
--------------------------------------------------------------------------------
/packages/dappauth/CHANGELOG.md:
--------------------------------------------------------------------------------
1 | # @blocto/dappauth
2 |
3 | ## 2.2.2
4 |
5 | ### Patch Changes
6 |
7 | - 48fa07a: fix: handle existedSDK handle window not defined
8 |
9 | ## 2.2.2-beta.0
10 |
11 | ### Patch Changes
12 |
13 | - 48fa07a: fix: handle existedSDK handle window not defined
14 |
15 | ## 2.2.1
16 |
17 | ### Patch Changes
18 |
19 | - 9609a4e: fix: build config and test Switched from Mocha to Jest
20 |
21 | ## 2.2.1-beta.0
22 |
23 | ### Patch Changes
24 |
25 | - 9609a4e: fix: build config and test Switched from Mocha to Jest
26 |
27 | ## 2.2.0
28 |
29 | ### Minor Changes
30 |
31 | - 5ae0ff0: restructure @blocto/dappauth and using rollup bundle
32 |
--------------------------------------------------------------------------------
/adapters/wagmi-connector/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "target": "ES2020",
4 | "useDefineForClassFields": true,
5 | "module": "ESNext",
6 | "lib": ["ES2020", "DOM", "DOM.Iterable"],
7 | "skipLibCheck": true,
8 |
9 | /* Bundler mode */
10 | "moduleResolution": "bundler",
11 | "allowImportingTsExtensions": true,
12 | "resolveJsonModule": true,
13 | "isolatedModules": true,
14 | "noEmit": true,
15 |
16 | /* Linting */
17 | "strict": true,
18 | "noUnusedLocals": false,
19 | "noUnusedParameters": false,
20 | "noFallthroughCasesInSwitch": true
21 |
22 | },
23 | "include": ["src"]
24 | }
25 |
--------------------------------------------------------------------------------
/adapters/web3modal-connector/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "target": "ES2020",
4 | "useDefineForClassFields": true,
5 | "module": "ESNext",
6 | "lib": ["ES2020", "DOM", "DOM.Iterable"],
7 | "skipLibCheck": true,
8 |
9 | /* Bundler mode */
10 | "moduleResolution": "bundler",
11 | "allowImportingTsExtensions": true,
12 | "resolveJsonModule": true,
13 | "isolatedModules": true,
14 | "noEmit": true,
15 |
16 | /* Linting */
17 | "strict": true,
18 | "noUnusedLocals": false,
19 | "noUnusedParameters": false,
20 | "noFallthroughCasesInSwitch": true
21 |
22 | },
23 | "include": ["src"]
24 | }
25 |
--------------------------------------------------------------------------------
/adapters/aptos-wallet-adapter-plugin/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "https://json.schemastore.org/tsconfig",
3 | "display": "Default",
4 | "compilerOptions": {
5 | "composite": false,
6 | "declaration": true,
7 | "declarationMap": true,
8 | "esModuleInterop": true,
9 | "forceConsistentCasingInFileNames": true,
10 | "inlineSources": false,
11 | "isolatedModules": true,
12 | "moduleResolution": "node",
13 | "noUnusedLocals": false,
14 | "noUnusedParameters": false,
15 | "preserveWatchOutput": true,
16 | "skipLibCheck": true,
17 | "strict": true,
18 | "lib": ["ES2015", "DOM"]
19 | },
20 | "exclude": ["node_modules"]
21 | }
22 |
--------------------------------------------------------------------------------
/adapters/rainbowkit-connector/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "https://json.schemastore.org/tsconfig",
3 | "display": "Default",
4 | "compilerOptions": {
5 | "composite": false,
6 | "declaration": true,
7 | "declarationMap": true,
8 | "esModuleInterop": true,
9 | "forceConsistentCasingInFileNames": true,
10 | "inlineSources": false,
11 | "isolatedModules": true,
12 | "moduleResolution": "node",
13 | "noUnusedLocals": false,
14 | "noUnusedParameters": false,
15 | "preserveWatchOutput": true,
16 | "skipLibCheck": true,
17 | "strict": true,
18 | "lib": ["ES2020", "DOM"],
19 | "target": "ES2020"
20 | },
21 | "exclude": ["node_modules"]
22 | }
23 |
--------------------------------------------------------------------------------
/adapters/web3-react-connector/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "https://json.schemastore.org/tsconfig",
3 | "display": "Default",
4 | "compilerOptions": {
5 | "composite": false,
6 | "declaration": true,
7 | "declarationMap": true,
8 | "esModuleInterop": true,
9 | "forceConsistentCasingInFileNames": true,
10 | "inlineSources": false,
11 | "isolatedModules": true,
12 | "moduleResolution": "node",
13 | "noUnusedLocals": false,
14 | "noUnusedParameters": false,
15 | "preserveWatchOutput": true,
16 | "skipLibCheck": true,
17 | "strict": true,
18 | "lib": ["ES2020", "DOM"],
19 | "target": "ES2020"
20 | },
21 | "exclude": ["node_modules"]
22 | }
23 |
--------------------------------------------------------------------------------
/packages/dappauth/src/utils/index.js:
--------------------------------------------------------------------------------
1 | export function isHexString(value) {
2 | return (
3 | typeof value === 'string' &&
4 | value.length % 2 === 0 &&
5 | value.match(/^0x[0-9A-Fa-f]*$/)
6 | );
7 | }
8 |
9 | export function removeHexPrefix(value) {
10 | return value.slice(0, 2) === '0x' ? value.slice(2) : value;
11 | }
12 |
13 | export function mergeErrors(errEOA, errCA) {
14 | const msgEOA = errEOA
15 | ? `errored with: ${errEOA.toString()}`
16 | : 'returned false';
17 | return `Authorisation check failed and errored in 2 alternative flows. 'External Owned Account' check ${msgEOA} . 'Contract Account' check errored with: ${errCA.toString()}`;
18 | }
19 |
--------------------------------------------------------------------------------
/.github/workflows/create-asana-attachment.yaml:
--------------------------------------------------------------------------------
1 | on:
2 | pull_request:
3 | types: [opened, reopened]
4 |
5 | jobs:
6 | create-asana-attachment-job:
7 | runs-on: ubuntu-latest
8 | name: Create pull request attachments on Asana tasks
9 | steps:
10 | - name: Create pull request attachments
11 | uses: Asana/create-app-attachment-github-action@latest
12 | id: postAttachment
13 | with:
14 | asana-secret: ${{ secrets.ASANA_SECRET }}
15 | allowed-projects: |
16 | 1207145524635866
17 | 1207160002492686
18 | 1203400194794992
19 | - name: Log output status
20 | run: echo "Status is ${{ steps.postAttachment.outputs.status }}"
21 |
--------------------------------------------------------------------------------
/adapters/connectkit-connector/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "jsx": "react-jsx",
4 | "target": "ES2020",
5 | "useDefineForClassFields": true,
6 | "module": "ESNext",
7 | "lib": ["ES2020", "DOM", "DOM.Iterable"],
8 | "skipLibCheck": true,
9 |
10 | /* Bundler mode */
11 | "moduleResolution": "bundler",
12 | "allowImportingTsExtensions": true,
13 | "resolveJsonModule": true,
14 | "isolatedModules": true,
15 | "noEmit": true,
16 | "resolvePackageJsonExports": false,
17 |
18 | /* Linting */
19 | "strict": true,
20 | "noUnusedLocals": false,
21 | "noUnusedParameters": false,
22 | "noFallthroughCasesInSwitch": true
23 |
24 | },
25 | "include": ["src"]
26 | }
27 |
--------------------------------------------------------------------------------
/packages/dappauth/test/unit/provider-mock.js:
--------------------------------------------------------------------------------
1 | export default class MockProvider {
2 | constructor(mockContract) {
3 | this.mockContract = mockContract;
4 | }
5 |
6 | sendAsync(payload, callback) {
7 | const data = payload.params[0].data.substring(2);
8 | const methodCall = data.substring(0, 8);
9 | const methodParams = data.substring(8);
10 |
11 | try {
12 | const result = this.mockContract.run(methodCall, methodParams);
13 | const { jsonrpc, id } = payload;
14 | callback(null, {
15 | jsonrpc,
16 | id,
17 | result,
18 | });
19 | } catch (error) {
20 | callback(error);
21 | }
22 | }
23 |
24 | send(payload, callback) {
25 | this.sendAsync(payload, callback);
26 | }
27 | }
28 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/responseSessionGuard.ts:
--------------------------------------------------------------------------------
1 | import { removeItem } from './storage';
2 | import { KEY_SESSION } from '../constants';
3 |
4 | export interface ICustomError extends Error {
5 | error_code?: string;
6 | }
7 |
8 | export default async function responseSessionGuard(
9 | response: Response,
10 | key: KEY_SESSION,
11 | disconnectHandler?: () => void
12 | ): Promise {
13 | if (response.status === 403 || response.status === 401) {
14 | if (disconnectHandler) {
15 | disconnectHandler();
16 | }
17 | removeItem(key);
18 | }
19 | if (!response.ok) {
20 | const data = await response.json();
21 | const e: ICustomError = new Error(data?.message || 'unknown error');
22 | e.error_code = data?.error_code;
23 | throw e;
24 | }
25 | return response.json();
26 | }
27 |
--------------------------------------------------------------------------------
/adapters/connectkit-connector/README.md:
--------------------------------------------------------------------------------
1 | # connectkit-connector
2 |
3 | `integrateBloctoConfig` is used to integrate Blocto with ConnectKit.
4 |
5 | ## How to use
6 |
7 | ### Install package
8 |
9 | ```
10 | yarn add @blocto/connectkit-connector
11 | ```
12 |
13 | ### Example
14 |
15 | ```typescript
16 | import { getDefaultConfig, supportedConnectors } from 'connectkit';
17 | import { integrateBloctoConfig } from '@blocto/connectkit-connector';
18 |
19 | // ...
20 |
21 | let config = integrateBloctoConfig(
22 | createConfig(
23 | getDefaultConfig({
24 | appName: 'ConnectKit CRA demo',
25 | autoConnect: false,
26 | chains: [mainnet, polygon, optimism, arbitrum],
27 | walletConnectProjectId: process.env.REACT_APP_WALLETCONNECT_PROJECT_ID!,
28 | })
29 | ),
30 | supportedConnectors
31 | );
32 | ```
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/frame.ts:
--------------------------------------------------------------------------------
1 | const IFRAME_STYLE =
2 | 'width:100vw;height:100%;position:fixed;top:0;left:0;z-index:2147483646;border:none;box-sizing:border-box;color-scheme:light;inset:0px;display:block;pointer-events:auto;';
3 |
4 | export function createFrame(url: string): HTMLIFrameElement {
5 | const frame = document.createElement('iframe');
6 |
7 | frame.setAttribute('src', url);
8 | frame.setAttribute('style', IFRAME_STYLE);
9 |
10 | return frame;
11 | }
12 |
13 | export function attachFrame(frame: HTMLIFrameElement): void {
14 | document.body.appendChild(frame);
15 | }
16 |
17 | export function detatchFrame(frame: HTMLIFrameElement): void {
18 | const parentNode = frame && frame.parentNode;
19 | if (parentNode && parentNode.removeChild instanceof Function) {
20 | parentNode.removeChild(frame);
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/getEvmSupport.ts:
--------------------------------------------------------------------------------
1 | export interface EvmSupportMapping {
2 | [id: string]: {
3 | chain_id: number;
4 | name: string; // backend defined chain name: ethereum, bsc, …
5 | display_name: string; // human readable name: Ethereum, Polygon, …
6 | network_type: string; // chain network type: mainnet / testnet / devnet
7 | blocto_service_environment: string; // backend service env: prod / dev (may return staging in future)
8 | rpc_endpoint_domains: string[]; // rpc endpoints whitelist
9 | };
10 | }
11 |
12 | export async function getEvmSupport(): Promise {
13 | const { networks } = await fetch('https://api.blocto.app/networks/evm').then(
14 | (response) => response.json()
15 | );
16 | const evmSupportMap = networks.reduce(
17 | (a: any, v: any) => ({ ...a, [v.chain_id]: v }),
18 | {}
19 | );
20 | return evmSupportMap;
21 | }
22 |
--------------------------------------------------------------------------------
/adapters/aptos-wallet-adapter-plugin/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@blocto/aptos-wallet-adapter-plugin",
3 | "version": "0.2.11",
4 | "description": "Blocto Wallet plugin to use with Aptos Wallet Adapter",
5 | "author": "Blocto Wallet",
6 | "main": "./dist/index.js",
7 | "module": "./dist/index.mjs",
8 | "types": "./dist/index.d.ts",
9 | "license": "Apache-2.0",
10 | "exports": {
11 | "require": "./dist/index.js",
12 | "import": "./dist/index.mjs",
13 | "types": "./dist/index.d.ts"
14 | },
15 | "scripts": {
16 | "build": "tsup src/index.ts --format esm,cjs --dts",
17 | "test": "jest",
18 | "lint": "eslint src --ext .ts"
19 | },
20 | "dependencies": {
21 | "@aptos-labs/wallet-adapter-core": "^2.2.0",
22 | "@blocto/sdk": "^0.10.4",
23 | "aptos": "^1.9.1"
24 | },
25 | "devDependencies": {
26 | "@types/jest": "^29.5.2",
27 | "jest": "^29.3.1",
28 | "ts-jest": "^29.0.3",
29 | "tsup": "^6.5.0",
30 | "typescript": "^5.1.3"
31 | },
32 | "files": [
33 | "/dist"
34 | ]
35 | }
36 |
--------------------------------------------------------------------------------
/commitlint.config.js:
--------------------------------------------------------------------------------
1 | "module.exports = {extends: ['@commitlint/config-conventional']}";
2 |
3 | module.exports = {
4 | extends: ['@commitlint/config-conventional'],
5 | rules: {
6 | 'type-enum': [
7 | 2,
8 | 'always',
9 | [
10 | 'feat',
11 | 'fix',
12 | 'docs',
13 | 'style',
14 | 'refactor',
15 | 'perf',
16 | 'test',
17 | 'build',
18 | 'ci',
19 | 'chore',
20 | 'revert',
21 | ],
22 | ],
23 | 'subject-exclamation-mark': [2, 'never'],
24 | 'body-leading-blank': [1, 'always'],
25 | 'footer-leading-blank': [1, 'always'],
26 | 'header-max-length': [2, 'always', 72],
27 | 'scope-case': [2, 'always', 'lower-case'],
28 | 'subject-case': [
29 | 2,
30 | 'never',
31 | ['sentence-case', 'start-case', 'pascal-case', 'upper-case'],
32 | ],
33 | 'subject-empty': [2, 'never'],
34 | 'subject-full-stop': [2, 'never', '.'],
35 | 'type-case': [2, 'always', 'lower-case'],
36 | 'type-empty': [2, 'never'],
37 | },
38 | };
39 |
--------------------------------------------------------------------------------
/packages/dappauth/README.md:
--------------------------------------------------------------------------------
1 | # dappauth.js
2 | dappauth.js is a tool that can verify EOA signature and smart contract signature (ERC1271).
3 |
4 | ## Usage
5 | ```js
6 | const Web3 = require('web3');
7 | const DappAuth = require('@blocto/dappauth');
8 |
9 | const dappAuth = new DappAuth(new Web3.providers.HttpProvider('http://localhost:8545'));
10 |
11 | async function debug() {
12 | const challenge = 'foo';
13 | const signature =
14 | '0x49b5e108d0b7ce795125acbc08b331aad170e4afe8daa5b4b484d12b9253316c4ca0cb36cec51682bcc9a65d6cb28002bd4835845d8f1de21afafe1132eaed3d1b28171e3bf436f2301c0bd657501cafb2a9344efe042495adba2852858db559da2fb4280007de4d631b9357493a979c4024927aec580ef3a22603918eb3b3071b1c';
15 | const address = '0x86aa354fc865925f945b803ceae0b3f9d856b269';
16 |
17 | try {
18 | const isAuthorizedSigner = await dappAuth.isAuthorizedSigner(
19 | challenge,
20 | signature,
21 | address,
22 | );
23 |
24 | console.log(isAuthorizedSigner); // true
25 | } catch (e) {
26 | console.log(e);
27 | }
28 | }
29 | ```
30 |
31 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/storage/memoryStorage.ts:
--------------------------------------------------------------------------------
1 | declare global {
2 | interface Window {
3 | memoryStorage: MemoryStorage;
4 | }
5 | }
6 |
7 | export const isStorageSupported = () => {
8 | if (typeof window === 'undefined') {
9 | return false;
10 | }
11 | try {
12 | window.sessionStorage.setItem('local_storage_supported', '1');
13 | const result = window.sessionStorage.getItem('local_storage_supported');
14 | window.sessionStorage.removeItem('local_storage_supported');
15 | return result === '1';
16 | } catch (error) {
17 | return false;
18 | }
19 | };
20 |
21 | class MemoryStorage {
22 | storage = {};
23 |
24 | getItem(key: string): string | null {
25 | return (this as any)[key] || null;
26 | }
27 |
28 | setItem(key: string, value: unknown): void {
29 | (this.storage as any)[key] = value;
30 | }
31 |
32 | removeItem(key: string): void {
33 | delete (this.storage as any)[key];
34 | }
35 | }
36 |
37 | const memoryStorage =
38 | typeof window !== 'undefined' ? window.memoryStorage : new MemoryStorage();
39 |
40 | export default memoryStorage;
41 | export { MemoryStorage };
42 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2023 portto (Blocto)
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/.github/workflows/test.yml:
--------------------------------------------------------------------------------
1 | name: Build and Test
2 |
3 | on:
4 | pull_request:
5 | types: [opened, synchronize, ready_for_review]
6 |
7 | jobs:
8 | build:
9 | name: Build and Test
10 | timeout-minutes: 15
11 | runs-on: ubuntu-latest
12 |
13 | steps:
14 | - name: Checkout Repo
15 | uses: actions/checkout@v3
16 |
17 | - name: Cache turbo build setup
18 | uses: actions/cache@v3
19 | with:
20 | path: .turbo
21 | key: ${{ runner.os }}-turbo-${{ github.sha }}
22 | restore-keys: |
23 | ${{ runner.os }}-turbo-
24 |
25 | - name: Setup Node.js 18
26 | uses: actions/setup-node@v3
27 | with:
28 | node-version: 18
29 |
30 | - name: Install dependencies
31 | run: yarn
32 |
33 | - name: Lint, Test, and Try to Build
34 | id: testWithCache
35 | run: yarn turbo lint test build --cache-dir=.turbo
36 | continue-on-error: true
37 |
38 | - name: Try to do again without cache
39 | if: steps.testWithCache.outcome == 'failure'
40 | run: yarn turbo lint test build --force --cache-dir=.turbo
41 |
--------------------------------------------------------------------------------
/adapters/wagmi-connector/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@blocto/wagmi-connector",
3 | "description": "Blocto wallet connector extend from wagmi Connector",
4 | "author": "Calvin Chang",
5 | "version": "2.0.5",
6 | "type": "module",
7 | "main": "./dist/index.umd.cjs",
8 | "module": "./dist/index.js",
9 | "types": "./dist/index.d.ts",
10 | "license": "Apache-2.0",
11 | "exports": {
12 | "import": "./dist/index.js",
13 | "require": "./dist/index.umd.cjs"
14 | },
15 | "scripts": {
16 | "build": "tsc && vite build",
17 | "lint": "eslint src --ext .ts",
18 | "test": "vitest run --coverage"
19 | },
20 | "devDependencies": {
21 | "@types/node": "^20.2.5",
22 | "@vitest/coverage-v8": "^1.2.0",
23 | "@wagmi/chains": "^1.8.0",
24 | "@wagmi/core": "^2.2.0",
25 | "typescript": "^5.1.3",
26 | "viem": "^2.x",
27 | "vite": "^4.3.9",
28 | "vite-plugin-dts": "^2.3.0",
29 | "vitest": "^1.2.0"
30 | },
31 | "files": [
32 | "/dist"
33 | ],
34 | "dependencies": {
35 | "@blocto/sdk": "^0.10.4"
36 | },
37 | "peerDependencies": {
38 | "@wagmi/core": "^2.2.0",
39 | "viem": "2.x"
40 | }
41 | }
42 |
--------------------------------------------------------------------------------
/packages/dappauth/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2018 Dapper Labs, Inc.
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/index.d.ts:
--------------------------------------------------------------------------------
1 | // eslint-disable-next-line spaced-comment
2 | ///
3 |
4 | import { BaseConfig } from './constants';
5 | import {
6 | EthereumProviderConfig,
7 | EthereumProviderInterface,
8 | } from './providers/types/ethereum.d';
9 | import {
10 | AptosProviderConfig,
11 | AptosProviderInterface,
12 | } from './providers/types/aptos.d';
13 | import * as AptosTypes from './providers/types/aptos.d';
14 | import * as EthereumTypes from './providers/types/ethereum.d';
15 |
16 | export * from './providers/types/blocto.d';
17 | export {
18 | BaseConfig,
19 | EthereumProviderConfig,
20 | EthereumProviderInterface,
21 | AptosProviderConfig,
22 | AptosProviderInterface,
23 | // Keep above types for backward compatibility
24 | AptosTypes,
25 | EthereumTypes,
26 | };
27 | export declare interface BloctoSDKConfig extends BaseConfig {
28 | ethereum?: EthereumProviderConfig;
29 | aptos?: AptosProviderConfig;
30 | }
31 | declare class BloctoSDK {
32 | ethereum?: EthereumProviderInterface;
33 | aptos?: AptosProviderInterface;
34 | constructor(config: BloctoSDKConfig);
35 | }
36 |
37 | export default BloctoSDK;
38 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/rollup.config.dev.js:
--------------------------------------------------------------------------------
1 | import commonjs from 'rollup-plugin-commonjs';
2 | import resolve from 'rollup-plugin-node-resolve';
3 | import typescript from 'rollup-plugin-typescript';
4 | import json from 'rollup-plugin-json';
5 | import polyfills from 'rollup-plugin-node-polyfills';
6 | import alias from '@rollup/plugin-alias';
7 | import { babel } from '@rollup/plugin-babel';
8 |
9 | const config = {
10 | input: 'src/main.ts',
11 | output: {
12 | file: 'dev/bundle.js',
13 | format: 'umd',
14 | name: 'BloctoSDK',
15 | },
16 | plugins: [
17 | alias({
18 | entries: {
19 | 'readable-stream': 'stream',
20 | },
21 | }),
22 | resolve({
23 | preferBuiltins: true,
24 | browser: true,
25 | }),
26 | typescript(),
27 | commonjs({
28 | include: 'node_modules/**',
29 | namedExports: {
30 | 'node_modules/js-sha3/src/sha3.js': ['keccak_256'],
31 | },
32 | }),
33 | json(),
34 | babel({
35 | babelHelpers: 'inline',
36 | exclude: 'node_modules/**',
37 | presets: [['@babel/preset-env']],
38 | }),
39 | polyfills(),
40 | ],
41 | };
42 |
43 | export default config;
44 |
--------------------------------------------------------------------------------
/adapters/connectkit-connector/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@blocto/connectkit-connector",
3 | "description": "blocto wallet connector to use with connectkit",
4 | "author": "Blocto Wallet",
5 | "version": "0.2.3",
6 | "type": "module",
7 | "main": "./dist/index.umd.cjs",
8 | "module": "./dist/index.js",
9 | "types": "./dist/index.d.ts",
10 | "license": "Apache-2.0",
11 | "exports": {
12 | "import": "./dist/index.js",
13 | "require": "./dist/index.umd.cjs"
14 | },
15 | "scripts": {
16 | "build": "tsc && vite build"
17 | },
18 | "devDependencies": {
19 | "@types/node": "^20.2.5",
20 | "@types/react": "^18",
21 | "connectkit": "^1",
22 | "react": "^18",
23 | "typescript": "^5.1.3",
24 | "viem": "^1.2.9",
25 | "vite": "^4.3.9",
26 | "vite-plugin-dts": "^2.3.0",
27 | "wagmi": "^1"
28 | },
29 | "files": [
30 | "/dist"
31 | ],
32 | "dependencies": {
33 | "@blocto/wagmi-connector": "npm:@blocto/wagmi-connector@^1.3.1"
34 | },
35 | "peerDependencies": {
36 | "connectkit": "^1.5.3",
37 | "react": "^18",
38 | "viem": "^1",
39 | "wagmi": "^1"
40 | },
41 | "resolutions": {
42 | "viem": "1.3.0"
43 | }
44 | }
45 |
--------------------------------------------------------------------------------
/adapters/web3modal-connector/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@blocto/web3modal-connector",
3 | "description": "Blocto connector extend from wagmi Connector to use with web3modal",
4 | "author": "Blocto Wallet",
5 | "version": "0.1.4",
6 | "type": "module",
7 | "main": "./dist/index.umd.cjs",
8 | "module": "./dist/index.js",
9 | "types": "./dist/index.d.ts",
10 | "license": "Apache-2.0",
11 | "keywords": [
12 | "web3modal",
13 | "wagmi",
14 | "javascript",
15 | "typescript",
16 | "web3",
17 | "context",
18 | "frontend",
19 | "dapp",
20 | "blocto",
21 | "web3-react",
22 | "blocto-sdk",
23 | "eth",
24 | "ethereum"
25 | ],
26 | "exports": {
27 | "import": "./dist/index.js",
28 | "require": "./dist/index.umd.cjs"
29 | },
30 | "scripts": {
31 | "build": "tsc && vite build",
32 | "lint": "eslint src --ext .ts"
33 | },
34 | "devDependencies": {
35 | "@types/node": "^20.2.5",
36 | "typescript": "^5.1.3",
37 | "vite": "^4.3.9",
38 | "vite-plugin-dts": "^2.3.0"
39 | },
40 | "files": [
41 | "/dist"
42 | ],
43 | "dependencies": {
44 | "@blocto/wagmi-connector": "npm:@blocto/wagmi-connector@^1.3.1"
45 | }
46 | }
47 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "blocto-sdk",
3 | "version": "0.1.0",
4 | "private": true,
5 | "packageManager": "yarn@1.19.0",
6 | "scripts": {
7 | "build": "turbo run build",
8 | "dev": "turbo run dev",
9 | "lint": "turbo run lint",
10 | "format": "prettier --write \"**/*.{ts,tsx,md}\"",
11 | "test": "turbo run test",
12 | "beta-ver": "changeset pre enter beta && changeset version && changeset pre exit",
13 | "ver": "changeset version",
14 | "beta-release": "turbo run test && changeset pre enter beta && changeset version && turbo run build && changeset publish && changeset pre exit",
15 | "release": "turbo run build && changeset publish && git push origin --tags",
16 | "postinstall": "husky install",
17 | "prepare": "husky install"
18 | },
19 | "devDependencies": {
20 | "@changesets/cli": "^2.26.1",
21 | "@commitlint/cli": "^17.6.5",
22 | "@commitlint/config-angular": "^17.6.5",
23 | "@commitlint/config-conventional": "^17.6.5",
24 | "eslint": "^8.42.0",
25 | "eslint-config-custom": "*",
26 | "husky": "^8.0.0",
27 | "prettier": "^2.8.8",
28 | "turbo": "^1.10.1"
29 | },
30 | "workspaces": [
31 | "packages/*",
32 | "adapters/*"
33 | ]
34 | }
35 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/isValidTransaction.ts:
--------------------------------------------------------------------------------
1 | import { ERROR_MESSAGE } from './constant';
2 | import { isHexString } from './utf8toHex';
3 |
4 | export const isValidTransaction = (
5 | transaction: Record
6 | ): { isValid: boolean; invalidMsg?: string } => {
7 | if (
8 | !transaction ||
9 | typeof transaction !== 'object' ||
10 | !('from' in transaction)
11 | ) {
12 | return { isValid: false, invalidMsg: ERROR_MESSAGE.INVALID_TRANSACTION };
13 | }
14 | if (transaction.value && !isHexString(transaction.value)) {
15 | return {
16 | isValid: false,
17 | invalidMsg: ERROR_MESSAGE.INVALID_TRANSACTION_VALUE,
18 | };
19 | }
20 |
21 | return { isValid: true };
22 | };
23 |
24 | export const isValidTransactions = (
25 | transactions: Record[]
26 | ): { isValid: boolean; invalidMsg?: string } => {
27 | if (!Array.isArray(transactions)) {
28 | return { isValid: false, invalidMsg: ERROR_MESSAGE.INVALID_TRANSACTIONS };
29 | }
30 |
31 | for (let i = 0; i < transactions.length; i++) {
32 | const { isValid, invalidMsg } = isValidTransaction(transactions[i]);
33 |
34 | if (!isValid) {
35 | return { isValid, invalidMsg };
36 | }
37 | }
38 |
39 | return { isValid: true };
40 | };
41 |
--------------------------------------------------------------------------------
/adapters/rainbowkit-connector/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@blocto/rainbowkit-connector",
3 | "version": "2.0.3",
4 | "description": "blocto wallet connector to use rainbowkit",
5 | "author": "Blocto Wallet",
6 | "main": "./dist/index.js",
7 | "module": "./dist/index.mjs",
8 | "types": "./dist/index.d.ts",
9 | "license": "Apache-2.0",
10 | "keywords": [
11 | "rainbowkit",
12 | "react",
13 | "wagmi",
14 | "javascript",
15 | "typescript",
16 | "web3",
17 | "context",
18 | "frontend",
19 | "dapp",
20 | "blocto",
21 | "web3-react",
22 | "blocto-sdk",
23 | "eth",
24 | "ethereum"
25 | ],
26 | "exports": {
27 | "require": "./dist/index.js",
28 | "import": "./dist/index.mjs",
29 | "types": "./dist/index.d.ts"
30 | },
31 | "scripts": {
32 | "build": "tsup src/index.ts --format esm,cjs --dts",
33 | "lint": "eslint src --ext .ts"
34 | },
35 | "dependencies": {
36 | "@rainbow-me/rainbowkit": "^2.0.5",
37 | "@blocto/wagmi-connector": "^2.0.5"
38 | },
39 | "devDependencies": {
40 | "@types/jest": "^29.5.2",
41 | "jest": "^29.3.1",
42 | "ts-jest": "^29.0.3",
43 | "tsup": "^6.5.0",
44 | "typescript": "^5.1.3"
45 | },
46 | "files": [
47 | "/dist"
48 | ]
49 | }
50 |
--------------------------------------------------------------------------------
/adapters/web3-react-connector/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@blocto/web3-react-connector",
3 | "version": "1.0.9",
4 | "description": "A Blocto SDK connector for web3-react",
5 | "main": "./dist/index.js",
6 | "module": "./dist/index.mjs",
7 | "types": "./dist/index.d.ts",
8 | "license": "MIT",
9 | "files": [
10 | "/dist"
11 | ],
12 | "publishConfig": {
13 | "access": "public"
14 | },
15 | "exports": {
16 | "require": "./dist/index.js",
17 | "import": "./dist/index.mjs",
18 | "types": "./dist/index.d.ts"
19 | },
20 | "scripts": {
21 | "build": "tsup src/index.ts --format esm,cjs --dts",
22 | "lint": "eslint src --ext .ts"
23 | },
24 | "keywords": [
25 | "react",
26 | "react-hooks",
27 | "hooks",
28 | "javascript",
29 | "typescript",
30 | "web3",
31 | "context",
32 | "frontend",
33 | "dapp",
34 | "blocto",
35 | "web3-react",
36 | "blocto-sdk",
37 | "eth",
38 | "ethereum"
39 | ],
40 | "author": "Chiaki.C",
41 | "repository": {
42 | "type": "git",
43 | "url": "git://github.com/portto/blocto-sdk.git"
44 | },
45 | "dependencies": {
46 | "@blocto/sdk": "^0.10.4",
47 | "@web3-react/types": "^8.1.2-beta.0"
48 | },
49 | "devDependencies": {
50 | "tsup": "^7.2.0",
51 | "typescript": "^5.1.3"
52 | }
53 | }
54 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/.gitignore:
--------------------------------------------------------------------------------
1 | # Logs
2 | logs
3 | *.log
4 | npm-debug.log*
5 | yarn-debug.log*
6 | yarn-error.log*
7 |
8 | # Runtime data
9 | pids
10 | *.pid
11 | *.seed
12 | *.pid.lock
13 |
14 | # Directory for instrumented libs generated by jscoverage/JSCover
15 | lib-cov
16 |
17 | # Coverage directory used by tools like istanbul
18 | coverage
19 |
20 | # nyc test coverage
21 | .nyc_output
22 |
23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
24 | .grunt
25 |
26 | # Bower dependency directory (https://bower.io/)
27 | bower_components
28 |
29 | # node-waf configuration
30 | .lock-wscript
31 |
32 | # Compiled binary addons (http://nodejs.org/api/addons.html)
33 | dist
34 | es
35 |
36 | # dev build
37 | dev
38 |
39 | # Dependency directories
40 | node_modules/
41 | jspm_packages/
42 |
43 | # Typescript v1 declaration files
44 | typings/
45 |
46 | # Optional npm cache directory
47 | .npm
48 |
49 | # Optional eslint cache
50 | .eslintcache
51 |
52 | # Optional REPL history
53 | .node_repl_history
54 |
55 | # Output of 'npm pack'
56 | *.tgz
57 |
58 | # dotenv environment variables file
59 | .env
60 |
61 | # gatsby files
62 | .cache/
63 | public
64 |
65 | # Mac files
66 | .DS_Store
67 |
68 | # Yarn
69 | yarn-error.log
70 | .pnp/
71 | .pnp.js
72 | # Yarn Integrity file
73 | .yarn-integrity
74 |
75 | # IntelijIDEA
76 | .idea
77 |
78 | # cert keys
79 | dev-cert/*.pem
80 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Logs
2 | logs
3 | *.log
4 | npm-debug.log*
5 | yarn-debug.log*
6 | yarn-error.log*
7 |
8 | # Runtime data
9 | pids
10 | *.pid
11 | *.seed
12 | *.pid.lock
13 |
14 | # Directory for instrumented libs generated by jscoverage/JSCover
15 | lib-cov
16 |
17 | # Coverage directory used by tools like istanbul
18 | coverage
19 |
20 | # nyc test coverage
21 | .nyc_output
22 |
23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
24 | .grunt
25 |
26 | # Bower dependency directory (https://bower.io/)
27 | bower_components
28 |
29 | # node-waf configuration
30 | .lock-wscript
31 |
32 | # Compiled binary addons (http://nodejs.org/api/addons.html)
33 | dist
34 | es
35 |
36 | # dev build
37 | dev
38 |
39 | # dependencies
40 | node_modules
41 | .pnp
42 | .pnp.js
43 |
44 | # Typescript v1 declaration files
45 | typings/
46 |
47 | # Optional npm cache directory
48 | .npm
49 |
50 | # Optional eslint cache
51 | .eslintcache
52 |
53 | # Optional REPL history
54 | .node_repl_history
55 |
56 | # Output of 'npm pack'
57 | *.tgz
58 |
59 | # dotenv environment variables file
60 | .env
61 |
62 | # Mac files
63 | .DS_Store
64 |
65 | # Yarn
66 | yarn-error.log
67 | .pnp/
68 | .pnp.js
69 | # Yarn Integrity file
70 | .yarn-integrity
71 |
72 | # IntelijIDEA
73 | .idea
74 |
75 | # cert keys
76 | dev-cert/*.pem
77 |
78 | # turbo repo
79 | .turbo
80 |
81 | # vercel
82 | .vercel
83 |
84 | # debug
85 | npm-debug.log*
86 | yarn-debug.log*
87 | yarn-error.log*
88 |
89 | # next.js
90 | .next
91 | /out/
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/providers/blocto.ts:
--------------------------------------------------------------------------------
1 | // The root class for all providers
2 |
3 | import { EIP1193Provider, RequestArguments } from 'eip1193-provider';
4 | import { EIP1193_EVENTS } from '../constants';
5 | import { DEFAULT_APP_ID } from '../constants';
6 |
7 | class BloctoProvider implements EIP1193Provider {
8 | isBlocto = true;
9 |
10 | isConnecting = false;
11 | appId: string;
12 |
13 | // eslint-disable-next-line @typescript-eslint/no-unused-vars
14 | eventListeners: { [key: string]: Array<(arg: any) => void> } = {};
15 |
16 | constructor() {
17 | // init event listeners
18 | EIP1193_EVENTS.forEach((event) => {
19 | this.eventListeners[event] = [];
20 | });
21 | this.appId = DEFAULT_APP_ID;
22 | }
23 |
24 | // implement by children
25 | // eslint-disable-next-line
26 | async request(payload: RequestArguments | Array) {}
27 |
28 | on(event: string, listener: (arg: any) => void): void {
29 | if (!EIP1193_EVENTS.includes(event)) return;
30 |
31 | this.eventListeners[event].push(listener);
32 | }
33 |
34 | // @todo: implement it
35 | // eslint-disable-next-line
36 | once() {}
37 |
38 | removeListener(event: string, listener: (arg: any) => void): void {
39 | const listeners = this.eventListeners[event];
40 | const index = listeners.findIndex((item) => item === listener);
41 | if (index !== -1) {
42 | this.eventListeners[event].splice(index, 1);
43 | }
44 | }
45 |
46 | // alias removeListener
47 | off = this.removeListener;
48 | }
49 |
50 | export default BloctoProvider;
51 |
--------------------------------------------------------------------------------
/adapters/wagmi-connector/README.md:
--------------------------------------------------------------------------------
1 | # wagmi-connector
2 |
3 | The `BloctoConnector` extends from the [wagmi](https://wagmi.sh/) `Connector`, allowing Blocto to be used with the most popular **Connect Wallet kits** (such as Web3Modal, ConnectKit, etc.) as solutions.
4 |
5 | For usage, please refer to below links:
6 | - [Blocto SDK Integrate with Web3Modal](https://docs.blocto.app/blocto-sdk/javascript-sdk/evm-sdk/integrate-with-web3modal)
7 | - [Blocto SDK Integrate with RainbowKit](https://docs.blocto.app/blocto-sdk/javascript-sdk/evm-sdk/integrate-with-rainbowkit)
8 | - [Blocto SDK Integrate with thirdweb](https://docs.blocto.app/blocto-sdk/javascript-sdk/evm-sdk/integrate-with-thirdweb)
9 |
10 | ## Migration Guide
11 | If you are coming from an earlier version of `@blocto/wagmi-connector`,
12 | you can see breaking changes in the list below.
13 |
14 | ### 1.2.0 Breaking Changes
15 | 1. Removed `BloctoWeb3modalConfig` exported; instead, we use `@blocto/web3modal-connector` to replace support for **Web3Modal**
16 |
17 | ```bash
18 | npm uninstall @blocto/wagmi-connector
19 | npm install @blocto/web3modal-connector
20 | ```
21 |
22 | 2. Removed `chainId` & `rpc` parameters from `BloctoConnector` constructor.
23 |
24 | ```diff
25 | - new BloctoConnector({ chains, options: { appId, chainId, rpc } })
26 | + new BloctoConnector({ chains, options: { appId } })
27 | ```
28 |
29 | ## Usage
30 |
31 | ```
32 | yarn add @blocto/wagmi-connector
33 | ```
34 |
35 | ## Development
36 |
37 | ### Install Dependencies
38 |
39 | ```bash
40 | yarn
41 | ```
42 |
43 | ### Build
44 |
45 | ```
46 | yarn build
47 | ```
48 |
--------------------------------------------------------------------------------
/adapters/aptos-wallet-adapter-plugin/src/__tests__/index.test.ts:
--------------------------------------------------------------------------------
1 | import { BloctoWallet } from '../index';
2 |
3 | /*
4 | Basic tests to make sure basic variables and functions are defined on the wallet
5 |
6 | */
7 | describe('BloctoWallet', () => {
8 | const wallet = new BloctoWallet({
9 | bloctoAppId: '6f6b97c5-d97b-4799-8ad7-d7e8426d3369',
10 | });
11 |
12 | test('defines name', () => {
13 | expect(typeof wallet.name).toBe('string');
14 | });
15 |
16 | test('defines url', () => {
17 | expect(typeof wallet.url).toBe('string');
18 | });
19 |
20 | test('defines icon', () => {
21 | expect(typeof wallet.icon).toBe('string');
22 | });
23 |
24 | test('defines connect()', () => {
25 | expect(typeof wallet.connect).toBe('function');
26 | });
27 |
28 | test('defines account()', () => {
29 | expect(typeof wallet.account).toBe('function');
30 | });
31 |
32 | test('defines disconnect()', () => {
33 | expect(typeof wallet.disconnect).toBe('function');
34 | });
35 |
36 | test('defines signAndSubmitTransaction()', () => {
37 | expect(typeof wallet.signAndSubmitTransaction).toBe('function');
38 | });
39 |
40 | test('defines signMessage()', () => {
41 | expect(typeof wallet.signMessage).toBe('function');
42 | });
43 |
44 | test('defines onNetworkChange()', () => {
45 | expect(typeof wallet.onNetworkChange).toBe('function');
46 | });
47 |
48 | test('defines onAccountChange()', () => {
49 | expect(typeof wallet.onAccountChange).toBe('function');
50 | });
51 |
52 | test('defines network()', () => {
53 | expect(typeof wallet.network).toBe('function');
54 | });
55 | });
56 |
--------------------------------------------------------------------------------
/packages/dappauth/test/unit/jest.config.cjs:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | preset: 'ts-jest',
3 | globals: {
4 | 'ts-jest': {
5 | tsconfig: './test/tsconfig.json',
6 | },
7 | },
8 | rootDir: '../..',
9 | transform: {
10 | '^.+\\.(ts|tsx)?$': 'ts-jest',
11 | '^.+\\.(js|jsx)$': 'babel-jest',
12 | },
13 | moduleNameMapper: {
14 | '^(\\.{1,2}/.*)\\.js$': '$1',
15 | },
16 | verbose: false,
17 | /**
18 | * restoreMocks [boolean]
19 | *
20 | * Default: false
21 | *
22 | * Automatically restore mock state between every test.
23 | * Equivalent to calling jest.restoreAllMocks() between each test.
24 | * This will lead to any mocks having their fake implementations removed
25 | * and restores their initial implementation.
26 | */
27 | restoreMocks: true,
28 | /**
29 | * resetModules [boolean]
30 | *
31 | * Default: false
32 | *
33 | * By default, each test file gets its own independent module registry.
34 | * Enabling resetModules goes a step further and resets the module registry before running each individual test.
35 | * This is useful to isolate modules for every test so that local module state doesn't conflict between tests.
36 | * This can be done programmatically using jest.resetModules().
37 | */
38 | resetModules: true,
39 | testMatch: ['/test/unit/**/*.(spec|test).(js|ts)'],
40 | coverageDirectory: '../../.coverage/unit',
41 | collectCoverageFrom: ['src/**'],
42 | collectCoverage: true,
43 | coverageReporters: [
44 | [
45 | 'json',
46 | {
47 | file: 'dappauth-unit-coverage.json',
48 | },
49 | ],
50 | ],
51 | };
52 |
--------------------------------------------------------------------------------
/.github/workflows/publish.yml:
--------------------------------------------------------------------------------
1 | name: Publish to npm
2 |
3 | on:
4 | push:
5 | branches:
6 | - main
7 |
8 | concurrency: ${{ github.workflow }}-${{ github.ref }}
9 |
10 | jobs:
11 | release:
12 | name: Publish to npm
13 | runs-on: ubuntu-latest
14 | steps:
15 | - name: Checkout Repo
16 | uses: actions/checkout@v3
17 |
18 | - name: Setup Node.js 18
19 | uses: actions/setup-node@v3
20 | with:
21 | node-version: 18
22 |
23 | - name: Install Dependencies
24 | run: yarn
25 |
26 | - name: Attempting to publish packages to npm
27 | uses: changesets/action@v1
28 | with:
29 | publish: yarn release
30 | env:
31 | GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
32 | NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }}
33 | NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
34 |
35 | - name: Set version from publishedPackages
36 | id: setVersion
37 | if: steps.changesets.outputs.published == 'true'
38 | env:
39 | PACKAGE_VERSION: ${{ steps.changesets.outputs.publishedPackages }}
40 | run: echo "PACKAGE_VERSION=$(echo $PACKAGE_VERSION | jq -r '.[0].version')" >> $GITHUB_OUTPUT
41 |
42 | - name: Trigger example repo github action
43 | if: steps.changesets.outputs.published == 'true'
44 | run: |
45 | curl -H "Authorization: Bearer ${{ secrets.DEV_GITHUB_TOKEN }}" \
46 | -H 'Accept: application/vnd.github.everest-preview+json' \
47 | "https://api.github.com/repos/blocto/blocto-sdk-examples/dispatches" \
48 | -d '{"event_type": "update-package", "client_payload": {"version": "${{ steps.setVersion.outputs.PACKAGE_VERSION }}" }}'
49 |
--------------------------------------------------------------------------------
/adapters/rainbowkit-connector/README.md:
--------------------------------------------------------------------------------
1 | # rainbowkit-connector
2 |
3 | [Blocto SDK](https://docs.blocto.app/blocto-sdk/javascript-sdk) connector for [RainbowKit](https://www.rainbowkit.com/) React library.
4 |
5 | ## How to use
6 |
7 | ### Install package
8 |
9 | `yarn add @blocto/rainbowkit-connector`
10 |
11 | ### Use it in your code
12 |
13 | ```TSX
14 | import {
15 | getDefaultWallets,
16 | connectorsForWallets,
17 | RainbowKitProvider,
18 | ConnectButton
19 | } from "@rainbow-me/rainbowkit";
20 | import { configureChains, createConfig, WagmiConfig } from "wagmi";
21 | import { polygon, optimism, arbitrum, bsc, mainnet } from "wagmi/chains";
22 | import { publicProvider } from 'wagmi/providers/public';
23 | import { alchemyProvider } from 'wagmi/providers/alchemy';
24 | import { bloctoWallet } from '@blocto/rainbowkit-connector';
25 |
26 | const { chains, publicClient, webSocketPublicClient } = configureChains(
27 | [polygon, optimism, arbitrum, bsc, mainnet],
28 | [alchemyProvider({ apiKey: process.env.ALCHEMY_ID || "" }), publicProvider()]
29 | );
30 |
31 | const { wallets } = getDefaultWallets({
32 | appName: "My RainbowKit App",
33 | projectId: "YOUR_PROJECT_ID",
34 | chains
35 | });
36 |
37 | const connectors = connectorsForWallets([
38 | ...wallets,
39 | {
40 | groupName: "Other",
41 | wallets: [
42 | bloctoWallet({ chains }), // add BloctoWallet
43 | ]
44 | }
45 | ]);
46 |
47 | const wagmiConfig = createConfig({
48 | autoConnect: true,
49 | connectors,
50 | publicClient,
51 | webSocketPublicClient,
52 | });
53 |
54 | export const App = () => {
55 | return (
56 |
57 |
58 |
59 |
60 |
61 | );
62 | };
63 |
64 | ```
65 |
--------------------------------------------------------------------------------
/.github/workflows/release-branch.yml:
--------------------------------------------------------------------------------
1 | name: Create version PR from release branch
2 |
3 | on:
4 | push:
5 | branches:
6 | - 'release**'
7 | - 'hotfix**'
8 |
9 | concurrency: ${{ github.workflow }}-${{ github.ref }}
10 |
11 | jobs:
12 | release:
13 | name: Create version PR from release branch
14 | runs-on: ubuntu-latest
15 | permissions:
16 | contents: write
17 | pull-requests: write
18 | steps:
19 | - name: Checkout Repo
20 | uses: actions/checkout@v3
21 |
22 | - name: Cache turbo build setup
23 | uses: actions/cache@v3
24 | with:
25 | path: .turbo
26 | key: ${{ runner.os }}-turbo-${{ github.sha }}
27 | restore-keys: |
28 | ${{ runner.os }}-turbo-
29 |
30 | - name: Setup Node.js 18
31 | uses: actions/setup-node@v3
32 | with:
33 | node-version: 18
34 |
35 | - name: Install Dependencies
36 | run: yarn
37 |
38 | - name: Bump Changeset Version
39 | run: yarn ver
40 |
41 | - name: Commit Changes
42 | uses: stefanzweifel/git-auto-commit-action@v4
43 | with:
44 | commit_message: 'ci: bump package version'
45 | commit_options: '--no-verify'
46 |
47 | - name: Create Release Pull Requests
48 | env:
49 | GITHUB_TOKEN: ${{ secrets.DEV_GITHUB_TOKEN }}
50 | MAIN_TITLE: 'Release: Merge to publish new version to npm'
51 | MAIN_BODY: 'Please review and merge this Pull Request. **Caution: Merge to main branch will trigger publish to npm.**'
52 | DEV_TITLE: 'Release: Merge back new version to develop'
53 | DEV_BODY: 'Merge new prod release back to develop'
54 | run: |
55 | gh pr create --title "$MAIN_TITLE" --body "$MAIN_BODY" --base main && gh pr create --title "$DEV_TITLE" --body "$DEV_BODY" --base develop
56 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/providers/types/aptos.d.ts:
--------------------------------------------------------------------------------
1 | import { BaseConfig, WalletAdapterNetwork } from '../../constants';
2 | import BloctoProviderInterface from './blocto.d';
3 |
4 | export { WalletAdapterNetwork } from '../../constants';
5 |
6 | export interface SignMessagePayload {
7 | address?: boolean;
8 | application?: boolean;
9 | chainId?: boolean;
10 | message: string;
11 | nonce: string;
12 | }
13 | export interface SignMessageResponse {
14 | address?: string;
15 | application?: string;
16 | chainId?: number;
17 | fullMessage: string;
18 | message: string;
19 | nonce: string;
20 | prefix: 'APTOS';
21 | signature: string | string[];
22 | bitmap?: Uint8Array;
23 | }
24 |
25 | export declare interface AptosProviderConfig extends BaseConfig {
26 | // @todo: support different network
27 | chainId: number;
28 | server?: string;
29 | }
30 |
31 | export declare interface PublicAccount {
32 | address: string | null;
33 | publicKey: string[] | null;
34 | authKey: string | null;
35 | minKeysRequired?: number;
36 | }
37 |
38 | export type NetworkInfo = {
39 | api?: string;
40 | chainId?: string;
41 | name: WalletAdapterNetwork | undefined;
42 | };
43 |
44 | export type TxOptions = {
45 | max_gas_amount?: string;
46 | gas_unit_price?: string;
47 | [key: string]: any;
48 | };
49 |
50 | export declare interface AptosProviderInterface
51 | extends BloctoProviderInterface {
52 | publicAccount: PublicAccount;
53 | network(): Promise;
54 | connect: () => Promise;
55 | isConnected: () => Promise;
56 | signAndSubmitTransaction(
57 | transaction: any,
58 | txOptions?: TxOptions
59 | ): Promise<{ hash: HexEncodedBytes }>;
60 | signTransaction(transaction: any): Promise;
61 | signMessage(payload: SignMessagePayload): Promise;
62 | disconnect(): Promise;
63 | }
64 |
--------------------------------------------------------------------------------
/packages/dappauth/test/unit/test-utils.js:
--------------------------------------------------------------------------------
1 | import { randomBytes } from 'crypto';
2 | import { Buffer } from 'safe-buffer';
3 | import { keccak, ecsign, toRpcSig, hashPersonalMessage, toBuffer, stripHexPrefix, bufferToHex, privateToAddress } from 'ethereumjs-util';
4 |
5 | export function stringToHex(str) {
6 | return bufferToHex(Buffer.from(str, 'utf8'));
7 | }
8 |
9 | /**
10 | * @param {Buffer} message
11 | * @param {String} address
12 | * @returns {Buffer}
13 | */
14 | export function generateErc191MessageHash(message, address) {
15 | const erc191Message = Buffer.concat([
16 | Buffer.from('19', 'hex'),
17 | Buffer.from('00', 'hex'),
18 | Buffer.from(stripHexPrefix(address), 'hex'),
19 | message,
20 | ]);
21 | const r = keccak(erc191Message)
22 | return r;
23 | }
24 |
25 |
26 | /**
27 | * @param {Buffer | String} message
28 | * @param {String} key
29 | * @param {String} address
30 | * @returns {String} the address of the wallet (identity)
31 | */
32 | export function signERC1654PersonalMessage(message, key, address) {
33 | // we hash once before erc191MessageHash as it will be transmitted to Ethereum nodes and potentially logged
34 | const messageHash = generateErc191MessageHash(keccak(toBuffer(stringToHex(message))), address);
35 | const signature = ecsign(messageHash, key);
36 | return toRpcSig(signature.v, signature.r, signature.s);
37 | }
38 |
39 | // emulates what EOA wallets like MetaMask perform
40 | export function signEOAPersonalMessage(message, key) {
41 | const messageHash = hashPersonalMessage(toBuffer(stringToHex(message)));
42 | const signature = ecsign(messageHash, key);
43 | return toRpcSig(signature.v, signature.r, signature.s);
44 | }
45 |
46 |
47 | export function generateRandomKey() {
48 | return toBuffer(`0x${randomBytes(32).toString('hex')}`);
49 | }
50 |
51 | export function keyToAddress(key) {
52 | return bufferToHex(privateToAddress(key));
53 | }
54 |
--------------------------------------------------------------------------------
/packages/dappauth/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@blocto/dappauth",
3 | "version": "2.2.2",
4 | "description": "A util to prove actionable control ('ownership') over a public Ethereum address using eth_sign",
5 | "keywords": [
6 | "ethereum",
7 | "ownership",
8 | "control",
9 | "eth_sign"
10 | ],
11 | "type": "module",
12 | "repository": "github:blocto/dappauth.js",
13 | "main": "./dist/dappauth.js",
14 | "module": "./dist/dappauth.module.js",
15 | "browser": "./dist/dappauth.umd.js",
16 | "exports": {
17 | "require": "./dist/dappauth.js",
18 | "import": "./dist/dappauth.module.js"
19 | },
20 | "scripts": {
21 | "build": "rollup -c",
22 | "lint": "eslint --ext .js src/",
23 | "test": "jest --config=./test/unit/jest.config.cjs --coverage=true --coverage-reporters=text",
24 | "prettier:watch": "onchange '**/*.js' -- prettier --write {{changed}}"
25 | },
26 | "author": "Pazams ",
27 | "license": "MIT",
28 | "dependencies": {
29 | "ethereumjs-abi": "^0.6.6",
30 | "ethereumjs-util": "^7.1.5",
31 | "safe-buffer": "^5.1.2",
32 | "web3": "^1.3.1"
33 | },
34 | "devDependencies": {
35 | "@babel/core": "^7.23.6",
36 | "@babel/preset-env": "^7.23.6",
37 | "@babel/register": "^7.22.15",
38 | "@rollup/plugin-commonjs": "^25.0.4",
39 | "@rollup/plugin-node-resolve": "^15.2.1",
40 | "babel-eslint": "^8.0.1",
41 | "babel-jest": "^29.7.0",
42 | "babel-plugin-istanbul": "^6.1.1",
43 | "coveralls": "^3.0.0",
44 | "eslint": "^8.42.0",
45 | "eslint-config-prettier": "^2.9.0",
46 | "eslint-plugin-prettier": "^2.6.0",
47 | "jest": "^29.7.0",
48 | "onchange": "^7.1.0",
49 | "prettier": "^1.14.3",
50 | "rollup": "^3.29.1",
51 | "rollup-plugin-copy": "^3.5.0",
52 | "rollup-plugin-json": "^4.0.0",
53 | "rollup-plugin-node-polyfills": "^0.2.1",
54 | "rollup-plugin-terser": "^7.0.2"
55 | }
56 | }
57 |
--------------------------------------------------------------------------------
/adapters/connectkit-connector/src/components/BloctoLogo.tsx:
--------------------------------------------------------------------------------
1 | const BLOCTO_LOGO =
2 | '';
3 |
4 | export const BloctoLogo = () =>
;
5 |
--------------------------------------------------------------------------------
/packages/dappauth/rollup.config.js:
--------------------------------------------------------------------------------
1 | import commonjs from '@rollup/plugin-commonjs';
2 | import resolve from '@rollup/plugin-node-resolve';
3 | import json from 'rollup-plugin-json';
4 | import polyfills from 'rollup-plugin-polyfill-node';
5 | import copy from 'rollup-plugin-copy';
6 |
7 | export default [
8 | // CommonJS
9 | {
10 | input: 'src/index.js',
11 | output: {
12 | file: 'dist/dappauth.js',
13 | format: 'cjs',
14 | name: 'DappAuth',
15 | exports: 'auto',
16 | },
17 | plugins: [
18 | copy({
19 | targets: [
20 | { src: './src/ABIs', dest: 'dist' },
21 | { src: './src/utils', dest: 'dist' },
22 | ],
23 | }),
24 | commonjs(),
25 | resolve({
26 | browser: true,
27 | preferBuiltins: true,
28 | }),
29 | json(),
30 | polyfills(),
31 | ],
32 | treeshake: {
33 | unknownGlobalSideEffects: false,
34 | },
35 | },
36 | // es
37 | {
38 | input: 'src/index.js',
39 | output: {
40 | file: 'dist/dappauth.module.js',
41 | format: 'es',
42 | name: 'DappAuth',
43 | },
44 | plugins: [
45 | copy({
46 | targets: [
47 | { src: './src/ABIs', dest: 'dist' },
48 | { src: './src/utils', dest: 'dist' },
49 | ],
50 | }),
51 | commonjs(),
52 | resolve({
53 | browser: true,
54 | preferBuiltins: true,
55 | }),
56 | json(),
57 | polyfills(),
58 | ],
59 | },
60 | // umd
61 | {
62 | input: 'src/index.js',
63 | output: {
64 | file: 'dist/dappauth.umd.js',
65 | format: 'umd',
66 | name: 'DappAuth',
67 | },
68 | plugins: [
69 | copy({
70 | targets: [
71 | { src: './src/ABIs', dest: 'dist' },
72 | { src: './src/utils', dest: 'dist' },
73 | ],
74 | }),
75 | commonjs(),
76 | resolve({
77 | preferBuiltins: true,
78 | browser: true,
79 | }),
80 | json(),
81 | polyfills(),
82 | ],
83 | },
84 | ];
85 |
--------------------------------------------------------------------------------
/adapters/web3modal-connector/src/web3ModalConfig.ts:
--------------------------------------------------------------------------------
1 | const BLOCTO_LOGO =
2 | '';
3 |
4 | const bloctoWeb3ModalConfig = {
5 | walletImages: {
6 | blocto: BLOCTO_LOGO,
7 | },
8 | };
9 |
10 | export default bloctoWeb3ModalConfig;
11 |
--------------------------------------------------------------------------------
/packages/dappauth/test/unit/contract-mock.js:
--------------------------------------------------------------------------------
1 | import { fromRpcSig, ecrecover, publicToAddress } from 'ethereumjs-util';
2 | import abi from 'ethereumjs-abi';
3 | import { Buffer } from 'safe-buffer';
4 | import { generateErc191MessageHash } from './test-utils.js';
5 |
6 | const { rawDecode } = abi;
7 | // bytes4(keccak256("isValidSignature(bytes32,bytes)")
8 | const ERC1271_METHOD_SIG = '1626ba7e';
9 |
10 | export default class MockContract {
11 | constructor(options) {
12 | this.authorizedKey = options.authorizedKey;
13 | this.address = options.address;
14 | this.errorIsValidSignature = options.errorIsValidSignature;
15 | }
16 |
17 | static _true() {
18 | return `0x${ERC1271_METHOD_SIG}00000000000000000000000000000000000000000000000000000000`; // a.k.a the "magic value".
19 | }
20 |
21 | static _false() {
22 | return '0x0000000000000000000000000000000000000000000000000000000000000000';
23 | }
24 |
25 | // @param {String} methodCall
26 | // @param {String} methodParams
27 | // @return {String}
28 | run(methodCall, methodParams) {
29 | switch (methodCall) {
30 | case ERC1271_METHOD_SIG: {
31 | const [hash, signature] = rawDecode(
32 | ['bytes32', 'bytes'],
33 | Buffer.from(methodParams, 'hex'),
34 | );
35 | return this._1626ba7e(hash, signature);
36 | }
37 | default:
38 | throw new Error(`Unexpected method ${methodCall}`);
39 | }
40 | }
41 |
42 | // "isValidSignature" method call
43 | // @param {Buffer} hash
44 | // @param {Buffer} signature
45 | // @return {String}
46 | _1626ba7e(hash, signature) {
47 | if (this.errorIsValidSignature) {
48 | throw new Error('isValidSignature call returned an error');
49 | }
50 | // split to 65 bytes (130 hex) chunks
51 | const multi_sigs = signature.toString('hex').match(/.{1,130}/g);
52 |
53 | // TODO: is it first or last?
54 | const expected_authrorised_sig = multi_sigs[0];
55 |
56 | // Get the address of whoever signed this message
57 | const { v, r, s } = fromRpcSig(`0x${expected_authrorised_sig}`);
58 | const erc191MessageHash = generateErc191MessageHash(hash, this.address);
59 | const recoveredKey = ecrecover(erc191MessageHash, v, r, s);
60 | const recoveredAddress = publicToAddress(recoveredKey);
61 |
62 | const expectedAddress = publicToAddress(this.authorizedKey);
63 |
64 | if (recoveredAddress.toString() === expectedAddress.toString()) {
65 | return MockContract._true();
66 | }
67 |
68 | return MockContract._false();
69 | }
70 | }
71 |
--------------------------------------------------------------------------------
/packages/dappauth/src/index.js:
--------------------------------------------------------------------------------
1 | import Web3 from 'web3';
2 | import { fromRpcSig, ecrecover, publicToAddress, bufferToHex, hashPersonalMessage, keccak } from 'ethereumjs-util';
3 | import ERC1271 from './ABIs/ERC1271.js';
4 | import { mergeErrors, isHexString, removeHexPrefix } from './utils/index.js';
5 |
6 | // bytes4(keccak256("isValidSignature(bytes32,bytes)")
7 | const ERC1271_MAGIC_VALUE = '0x1626ba7e';
8 |
9 | export default class DappAuth {
10 | constructor(web3Provider) {
11 | this.web3 = new Web3(web3Provider);
12 | }
13 |
14 | async isAuthorizedSigner(challenge, signature, address) {
15 | const eoaChallengeHash = this._hashEOAPersonalMessage(challenge);
16 | let isAuthorizedDirectKey;
17 | let errEOA;
18 |
19 | // try direct-keyed wallet
20 | try {
21 | // Get the address of whoever signed this message
22 | const { v, r, s } = fromRpcSig(signature);
23 | const recoveredKey = ecrecover(eoaChallengeHash, v, r, s);
24 | const recoveredAddress = publicToAddress(recoveredKey);
25 |
26 | if (
27 | address.toLowerCase() ===
28 | bufferToHex(recoveredAddress).toLowerCase()
29 | ) {
30 | isAuthorizedDirectKey = true;
31 | }
32 | } catch (err) {
33 | errEOA = err;
34 | // if either direct-keyed auth flow threw an error, or it did not conclude to be authorized, proceed to try smart-contract wallet.
35 | }
36 | try {
37 | if (isAuthorizedDirectKey === true) return isAuthorizedDirectKey;
38 | // try smart-contract wallet
39 | const erc1271CoreContract = new this.web3.eth.Contract(ERC1271, address);
40 |
41 | // check persoonalSign hash first
42 | const magicValue = await erc1271CoreContract.methods
43 | .isValidSignature(eoaChallengeHash, signature)
44 | .call();
45 | if (magicValue === ERC1271_MAGIC_VALUE) {
46 | return true;
47 | } else {
48 | // then check SCMessage hash (ERC-1654)
49 | const magicValue2 = await erc1271CoreContract.methods
50 | .isValidSignature(this._hashSCMessage(challenge), signature)
51 | .call();
52 |
53 | return magicValue2 === ERC1271_MAGIC_VALUE;
54 | }
55 | } catch (err) {
56 | throw mergeErrors(errEOA, err);
57 | }
58 | }
59 |
60 | _hashEOAPersonalMessage(challenge) {
61 | return hashPersonalMessage(this._decodeChallenge(challenge));
62 | }
63 |
64 | // This is a hash just over the challenge. The smart contract takes this result and hashes on top to an erc191 hash.
65 | _hashSCMessage(challenge) {
66 | return keccak(this._decodeChallenge(challenge));
67 | }
68 |
69 | // See https://github.com/MetaMask/eth-sig-util/issues/60
70 | _decodeChallenge(challenge) {
71 | return isHexString(challenge)
72 | ? Buffer.from(removeHexPrefix(challenge), 'hex')
73 | : Buffer.from(challenge);
74 | }
75 | }
76 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@blocto/sdk",
3 | "version": "0.10.4",
4 | "repository": "git@github.com:portto/blocto-sdk.git",
5 | "author": "Chiaki.C",
6 | "license": "MIT",
7 | "main": "dist/blocto-sdk.js",
8 | "module": "dist/blocto-sdk.module.js",
9 | "browser": "dist/blocto-sdk.umd.js",
10 | "types": "dist/blocto-sdk.d.ts",
11 | "type": "module",
12 | "exports": {
13 | "require": "./dist/blocto-sdk.js",
14 | "import": "./dist/blocto-sdk.module.js",
15 | "types": "./dist/blocto-sdk.d.ts"
16 | },
17 | "files": [
18 | "/dist/**/!(*.html)"
19 | ],
20 | "scripts": {
21 | "build": "NODE_ENV=production rollup -c",
22 | "clean": "rimraf dist",
23 | "lint": "eslint --ext .js,.ts src/",
24 | "test": "jest",
25 | "prepublishOnly": "npm-run-all clean build",
26 | "rollup-watch": "NODE_ENV=development rollup -c rollup.config.dev.js --watch",
27 | "serve": "live-server --port=7777 --https=dev-cert/index.js dev",
28 | "start": "npm-run-all --parallel rollup-watch serve"
29 | },
30 | "dependencies": {
31 | "buffer": "^6.0.3",
32 | "eip1193-provider": "^1.0.1",
33 | "js-sha3": "^0.8.0"
34 | },
35 | "devDependencies": {
36 | "@babel/core": "^7.22.1",
37 | "@babel/eslint-parser": "^7.21.8",
38 | "@babel/eslint-plugin": "^7.14.5",
39 | "@babel/plugin-transform-runtime": "^7.22.4",
40 | "@babel/preset-env": "^7.22.4",
41 | "@babel/runtime": "^7.22.3",
42 | "@rollup/plugin-alias": "^5.0.0",
43 | "@rollup/plugin-babel": "^6.0.3",
44 | "@rollup/plugin-commonjs": "^25.0.0",
45 | "@rollup/plugin-node-resolve": "^15.1.0",
46 | "@rollup/plugin-typescript": "^11.1.1",
47 | "@types/jest": "^29.5.2",
48 | "@typescript-eslint/eslint-plugin": "^5.59.9",
49 | "@typescript-eslint/parser": "^5.59.9",
50 | "aptos": "^1.15.0",
51 | "babel-jest": "^29.5.0",
52 | "babel-plugin-module-resolver": "^5.0.0",
53 | "eslint": "^8.42.0",
54 | "eslint-import-resolver-babel-module": "^5.3.1",
55 | "eslint-plugin-import": "^2.23.4",
56 | "eth-rpc-errors": "^4.0.3",
57 | "jest": "^29.5.0",
58 | "jest-environment-jsdom": "^29.5.0",
59 | "jest-fetch-mock": "^3.0.3",
60 | "live-server": "^1.2.1",
61 | "rimraf": "^5.0.1",
62 | "rollup": "^3.23.1",
63 | "rollup-plugin-dts": "^5.3.0",
64 | "rollup-plugin-json": "^4.0.0",
65 | "rollup-plugin-peer-deps-external": "^2.2.4",
66 | "rollup-plugin-polyfill-node": "^0.12.0",
67 | "rollup-plugin-terser": "^7.0.2",
68 | "rollup-plugin-version-injector": "^1.3.3",
69 | "rollup-plugin-visualizer": "^5.5.4",
70 | "ts-jest": "^29.1.0",
71 | "tslib": "^2.5.3",
72 | "typescript": "~5.1.3",
73 | "yarn-run-all": "^3.1.1"
74 | },
75 | "peerDependencies": {
76 | "aptos": "^1.3.14"
77 | },
78 | "peerDependenciesMeta": {
79 | "aptos": {
80 | "optional": true
81 | }
82 | },
83 | "keywords": [
84 | "portto",
85 | "blocto",
86 | "ethereum",
87 | "aptos",
88 | "wallet",
89 | "sdk",
90 | "web3",
91 | "web3.js",
92 | "javascript",
93 | "typescript",
94 | "dapp",
95 | "react",
96 | "vue"
97 | ]
98 | }
99 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/__tests__/isValidTransaction.test.js:
--------------------------------------------------------------------------------
1 | import { isValidTransaction, isValidTransactions } from "../lib/isValidTransaction";
2 | import { ERROR_MESSAGE } from "../lib/constant";
3 |
4 | describe("isValidTransaction", () => {
5 | it("should return valid when passing valid transaction", () => {
6 | const mockTransaction = {
7 | from: "0xC5a6382a81CDA092DBdE9dFeEfa2642306cF0006",
8 | to: "0x509Ee0d083DdF8AC028f2a56731412edD63223B9",
9 | value: "0x1"
10 | }
11 |
12 | expect(isValidTransaction(mockTransaction)).toEqual({ isValid: true });
13 | });
14 |
15 | it("should return invalid when passing invalid transaction without 'from'", () => {
16 | const mockTransaction = {
17 | to: "0x509Ee0d083DdF8AC028f2a56731412edD63223B9",
18 | value: "0x1"
19 | };
20 |
21 | expect(isValidTransaction(mockTransaction)).toEqual({ isValid: false, invalidMsg: ERROR_MESSAGE.INVALID_TRANSACTION });
22 | });
23 |
24 | it("should return invalid when passing transaction with invalid 'value'", () => {
25 | const mockTransaction = {
26 | from: "0xC5a6382a81CDA092DBdE9dFeEfa2642306cF0006",
27 | to: "0x509Ee0d083DdF8AC028f2a56731412edD63223B9",
28 | value: 1
29 | };
30 |
31 | expect(isValidTransaction(mockTransaction)).toEqual({ isValid: false, invalidMsg: ERROR_MESSAGE.INVALID_TRANSACTION_VALUE });
32 | });
33 | });
34 |
35 | describe("isValidTransactions", () => {
36 | it("should return valid for an array of valid transactions", () => {
37 | const mockTransactions = [
38 | {
39 | from: "0xC5a6382a81CDA092DBdE9dFeEfa2642306cF0006",
40 | to: "0x509Ee0d083DdF8AC028f2a56731412edD63223B9",
41 | value: "0x1"
42 | },
43 | {
44 | from: "0xC5a6382a81CDA092DBdE9dFeEfa2642306cF0006",
45 | to: "0x509Ee0d083DdF8AC028f2a56731412edD63223B9",
46 | data: "0xa9059cbb00000000000000000000000076c7a9cac72db4865bd1db3acfa78a261755c4fb0000000000000000000000000000000000000000000000000000000005f5e100",
47 | }
48 | ];
49 |
50 | expect(isValidTransactions(mockTransactions)).toEqual({ isValid: true });
51 | });
52 |
53 | it("should return invalid when passing invalid array of transactions", () => {
54 | const mockTransactions = "notAnArray";
55 |
56 | expect(isValidTransactions(mockTransactions)).toEqual({ isValid: false, invalidMsg: ERROR_MESSAGE.INVALID_TRANSACTIONS });
57 | });
58 |
59 | it("should return invalid when passing contain invalid transactions", () => {
60 | const mockTransactions = [
61 | {
62 | from: "0xC5a6382a81CDA092DBdE9dFeEfa2642306cF0006",
63 | to: "0x509Ee0d083DdF8AC028f2a56731412edD63223B9",
64 | value: 1
65 | },
66 | {
67 | from: "0xC5a6382a81CDA092DBdE9dFeEfa2642306cF0006",
68 | to: "0x509Ee0d083DdF8AC028f2a56731412edD63223B9",
69 | data: "0xa9059cbb000000000000000000000000d3ea1bfe3d1f99278c2ac7b6429e2e2ac32564e30000000000000000000000000000000000000000000000000000000005f5e100",
70 | }
71 | ];
72 |
73 | expect(isValidTransactions(mockTransactions)).toEqual({ isValid: false, invalidMsg: ERROR_MESSAGE.INVALID_TRANSACTION_VALUE });
74 | });
75 | });
76 |
--------------------------------------------------------------------------------
/adapters/rainbowkit-connector/src/index.ts:
--------------------------------------------------------------------------------
1 | import { Wallet, WalletDetailsParams } from '@rainbow-me/rainbowkit';
2 | import { blocto, createConnector } from '@blocto/wagmi-connector';
3 |
4 | export const bloctoWallet =
5 | ({ appId }: { appId?: string } = {}) =>
6 | (): Wallet => {
7 | return {
8 | id: 'blocto',
9 | name: 'Blocto',
10 | shortName: 'Blocto',
11 | rdns: 'io.blocto',
12 | iconBackground: '#ffffff',
13 | iconUrl:
14 | '',
15 | downloadUrls: {
16 | ios: 'https://apps.apple.com/app/blocto/id1481181682',
17 | android:
18 | 'https://play.google.com/store/apps/details?id=com.portto.blocto',
19 | },
20 | installed: true,
21 | createConnector: (walletDetails: WalletDetailsParams) =>
22 | createConnector((config) => ({
23 | ...blocto({ appId })(config),
24 | ...walletDetails,
25 | })),
26 | };
27 | };
28 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/README.md:
--------------------------------------------------------------------------------
1 | # Blocto JavaScript SDK
2 |
3 | [](https://www.npmjs.com/package/@blocto/sdk)
4 | [](https://www.npmjs.com/package/@blocto/sdk)
5 | [](https://www.npmjs.com/package/@blocto/sdk)
6 | [](https://github.com/portto/blocto-sdk/actions/workflows/test.yml)
7 | [](https://www.npmjs.com/package/@blocto/sdk)
8 | [](https://github.com/portto/blocto-sdk/blob/main/LICENSE)
9 | [](https://discord.com/invite/QRZTr6yHmY)
10 |
11 | Use Blocto SDK in your dApp to provide excellent user experience!
12 |
13 | ## Looking for the documentation?
14 |
15 | [https://docs.blocto.app/blocto-sdk/javascript-sdk](https://docs.blocto.app/blocto-sdk/javascript-sdk)
16 |
17 | ## What can I do with Blocto SDK?
18 |
19 | - Interact with multiple blockchains
20 | - Interact with Ethereum-compatible blockchains
21 | Ethereum Mainnet & Goerli Testnet
22 | Arbitrum Mainnet & Arbitrum Goerli Testnet
23 | Optimism Mainnet & Optimism Goerli Testnet
24 | Polygon Mainnet & Testnet
25 | BNB Smart Chain Mainnet & Tetsnet
26 | Avalanche Mainnet & Testnet
27 | - Interact with Aptos
28 | - Sign messages
29 | - Send transactions
30 | - ... and a lot more
31 | - Seamless onboarding experience
32 | Users can sign up easily with email and start exploring you dApp in seconds.
33 | - Fee subsidization
34 | You have the option to pay transaction fee for your users and provide a better experience. In that case, we will generate daily fee reports for you to review.
35 | - Integrated payment
36 | Get paid easily with our payment APIs. Users can pay easily with credit cards or other crypto currencies like Bitcoin, Ethereum, Tron, USDT, ...
37 | - Connected to Blocto App
38 | Once you've integrated with Blocto SDK, your users can manage their assets easily and securely through Blocto App. Your dApp can tap into the vast blockchain ecosystem instantly.
39 |
40 | ## Installing
41 |
42 | ```bash
43 | $ yarn add @blocto/sdk
44 | # or
45 | $ npm i @blocto/sdk
46 | ```
47 |
48 | ## Usage
49 |
50 | ```ts
51 | import BloctoSDK from '@blocto/sdk';
52 |
53 | const bloctoSDK = new BloctoSDK({
54 | // advance settings please refer to docs
55 | ethereum: {
56 | // (required) chainId to be used
57 | chainId: '0x5',
58 | // (required for Ethereum) JSON RPC endpoint
59 | rpc: 'https://mainnet.infura.io/v3/YOUR_INFURA_ID',
60 | },
61 |
62 | // (optional) Blocto app ID
63 | appId: 'YOUR_BLOCTO_APP_ID',
64 | });
65 | ```
66 |
67 | ## Testing on local
68 |
69 | ```
70 | yarn
71 |
72 | # generate local https cert
73 | brew install mkcert
74 | mkcert -install
75 | cd dev-cert
76 | mkcert localhost
77 |
78 | # make dev directory
79 | mkdir dev
80 | ln -s src/main.js dev/main.js
81 | ```
82 |
83 | ## Scripts
84 |
85 | `build`: build dist
86 | `start`: run live-reload dev server on `https://localhost:7777`
87 |
88 | ## Develop
89 |
90 | open browser and navigate to `https://localhost:7777/test.html`
91 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/rollup.config.js:
--------------------------------------------------------------------------------
1 | import { terser } from 'rollup-plugin-terser';
2 | import { visualizer } from 'rollup-plugin-visualizer';
3 | import typescript from '@rollup/plugin-typescript';
4 | import commonjs from '@rollup/plugin-commonjs';
5 | import resolve from '@rollup/plugin-node-resolve';
6 | import json from 'rollup-plugin-json';
7 | import polyfills from 'rollup-plugin-polyfill-node';
8 | import alias from '@rollup/plugin-alias';
9 | import { babel } from '@rollup/plugin-babel';
10 | import dts from 'rollup-plugin-dts';
11 | import peerDepsExternal from 'rollup-plugin-peer-deps-external';
12 | import versionInjector from 'rollup-plugin-version-injector';
13 |
14 | export default [
15 | // CommonJS
16 | {
17 | input: 'src/main.ts',
18 | output: {
19 | file: 'dist/blocto-sdk.js',
20 | format: 'cjs',
21 | name: 'BloctoSDK',
22 | exports: 'auto',
23 | },
24 | plugins: [
25 | versionInjector(),
26 | alias({
27 | entries: {
28 | 'readable-stream': 'stream',
29 | },
30 | }),
31 | peerDepsExternal(),
32 | resolve({
33 | preferBuiltins: true,
34 | browser: true,
35 | }),
36 | typescript(),
37 | commonjs(),
38 | json(),
39 | babel({
40 | babelHelpers: 'runtime',
41 | exclude: 'node_modules/**',
42 | plugins: [['@babel/plugin-transform-runtime']],
43 | presets: [['@babel/preset-env']],
44 | }),
45 | polyfills(),
46 | visualizer({ filename: 'dist/stats.cjs.html' }),
47 | ],
48 | },
49 | // es
50 | {
51 | input: 'src/main.ts',
52 | output: {
53 | file: 'dist/blocto-sdk.module.js',
54 | format: 'es',
55 | name: 'BloctoSDK',
56 | },
57 | plugins: [
58 | versionInjector(),
59 | alias({
60 | entries: {
61 | 'readable-stream': 'stream',
62 | },
63 | }),
64 | peerDepsExternal(),
65 | resolve({
66 | preferBuiltins: true,
67 | browser: true,
68 | }),
69 | typescript(),
70 | commonjs(),
71 | json(),
72 | babel({
73 | babelHelpers: 'runtime',
74 | exclude: 'node_modules/**',
75 | plugins: [['@babel/plugin-transform-runtime']],
76 | presets: [['@babel/preset-env']],
77 | }),
78 | polyfills(),
79 | visualizer({ filename: 'dist/stats.es.html' }),
80 | ],
81 | },
82 | // umd
83 | {
84 | input: 'src/main.ts',
85 | output: {
86 | file: 'dist/blocto-sdk.umd.js',
87 | format: 'umd',
88 | name: 'BloctoSDK',
89 | },
90 | plugins: [
91 | versionInjector(),
92 | alias({
93 | entries: {
94 | 'readable-stream': 'stream',
95 | },
96 | }),
97 | resolve({
98 | preferBuiltins: true,
99 | browser: true,
100 | }),
101 | typescript(),
102 | commonjs(),
103 | json(),
104 | babel({
105 | babelHelpers: 'bundled',
106 | exclude: 'node_modules/**',
107 | presets: [['@babel/preset-env']],
108 | }),
109 | polyfills(),
110 | terser(),
111 | visualizer({ filename: 'dist/stats.umd.html' }),
112 | ],
113 | },
114 | {
115 | input: './src/index.d.ts',
116 | output: [{ file: 'dist/blocto-sdk.d.ts', format: 'es' }],
117 | plugins: [dts()],
118 | },
119 | ];
120 |
--------------------------------------------------------------------------------
/adapters/web3-react-connector/src/index.ts:
--------------------------------------------------------------------------------
1 | import type {
2 | Actions,
3 | AddEthereumChainParameter,
4 | ProviderConnectInfo,
5 | ProviderRpcError,
6 | } from '@web3-react/types';
7 | import { Connector } from '@web3-react/types';
8 | import BloctoSDK from '@blocto/sdk';
9 |
10 | function parseChainId(chainId: string | number): number {
11 | return typeof chainId === 'number'
12 | ? chainId
13 | : Number.parseInt(chainId, chainId.startsWith('0x') ? 16 : 10);
14 | }
15 |
16 | /**
17 | * @param options - Options to pass to Blocto SDK.
18 | * @param onError - Handler to report errors thrown from eventListeners.
19 | */
20 | type BloctoOptions = {
21 | chainId: number;
22 | rpc: string;
23 | };
24 |
25 | export interface BloctoConstructorArgs {
26 | actions: Actions;
27 | options: BloctoOptions;
28 | onError?: (error: Error) => void;
29 | }
30 |
31 | export class BloctoConnector extends Connector {
32 | public provider: any;
33 |
34 | constructor({ actions, options, onError }: BloctoConstructorArgs) {
35 | super(actions, onError);
36 | const bloctoSDK = new BloctoSDK({
37 | ethereum: {
38 | chainId: options.chainId,
39 | rpc: options.rpc,
40 | },
41 | });
42 |
43 | this.provider = bloctoSDK.ethereum;
44 |
45 | this.provider.on(
46 | 'connect',
47 | async ({ chainId }: ProviderConnectInfo): Promise => {
48 | const accounts = await this.provider.request({
49 | method: 'eth_requestAccounts',
50 | });
51 | this.actions.update({ chainId: parseChainId(chainId), accounts });
52 | }
53 | );
54 | this.provider.on('disconnect', (error: ProviderRpcError): void => {
55 | this.actions.resetState();
56 | this.onError?.(error);
57 | });
58 | this.provider.on('chainChanged', async (chainId: string): Promise => {
59 | const accounts = await this.provider.request({
60 | method: 'eth_requestAccounts',
61 | });
62 | this.actions.update({ chainId: parseChainId(chainId), accounts });
63 | });
64 | }
65 |
66 | public async activate(
67 | desiredChainIdOrChainParameters?: number | AddEthereumChainParameter
68 | ): Promise {
69 | const desiredChainId =
70 | typeof desiredChainIdOrChainParameters === 'number'
71 | ? desiredChainIdOrChainParameters
72 | : desiredChainIdOrChainParameters?.chainId;
73 |
74 | if (!this.provider) throw new Error('No provider');
75 | if (
76 | !desiredChainId ||
77 | parseChainId(desiredChainId) === parseChainId(this.provider.chainId)
78 | ) {
79 | const accounts = await this.provider.request({
80 | method: 'eth_requestAccounts',
81 | });
82 |
83 | return this.actions.update({
84 | chainId: parseChainId(this.provider.chainId),
85 | accounts,
86 | });
87 | }
88 | const addEthereumChainParameters =
89 | typeof desiredChainIdOrChainParameters === 'number'
90 | ? { chainId: desiredChainId }
91 | : desiredChainIdOrChainParameters;
92 |
93 | await this.provider.request({
94 | method: 'wallet_addEthereumChain',
95 | params: [addEthereumChainParameters],
96 | });
97 | await this.provider.request({
98 | method: 'wallet_switchEthereumChain',
99 | params: [{ chainId: desiredChainId }],
100 | });
101 | }
102 |
103 | public deactivate(): void {
104 | this.provider.request({ method: 'wallet_disconnect' });
105 | this.actions.resetState();
106 | }
107 | }
108 |
--------------------------------------------------------------------------------
/CONTRIBUTING.md:
--------------------------------------------------------------------------------
1 | # Contributing to Blocto JavaScript SDK
2 |
3 | Thanks for showing interest to contribute to Blocto JavaScript SDK, you rock!
4 |
5 | ## When it comes to open source, there are different ways you can contribute, all of which are valuable. Here's a few guidelines that should help you as you prepare your contribution.
6 |
7 | ## Setup the Project
8 |
9 | The following steps will get you up and running to contribute to Chakra UI:
10 |
11 | 1. Fork the repo (click the Fork button at the top right of
12 | [this page](https://github.com/portto/blocto-sdk))
13 |
14 | 2. Clone your fork locally
15 |
16 | ```sh
17 | git clone https://github.com//blocto-sdk.git
18 | cd blocto-sdk
19 | ```
20 |
21 | 3. Setup all the dependencies and packages by running `yarn`. This
22 | command will install dependencies.
23 |
24 | ## Development
25 |
26 | To improve our development process, we've set up tooling and systems. Blocto JavaScript SDK is a monorepo built with [turbo](https://turbo.build/repo/docs) and follows its file structure convention. The repo has 2 workspaces `adapters` and `packages`.
27 |
28 | ### Commit Convention
29 |
30 | Before you create a Pull Request, please check whether your commits comply with
31 | the commit conventions used in this repository.
32 |
33 | When you create a commit we kindly ask you to follow the convention
34 | `category(scope or module): message` in your commit message while using one of
35 | the following categories:
36 |
37 | - `feat`: all changes that introduce completely new code or new
38 | features
39 | - `fix`: changes that fix a bug (ideally you will additionally reference an
40 | issue if present)
41 | - `refactor`: any code related change that is not a fix nor a feature
42 | - `docs`: changing existing or creating new documentation (i.e. README, docs for
43 | usage of a lib or cli usage)
44 | - `build`: all changes regarding the build of the software, changes to
45 | dependencies or the addition of new dependencies
46 | - `test`: all changes regarding tests (adding new tests or changing existing
47 | ones)
48 | - `ci`: all changes regarding the configuration of continuous integration (i.e.
49 | github actions, ci system)
50 | - `chore`: all changes to the repository that do not fit into any of the above
51 | categories
52 |
53 | If you are interested in the detailed specification you can visit
54 | https://www.conventionalcommits.org/ or check out the
55 | [Angular Commit Message Guidelines](https://github.com/angular/angular/blob/22b96b9/CONTRIBUTING.md#-commit-message-guidelines).
56 |
57 | ### Creating a Pull Request
58 |
59 | You are welcome to create a pull reuqest against the `develop` branch. The beta release channel goes here. When it's stable, we will pack it all and release in stable channel.
60 |
61 | Before creating a PR
62 |
63 | 1. Make sure your branch is up to date with the `develop` branch
64 | 2. On the `root` folder, run `yarn`
65 | 3. On the `root` folder, run `yarn build`
66 | 4. If a pull request created needs to bump a package version, please see [Changesets](#Changesets) part and add a changesets.
67 |
68 | If everything passes, you should be able to create a PR.
69 |
70 | #### Changesets
71 |
72 | We use [changesets](https://github.com/changesets/changesets) to handle any changes in the `changelog`.
73 | If a pull request created needs to bump a package version, please follow those steps to create a `changelog`
74 |
75 | 1. On the `root` folder, run `yarn changeset` and follow the prompt instructions (we follow [SemVer](https://semver.org/))
76 | > Tips: Skip Major and Minor in prompt means a Patch update
77 | 2. Under `.changeset/` you will notice a new markdown file (its name is randomly generated), with the change-type and summary.
78 | 3. Push the file along with the rest of the changes
79 |
80 | Once your PR will be merged, our Github action will create a new PR with that generated changelog for us to merge, once the generated PR is merged a new version will be published to npm.
81 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/example/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
14 |
15 |
16 |
17 | Custom Provider
18 |
19 |
20 |
37 | Enabling account ...
38 |
111 |
112 |
113 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # PLEASE NOTE, THIS PROJECT IS NO LONGER BEING MAINTAINED
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 | # Blocto JavaScript SDK
10 |
11 | > ⚠️ **Deprecation Notice**
12 | >
13 | > This repository is **no longer maintained** and has been officially deprecated.
14 | >
15 | > We recommend NOT using this SDK in new projects.
16 | >
17 | > For existing integrations, please be aware that no further updates, bug fixes, or support will be provided.
18 | >
19 | > **IMPORTANT**: As of June 11, 2025, all services associated with this SDK will be completely discontinued and existing integrations will cease to function.
20 | >
21 | > Thank you for your support and for being part of the Blocto developer community.
22 |
23 | This is a monorepo of Blocto JavaScript SDK.
24 |
25 | ## What's inside?
26 |
27 | This repo includes the following packages/apps:
28 |
29 | ### Apps and Packages
30 |
31 | - `packages`: shared packages
32 | - `adapters`: adapters using Blocto JavaScript SDK
33 | - `eslint-config-custom`: `eslint` configurations (includes `eslint-config-next` and `eslint-config-prettier`)
34 | - `tsconfig`: `tsconfig.json`s used throughout the monorepo
35 |
36 | ### Utilities
37 |
38 | This Turborepo has some additional tools already setup for you:
39 |
40 | - [TypeScript](https://www.typescriptlang.org/) for static type checking
41 | - [ESLint](https://eslint.org/) for code linting
42 | - [Prettier](https://prettier.io) for code formatting
43 |
44 | ## Repo structure
45 |
46 | ```
47 | blocto-sdk
48 | ├─ packages
49 | │ ├─ @blocto/sdk
50 | │ │ └─ package.json
51 | │ ├─ tsconfig
52 | │ │ └─ package.json
53 | │ └─ eslint-config-custom
54 | │ └─ package.json
55 | ├─ adapters
56 | │ ├─ @blocto/aptos-wallet-adapter-plugin
57 | │ │ └─ package.json
58 | │ ├─ @blocto/rainbowkit-connector
59 | │ │ └─ package.json
60 | │ └─ @blocto/wagmi-connector
61 | │ │ └─ package.json
62 | │ ├─ @blocto/web3-react-connector
63 | │ │ └─ package.json
64 | └─ package.json
65 | ```
66 |
67 | ## Build
68 |
69 | To build all apps and packages, run the following command:
70 |
71 | ```
72 | yarn build
73 | ```
74 |
75 | ## Contributing
76 |
77 | Feel like contributing? That's awesome! We have a
78 | [contributing guide](./CONTRIBUTING.md) to help guide you.
79 |
80 | ## Add changeset
81 |
82 | 1. Run the command line script `yarn changeset`.
83 | 2. Select the packages you want to include in the changeset using ↑ and ↓ to navigate to packages, and space to select a package. Hit enter when all desired packages are selected.
84 | 3. You will be prompted to select a bump type for each selected package. Select an appropriate bump type for the changes made. See [here](https://semver.org/) for information on semver versioning
85 | 4. Your final prompt will be to provide a message to go alongside the changeset. This will be written into the changelog when the next release occurs.
86 | After this, a new changeset will be added which is a markdown file with YAML front matter.
87 |
88 | ```
89 | -| .changeset/
90 | -|-| UNIQUE_ID.md
91 | ```
92 |
93 | The message you typed can be found in the markdown file. If you want to expand on it, you can write as much markdown as you want, which will all be added to the changelog on publish. If you want to add more packages or change the bump types of any packages, that's also fine.
94 |
95 | While not every changeset is going to need a huge amount of detail, a good idea of what should be in a changeset is:
96 |
97 | - WHAT the change is
98 | - WHY the change was made
99 | - HOW a consumer should update their code
100 |
101 | 5. Once you are happy with the changeset, commit the file to your branch.
102 |
103 | ### Tips on adding changesets
104 |
105 | #### You can add more than one changeset to a pull request
106 |
107 | Changesets are designed to stack, so there's no problem with adding multiple. You might want to add more than one changeset when:
108 |
109 | - You want to release multiple packages with different changelog entries
110 | - You have made multiple changes to a package that should each be called out separately
111 |
--------------------------------------------------------------------------------
/adapters/rainbowkit-connector/src/__tests__/index.test.ts:
--------------------------------------------------------------------------------
1 | import { bloctoWallet } from '../index';
2 |
3 | describe('rainbowkit-connector', () => {
4 |
5 | const wallet = bloctoWallet();
6 |
7 | test('defines name', () => {
8 | expect(typeof wallet.name).toBe('string');
9 | });
10 |
11 | test('defines id', () => {
12 | expect(typeof wallet.id).toBe('string');
13 | });
14 |
15 | test('defines icon', () => {
16 | expect(typeof wallet.iconUrl).toBe('string');
17 | });
18 |
19 | test('defines connect()', () => {
20 | expect(typeof wallet.createConnector).toBe('function');
21 | });
22 |
23 | const { connector } = wallet.createConnector({
24 | rkDetails: {
25 | id: 'blocto',
26 | name: 'Blocto',
27 | shortName: 'Blocto',
28 | rdns: 'io.blocto',
29 | iconBackground: '#ffffff',
30 | iconUrl:
31 | '',
32 | downloadUrls: {
33 | ios: 'https://apps.apple.com/app/blocto/id1481181682',
34 | android:
35 | 'https://play.google.com/store/apps/details?id=com.portto.blocto',
36 | },
37 | installed: true,
38 | index: 4,
39 | groupIndex: 2,
40 | groupName: 'Other',
41 | isRainbowKitConnector: true,
42 | },
43 | });
44 |
45 | test('defines account()', () => {
46 | expect(typeof connector.connect).toBe('function');
47 | });
48 |
49 | test('defines disconnect()', () => {
50 | expect(typeof connector.disconnect).toBe('function');
51 | });
52 |
53 | test('defines getAccount()', () => {
54 | expect(typeof connector.getAccount).toBe('function');
55 | });
56 |
57 | test('defines getChainId()', () => {
58 | expect(typeof connector.getChainId).toBe('function');
59 | });
60 |
61 | test('defines getProvider()', () => {
62 | expect(typeof connector.getProvider).toBe('function');
63 | });
64 |
65 | test('defines switchChain()', () => {
66 | expect(typeof connector.switchChain).toBe('function');
67 | });
68 | });
69 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/providers/types/ethereum.d.ts:
--------------------------------------------------------------------------------
1 | import { IEthereumProvider } from 'eip1193-provider';
2 | import { BaseConfig, KEY_SESSION } from '../../constants';
3 | import BloctoProviderInterface from './blocto.d';
4 |
5 | interface SingleChainConfig extends BaseConfig {
6 | chainId: string | number | null;
7 | rpc?: string;
8 | walletServer?: string;
9 | }
10 |
11 | interface MultiChainConfig extends BaseConfig {
12 | defaultChainId: string | number | null;
13 | walletServer?: string;
14 | switchableChains: AddEthereumChainParameter[];
15 | }
16 | // EthereumProviderConfig can be both single chain or multi chain config.
17 | export type EthereumProviderConfig = SingleChainConfig | MultiChainConfig;
18 |
19 | export interface EIP1193RequestPayload {
20 | id?: number;
21 | jsonrpc?: string;
22 | method: string;
23 | params?: Array;
24 | callback?: JsonRpcCallback;
25 | }
26 |
27 | interface SwitchableNetwork {
28 | [id: number | string]: {
29 | name: string;
30 | display_name: string;
31 | network_type: string;
32 | wallet_web_url: string;
33 | rpc_url: string;
34 | };
35 | }
36 |
37 | export interface EthereumProviderInterface
38 | extends BloctoProviderInterface,
39 | IEthereumProvider {
40 | chainId: string | number;
41 | networkVersion: string | number;
42 | rpc: string;
43 | _blocto: {
44 | sessionKeyEnv: KEY_SESSION;
45 | walletServer: string;
46 | blockchainName: string;
47 | networkType: string;
48 | switchableNetwork: SwitchableNetwork;
49 | };
50 | sendUserOperation(userOp: IUserOperation): Promise;
51 | request(
52 | args: EIP1193RequestPayload | Array
53 | ): Promise;
54 | loadSwitchableNetwork(
55 | networkList: {
56 | chainId: string;
57 | rpcUrls?: string[];
58 | }[]
59 | ): Promise;
60 | supportChainList(): Promise<{ chainId: string; chainName: string }[]>;
61 | injectedWalletServer?: string;
62 | }
63 |
64 | export interface AddEthereumChainParameter {
65 | chainId: string;
66 | rpcUrls: string[];
67 | [key: string]: any;
68 | }
69 |
70 | export interface JsonRpcRequest {
71 | id?: string | undefined;
72 | jsonrpc: '2.0';
73 | method: string;
74 | params?: Array;
75 | }
76 |
77 | export interface JsonRpcResponse {
78 | id: string | undefined;
79 | jsonrpc: '2.0';
80 | method: string;
81 | result?: unknown;
82 | error?: Error;
83 | }
84 |
85 | export type JsonRpcCallback = (
86 | error: Error | null,
87 | response?: JsonRpcResponse
88 | ) => unknown;
89 |
90 | export interface PromiseResponseItem {
91 | status: 'fulfilled' | 'rejected';
92 | value?: any;
93 | reason?: any;
94 | }
95 |
96 | /**
97 | * A [[HexString]] whose length is even, which ensures it is a valid
98 | * representation of binary data.
99 | */
100 | export type DataHexString = string;
101 |
102 | /**
103 | * An object that can be used to represent binary data.
104 | */
105 | export type BytesLike = DataHexString | Uint8Array;
106 |
107 | /**
108 | * Any type that can be used where a numeric value is needed.
109 | */
110 | export type Numeric = number | bigint;
111 |
112 | /**
113 | * Any type that can be used where a big number is needed.
114 | */
115 | export type BigNumberish = string | Numeric;
116 |
117 | /**
118 | * An interface for an ERC-4337 transaction object.
119 | * Note: BloctoSDK do not need sender, nonce, initCode, signature to send userOperation.
120 | * These parameters will be ignored.
121 | */
122 | export interface IUserOperation {
123 | callData: BytesLike;
124 | callGasLimit?: BigNumberish;
125 | verificationGasLimit?: BigNumberish;
126 | preVerificationGas?: BigNumberish;
127 | maxFeePerGas?: BigNumberish;
128 | maxPriorityFeePerGas?: BigNumberish;
129 | paymasterAndData?: BytesLike;
130 | /**
131 | * If provided, please ensure it is same as login account.
132 | */
133 | sender?: string;
134 | /**
135 | * BloctoSDK do not need nonce to send userOperation. Will be ignored.
136 | * */
137 | nonce?: BigNumberish;
138 | /**
139 | * BloctoSDK do not need initCode to send userOperation. Will be ignored.
140 | * */
141 | initCode?: BytesLike;
142 | /**
143 | * BloctoSDK do not need signature to send userOperation. Will be ignored.
144 | * */
145 | signature?: BytesLike;
146 | }
147 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/lib/storage/storage.ts:
--------------------------------------------------------------------------------
1 | import MemoryStorage, { isStorageSupported } from './memoryStorage';
2 | import {
3 | LOGIN_PERSISTING_TIME,
4 | SDK_VERSION,
5 | KEY_SESSION,
6 | CHAIN,
7 | } from '../../constants';
8 |
9 | export interface ProviderSession {
10 | code?: string | null;
11 | accounts: Partial>;
12 | evm: {
13 | [chainName: string]: string[] | undefined;
14 | };
15 | }
16 |
17 | export interface AccountStorage {
18 | expiry: number;
19 | v: string;
20 | data: ProviderSession;
21 | }
22 |
23 | const storage = isStorageSupported() ? window.sessionStorage : MemoryStorage;
24 |
25 | export const getItem = (
26 | key: string,
27 | defaultValue: T | null = null
28 | ): T | null => {
29 | const value = storage.getItem(key);
30 | try {
31 | return (value && JSON.parse(value)) || defaultValue;
32 | } catch (SyntaxError) {
33 | return (value as T) || defaultValue;
34 | }
35 | };
36 |
37 | export const getRawItem = (key: string): string | null => storage.getItem(key);
38 |
39 | export const setItem = (key: string, value: unknown): void =>
40 | storage.setItem(
41 | key,
42 | typeof value === 'string' ? value : JSON.stringify(value)
43 | );
44 |
45 | export const removeItem = (key: string): void => {
46 | setItem(key, ''); // Due to some versions of browser bug can't removeItem correctly.
47 | storage.removeItem(key);
48 | };
49 |
50 | /**
51 | * @param {KEY_SESSION} key - key to retrieve the data
52 | * @returns {ProviderSession | null} ProviderSession | null
53 | * @description
54 | * Get ProviderSession from storage.
55 | * If the data is expired, will remove the data and return null
56 | */
57 | export const getAccountStorage = (key: KEY_SESSION): ProviderSession | null => {
58 | const rawAccountStorage = getItem(key, null);
59 | if (!rawAccountStorage) return null;
60 |
61 | // compare the expiry time of the item with the current time
62 | if (
63 | new Date().getTime() > rawAccountStorage.expiry ||
64 | rawAccountStorage.v !== SDK_VERSION
65 | ) {
66 | removeItem(key);
67 | return null;
68 | }
69 |
70 | return rawAccountStorage?.data;
71 | };
72 |
73 | /**
74 | @param {KEY_SESSION} key - key to store the data
75 | @param {ProviderSession} data - Only the part of ProviderSession that needs to be updated
76 | @param {number} expiry - expiry time of the data
77 | */
78 | export const setAccountStorage = (
79 | key: KEY_SESSION,
80 | data: Partial,
81 | expiry?: number
82 | ): void => {
83 | const rawAccountStorage = getItem(key);
84 | const newAccountStorage: AccountStorage = {
85 | data: {
86 | code: data?.code || rawAccountStorage?.data?.code,
87 | accounts: {
88 | ...rawAccountStorage?.data?.accounts,
89 | ...data?.accounts,
90 | },
91 | evm: {
92 | ...rawAccountStorage?.data?.evm,
93 | ...data?.evm,
94 | },
95 | },
96 | expiry:
97 | expiry ||
98 | rawAccountStorage?.expiry ||
99 | new Date().getTime() + LOGIN_PERSISTING_TIME,
100 | v: SDK_VERSION,
101 | };
102 | setItem(key, newAccountStorage);
103 | return;
104 | };
105 |
106 | export const getChainAddress = (
107 | key: KEY_SESSION,
108 | chain: CHAIN
109 | ): string[] | null => {
110 | if (!getAccountStorage(key)?.code) {
111 | removeItem(key);
112 | return null;
113 | }
114 | return getAccountStorage(key)?.accounts?.[chain] || null;
115 | };
116 |
117 | export const setChainAddress = (
118 | key: KEY_SESSION,
119 | chain: CHAIN,
120 | account: string[]
121 | ): void => {
122 | setAccountStorage(key, { accounts: { [chain]: account } });
123 | return;
124 | };
125 |
126 | export const removeChainAddress = (key: KEY_SESSION, chain: string): void => {
127 | setAccountStorage(key, { accounts: { [chain]: undefined } });
128 | return;
129 | };
130 |
131 | export const getEvmAddress = (
132 | key: KEY_SESSION,
133 | chain: string
134 | ): string[] | null => {
135 | if (!getAccountStorage(key)?.code) {
136 | removeItem(key);
137 | return null;
138 | }
139 | return getAccountStorage(key)?.evm?.[chain] || null;
140 | };
141 |
142 | export const setEvmAddress = (
143 | key: KEY_SESSION,
144 | chain: string,
145 | accounts: string[]
146 | ): void => {
147 | setAccountStorage(key, { evm: { [chain]: accounts } });
148 | return;
149 | };
150 |
151 | export const removeAllEvmAddress = (key: KEY_SESSION): void => {
152 | const newAccountStorage = getItem(key);
153 | if (!newAccountStorage) return;
154 | newAccountStorage.data.evm = {};
155 | setItem(key, newAccountStorage);
156 | return;
157 | };
158 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/constants.ts:
--------------------------------------------------------------------------------
1 | export enum KEY_SESSION {
2 | prod = 'BLOCTO_SDK',
3 | dev = 'BLOCTO_SDK_DEV',
4 | staging = 'BLOCTO_SDK_STAGING',
5 | }
6 | export enum CHAIN {
7 | ETHEREUM = 'ethereum',
8 | APTOS = 'aptos',
9 | }
10 |
11 | export interface BaseConfig {
12 | appId?: string;
13 | }
14 |
15 | type Mapping = Record;
16 |
17 | /* eth series constants begin */
18 |
19 | export const ETH_RPC_LIST: Mapping = {
20 | // This is the list of public RPC endpoints that we known to be working
21 | // Used to help developers did not set up their own RPC endpoints
22 |
23 | // BSC mainnet
24 | 56: 'https://bsc-dataseed1.binance.org',
25 | // BSC testnet
26 | 97: 'https://data-seed-prebsc-1-s1.binance.org:8545',
27 |
28 | // Polygon Mainnet
29 | 137: 'https://polygon-rpc.com/',
30 | // Polygon Amoy Testnet
31 | 80002: 'https://rpc-amoy.polygon.technology/',
32 |
33 | // Avalanche Mainnet
34 | 43114: 'https://api.avax.network/ext/bc/C/rpc',
35 | // Avalanche Fuji Testnet
36 | 43113: 'https://api.avax-test.network/ext/bc/C/rpc',
37 |
38 | // Arbitrum Mainnet
39 | 42161: 'https://arb1.arbitrum.io/rpc',
40 | // Arbitrum Sepolia Testnet
41 | 421614: 'https://arbitrum-sepolia.blockpi.network/v1/rpc/public',
42 |
43 | // Optimism Mainnet
44 | 10: 'https://mainnet.optimism.io',
45 | // Optimism Sepolia Testnet
46 | 11155420: 'https://sepolia.optimism.io',
47 |
48 | // Base Mainnet
49 | 8453: 'https://mainnet.base.org',
50 | // Base Sepolia Testnet
51 | 84532: 'https://sepolia.base.org',
52 |
53 | // Zora
54 | 7777777: 'https://rpc.zora.energy',
55 | // Zora Sepolia Testnet
56 | 999999999: 'https://sepolia.rpc.zora.energy',
57 |
58 | // Scroll
59 | 534352: 'https://rpc.scroll.io',
60 | // Scroll Sepolia Testnet
61 | 534351: 'https://sepolia-rpc.scroll.io',
62 |
63 | // Linea
64 | 59144: 'https://rpc.linea.build',
65 |
66 | // zKatana Sepolia Testnet
67 | 1261120: 'https://rpc.startale.com/zkatana',
68 |
69 | // Blast
70 | 81457: 'https://rpc.blast.io',
71 | // Blast Sepolia Testnet
72 | 168587773: 'https://sepolia.blast.io',
73 |
74 | // Merlin Chain
75 | 4200: 'https://rpc.merlinchain.io',
76 | // Merlin Testnet
77 | 686868: 'https://testnet-rpc.merlinchain.io',
78 |
79 | // Story Testnet
80 | 1513: 'https://testnet.storyrpc.io',
81 | };
82 |
83 | export const ETH_ENV_WALLET_SERVER_MAPPING: Mapping = {
84 | prod: 'https://wallet-v2.blocto.app',
85 | staging: 'https://wallet-v2-staging.blocto.app',
86 | dev: 'https://wallet-v2-dev.blocto.app',
87 | };
88 |
89 | export const ETH_SESSION_KEY_MAPPING: Record = {
90 | prod: KEY_SESSION.prod,
91 | staging: KEY_SESSION.staging,
92 | dev: KEY_SESSION.dev,
93 | };
94 |
95 | /* eth series constants end */
96 |
97 | /* aptos constants begin */
98 |
99 | export const APT_SESSION_KEY_MAPPING: Record = {
100 | 1: KEY_SESSION.prod,
101 | 2: KEY_SESSION.dev,
102 | 3: KEY_SESSION.dev,
103 | 4: KEY_SESSION.dev,
104 | 5: KEY_SESSION.staging,
105 | };
106 |
107 | export const APT_CHAIN_ID_SERVER_MAPPING: Mapping = {
108 | // MAINNET
109 | 1: 'https://wallet-v2.blocto.app',
110 | // TESTNET
111 | 2: 'https://wallet-v2-dev.blocto.app',
112 | // DEVNET
113 | 3: 'https://wallet-v2-dev.blocto.app',
114 | // TESTING
115 | 4: 'https://wallet-v2-dev.blocto.app',
116 | // PREMAINNET
117 | 5: 'https://wallet-v2-staging.blocto.app',
118 | };
119 |
120 | export enum WalletAdapterNetwork {
121 | Mainnet = 'mainnet',
122 | Testnet = 'testnet',
123 | Devnet = 'devnet',
124 | Testing = 'testing',
125 | Premainnet = 'premainnet',
126 | }
127 |
128 | export const APT_CHAIN_ID_NAME_MAPPING: Record = {
129 | 1: WalletAdapterNetwork.Mainnet,
130 | 2: WalletAdapterNetwork.Testnet,
131 | 3: WalletAdapterNetwork.Devnet,
132 | 4: WalletAdapterNetwork.Testing,
133 | 5: WalletAdapterNetwork.Premainnet,
134 | };
135 |
136 | export const APT_CHAIN_ID_RPC_MAPPING: Mapping = {
137 | 1: 'https://fullnode.mainnet.aptoslabs.com/v1',
138 | 2: 'https://fullnode.testnet.aptoslabs.com/v1',
139 | 3: 'https://fullnode.devnet.aptoslabs.com/v1',
140 | 4: '',
141 | 5: 'https://premainnet.aptosdev.com/v1',
142 | };
143 |
144 | /* aptos constants end */
145 |
146 | export const EIP1193_EVENTS: Array = [
147 | 'connect',
148 | 'disconnect',
149 | 'message',
150 | 'chainChanged',
151 | 'accountsChanged',
152 | ];
153 |
154 | // Preserve login for 1 day
155 | export const LOGIN_PERSISTING_TIME = 86400 * 1000;
156 | export const DEFAULT_APP_ID = '00000000-0000-0000-0000-000000000000';
157 |
158 | // Will inject the version of the SDK by rollup versionInjector during build time
159 | export const SDK_VERSION = '[VI]{version}[/VI]';
160 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/__tests__/ethereum.test.ts:
--------------------------------------------------------------------------------
1 | import EthereumProvider from '../providers/ethereum';
2 | import { getEvmSupport } from '../lib/getEvmSupport';
3 | import { getEvmSupportList } from './fixtures/getEvmSupport';
4 | jest.mock('../lib/getEvmSupport');
5 |
6 | describe('Testing BloctoSDK ethereum provider initialization and network loading', () => {
7 | beforeEach(() => {
8 | (getEvmSupport as jest.Mock).mockResolvedValue(getEvmSupportList);
9 | });
10 |
11 | test('should initialize with unloadedNetwork', () => {
12 | const ethereumWithSwitchable = new EthereumProvider({
13 | defaultChainId: '0xaa36a7',
14 | switchableChains: [
15 | {
16 | chainId: '0xaa36a7',
17 | rpcUrls: ['https://ethereum-sepolia.blockpi.network/v1/rpc/public'],
18 | },
19 | {
20 | chainId: '0x61',
21 | rpcUrls: ['https://data-seed-prebsc-1-s1.binance.org:8545'],
22 | },
23 | ],
24 | });
25 |
26 | expect(ethereumWithSwitchable['_blocto'].unloadedNetwork).toBeDefined();
27 | expect(
28 | ethereumWithSwitchable['_blocto']?.unloadedNetwork?.[0].chainId
29 | ).toBe('0xaa36a7');
30 | expect(
31 | ethereumWithSwitchable['_blocto']?.unloadedNetwork?.[1].chainId
32 | ).toBe('0x61');
33 | expect(ethereumWithSwitchable['_blocto'].unloadedNetwork?.length).toBe(2);
34 | });
35 |
36 | test('get support chain list', async () => {
37 | const ethereum = new EthereumProvider({
38 | chainId: '0xaa36a7',
39 | rpc: 'https://ethereum-sepolia.blockpi.network/v1/rpc/public',
40 | });
41 | const supportedChains = await ethereum.supportChainList();
42 | expect(supportedChains).toContainEqual({
43 | chainId: '11155111',
44 | chainName: 'Sepolia',
45 | });
46 | });
47 |
48 | test('should add chain and switch to it', async () => {
49 | const ethereum = new EthereumProvider({
50 | chainId: '0xaa36a7',
51 | rpc: 'https://ethereum-sepolia.blockpi.network/v1/rpc/public',
52 | });
53 | await expect(
54 | ethereum.request({
55 | method: 'wallet_switchEthereumChain',
56 | params: [{ chainId: '0x61' }],
57 | })
58 | ).rejects.toThrow(
59 | 'Unrecognized chain ID "97". Try adding the chain using wallet_addEthereumChain first.'
60 | );
61 |
62 | await ethereum.request({
63 | method: 'wallet_addEthereumChain',
64 | params: [
65 | {
66 | chainId: '0x61',
67 | rpcUrls: ['https://data-seed-prebsc-1-s1.binance.org:8545'],
68 | },
69 | ],
70 | });
71 | await ethereum.request({
72 | method: 'wallet_switchEthereumChain',
73 | params: [{ chainId: '0x61' }],
74 | });
75 | expect(ethereum.chainId).toBe('0x61');
76 | });
77 |
78 | test('create sdk instance with switchableChains and switch to it', async () => {
79 | const ethereum = new EthereumProvider({
80 | defaultChainId: '0xaa36a7',
81 | switchableChains: [
82 | {
83 | chainId: '0xaa36a7',
84 | rpcUrls: ['https://ethereum-sepolia.blockpi.network/v1/rpc/public'],
85 | },
86 | {
87 | chainId: '0x61',
88 | rpcUrls: ['https://data-seed-prebsc-1-s1.binance.org:8545'],
89 | },
90 | ],
91 | });
92 | await ethereum.request({
93 | method: 'wallet_switchEthereumChain',
94 | params: [{ chainId: '0x61' }],
95 | });
96 | expect(ethereum.chainId).toBe('0x61');
97 | });
98 | test('create sdk instance with switchableChains and call eth_accounts', async () => {
99 | const ethereum = new EthereumProvider({
100 | defaultChainId: '0xaa36a7',
101 | switchableChains: [
102 | {
103 | chainId: '0xaa36a7',
104 | rpcUrls: ['https://ethereum-sepolia.blockpi.network/v1/rpc/public'],
105 | },
106 | {
107 | chainId: '0x61',
108 | rpcUrls: ['https://data-seed-prebsc-1-s1.binance.org:8545'],
109 | },
110 | ],
111 | });
112 | // Trigger the loading of switchable networks
113 | await ethereum.request({
114 | method: 'eth_accounts',
115 | });
116 | expect(ethereum.chainId).toBe('0xaa36a7');
117 | // should remove unloadedNetwork after loading
118 | expect(ethereum['_blocto'].unloadedNetwork).toBeUndefined();
119 | });
120 |
121 | test('should not call loadSwitchableNetwork if unloadedNetwork is empty', async () => {
122 | const ethereum = new EthereumProvider({
123 | chainId: '0xaa36a7',
124 | rpc: 'https://ethereum-sepolia.blockpi.network/v1/rpc/public',
125 | });
126 | const loadSwitchableNetworkSpy = jest.spyOn(ethereum, 'loadSwitchableNetwork');
127 |
128 | await ethereum.request({ method: 'eth_accounts' });
129 |
130 | expect(loadSwitchableNetworkSpy).not.toHaveBeenCalled();
131 | });
132 | });
133 |
--------------------------------------------------------------------------------
/adapters/web3modal-connector/CHANGELOG.md:
--------------------------------------------------------------------------------
1 | # @blocto/web3modal-connector
2 |
3 | ## 0.1.4
4 |
5 | ### Patch Changes
6 |
7 | - 267f154: fix(wagmi): correct typo with chainId
8 | - 48fa07a: fix: handle existedSDK handle window not defined
9 | - Updated dependencies [267f154]
10 | - Updated dependencies [48fa07a]
11 | - @blocto/wagmi-connector@1.3.1
12 |
13 | ## 0.1.4-beta.2
14 |
15 | ### Patch Changes
16 |
17 | - 267f154: fix(wagmi): correct typo with chainId
18 | - Updated dependencies [267f154]
19 | - @blocto/wagmi-connector@1.3.1-beta.2
20 |
21 | ## 0.1.4-beta.1
22 |
23 | ### Patch Changes
24 |
25 | - 48fa07a: fix: handle existedSDK handle window not defined
26 | - Updated dependencies [48fa07a]
27 | - @blocto/wagmi-connector@1.3.1-beta.1
28 |
29 | ## 0.1.4-beta.0
30 |
31 | ### Patch Changes
32 |
33 | - @blocto/wagmi-connector@1.3.1-beta.0
34 |
35 | ## 0.1.3
36 |
37 | ### Patch Changes
38 |
39 | - b9dac7d: support web3js v4
40 | - 8d0b5bf: enhance in-app-sdk event support
41 | - 00a3832: sendAsync can send another requests
42 | - Updated dependencies [b9dac7d]
43 | - Updated dependencies [8d0b5bf]
44 | - Updated dependencies [e075c22]
45 | - Updated dependencies [00a3832]
46 | - @blocto/wagmi-connector@1.3.0
47 |
48 | ## 0.1.3-beta.2
49 |
50 | ### Patch Changes
51 |
52 | - Updated dependencies [fce0e50]
53 | - @blocto/wagmi-connector@1.3.0-beta.2
54 |
55 | ## 0.1.3-beta.1
56 |
57 | ### Patch Changes
58 |
59 | - b9dac7d: support web3js v4
60 | - 00a3832: sendAsync can send another requests
61 | - Updated dependencies [b9dac7d]
62 | - Updated dependencies [00a3832]
63 | - @blocto/wagmi-connector@1.2.4-beta.1
64 |
65 | ## 0.1.3-beta.0
66 |
67 | ### Patch Changes
68 |
69 | - 8d0b5bf: enhance in-app-sdk event support
70 | - Updated dependencies [8d0b5bf]
71 | - @blocto/wagmi-connector@1.2.4-beta.0
72 |
73 | ## 0.1.2
74 |
75 | ### Patch Changes
76 |
77 | - 4835522: Fix export unreconize type for old typescript
78 | - Updated dependencies [4835522]
79 | - @blocto/wagmi-connector@1.2.2
80 |
81 | ## 0.1.2-beta.0
82 |
83 | ### Patch Changes
84 |
85 | - 4835522: Fix export unreconize type for old typescript
86 | - Updated dependencies [4835522]
87 | - @blocto/wagmi-connector@1.2.2-beta.0
88 |
89 | ## 0.1.1
90 |
91 | ### Patch Changes
92 |
93 | - 2515a8f: Fix wrongly disconnect chain when account not changed
94 | - b40fbed: Fix skip calling enable() when request wallet_disconnect method
95 | - 4c769f9: Clear error message when add not supported chain
96 | - bbf2160: Make loadSwitchableNetwork rpcUrls optional
97 | - a3b0243: Seperate evm accounts from other chains in storage data structure
98 | - Updated dependencies [2515a8f]
99 | - Updated dependencies [b40fbed]
100 | - Updated dependencies [4c769f9]
101 | - Updated dependencies [bbf2160]
102 | - Updated dependencies [a3b0243]
103 | - @blocto/wagmi-connector@1.2.1
104 |
105 | ## 0.1.1-beta.1
106 |
107 | ### Patch Changes
108 |
109 | - 2515a8f: Fix wrongly disconnect chain when account not changed
110 | - Updated dependencies [2515a8f]
111 | - @blocto/wagmi-connector@1.2.1-beta.1
112 |
113 | ## 0.1.1-beta.0
114 |
115 | ### Patch Changes
116 |
117 | - b40fbed: Fix skip calling enable() when request wallet_disconnect method
118 | - 4c769f9: Clear error message when add not supported chain
119 | - bbf2160: Make loadSwitchableNetwork rpcUrls optional
120 | - a3b0243: Seperate evm accounts from other chains in storage data structure
121 | - Updated dependencies [b40fbed]
122 | - Updated dependencies [4c769f9]
123 | - Updated dependencies [bbf2160]
124 | - Updated dependencies [a3b0243]
125 | - @blocto/wagmi-connector@1.2.1-beta.0
126 |
127 | ## 0.1.0
128 |
129 | ### Minor Changes
130 |
131 | - df576fe: - Adding Package `@blocto/web3modal-connector` to replace Package `@blocto/wagmi-connector`'s support for web3Modal
132 | - Removing `BloctoWeb3ModalConfig` exported by Package `@blocto/wagmi-connector`
133 | - Removing `BloctoOptions`'s deprecate params `chainId, rpc` from `@blocto/wagmi-connector`
134 |
135 | ### Patch Changes
136 |
137 | - Updated dependencies [21097d9]
138 | - Updated dependencies [83eb33d]
139 | - Updated dependencies [e0e8fb7]
140 | - Updated dependencies [55f3395]
141 | - Updated dependencies [3ceb547]
142 | - Updated dependencies [21097d9]
143 | - Updated dependencies [df576fe]
144 | - Updated dependencies [21097d9]
145 | - Updated dependencies [21097d9]
146 | - Updated dependencies [21097d9]
147 | - @blocto/wagmi-connector@1.2.0
148 |
149 | ## 0.1.0-beta.0
150 |
151 | ### Minor Changes
152 |
153 | - bc5cc4b: - Adding Package `@blocto/web3modal-connector` to replace Package `@blocto/wagmi-connector`'s support for web3Modal
154 | - Removing `BloctoWeb3ModalConfig` exported by Package `@blocto/wagmi-connector`
155 | - Removing `BloctoOptions`'s deprecate params `chainId, rpc` from `@blocto/wagmi-connector`
156 |
157 | ### Patch Changes
158 |
159 | - Updated dependencies [bc5cc4b]
160 | - @blocto/wagmi-connector@1.2.0-beta.8
161 |
--------------------------------------------------------------------------------
/adapters/wagmi-connector/src/connector.ts:
--------------------------------------------------------------------------------
1 | import type {
2 | EthereumProviderConfig as BloctoEthereumProviderParameters,
3 | EthereumProviderInterface as BloctoProvider,
4 | } from '@blocto/sdk';
5 | import BloctoSDK from '@blocto/sdk';
6 | import { createConnector, normalizeChainId } from '@wagmi/core';
7 | import {
8 | RpcError,
9 | SwitchChainError,
10 | UserRejectedRequestError,
11 | getAddress,
12 | numberToHex,
13 | } from 'viem';
14 |
15 | export type BloctoParameters = {
16 | /**
17 | * Your app’s unique identifier that can be obtained at https://developers.blocto.app,
18 | * To get advanced features and support with Blocto.
19 | *
20 | * https://docs.blocto.app/blocto-sdk/register-app-id
21 | */
22 | appId?: string;
23 | };
24 |
25 | blocto.type = 'blocto' as const;
26 | export function blocto({ appId }: BloctoParameters = {}) {
27 | // eslint-disable-next-line @typescript-eslint/ban-types
28 | type Properties = {};
29 | type StorageItem = {
30 | store: any;
31 | 'wagmi.recentConnectorId': string;
32 | };
33 |
34 | let walletProvider: BloctoProvider | undefined;
35 | const handleConnectReset = () => {
36 | walletProvider = undefined;
37 | };
38 |
39 | return createConnector((config) => ({
40 | id: 'blocto',
41 | name: 'Blocto',
42 | type: blocto.type,
43 | async connect({ chainId } = {}) {
44 | try {
45 | const provider = await this.getProvider({ chainId });
46 |
47 | config.emitter.emit('message', { type: 'connecting' });
48 |
49 | await provider.request({
50 | method: 'eth_requestAccounts',
51 | });
52 |
53 | const accounts = await this.getAccounts();
54 | const _chainId = await this.getChainId();
55 |
56 | return { accounts, chainId: _chainId };
57 | } catch (error: unknown) {
58 | handleConnectReset();
59 | throw error;
60 | }
61 | },
62 | async disconnect() {
63 | const provider = await this.getProvider();
64 | await provider.request({ method: 'wallet_disconnect' });
65 | handleConnectReset();
66 | },
67 | async getAccounts() {
68 | const provider = await this.getProvider();
69 | const accounts = (await provider.request({
70 | method: 'eth_accounts',
71 | })) as string[];
72 |
73 | return accounts.map((x) => getAddress(x));
74 | },
75 | async getChainId() {
76 | const provider = await this.getProvider();
77 | const chainId = await provider?.request({ method: 'eth_chainId' });
78 | return normalizeChainId(chainId);
79 | },
80 | async getProvider({ chainId } = {}) {
81 | if (!walletProvider) {
82 | const store = await config.storage?.getItem('store');
83 | const lastConnectedChainId = store?.state?.chainId;
84 | const desiredChainId = chainId ?? lastConnectedChainId;
85 | const ethereum: BloctoEthereumProviderParameters = {
86 | chainId: desiredChainId,
87 | rpc: config.chains.find((x) => x.id === desiredChainId)?.rpcUrls
88 | .default.http[0],
89 | };
90 |
91 | walletProvider = new BloctoSDK({ ethereum, appId })?.ethereum;
92 | if (!walletProvider) {
93 | throw new Error('Blocto SDK is not initialized.');
94 | }
95 |
96 | walletProvider.on('accountsChanged', this.onAccountsChanged.bind(this));
97 | walletProvider.on('chainChanged', this.onChainChanged.bind(this));
98 | walletProvider.on('disconnect', this.onDisconnect.bind(this));
99 | }
100 |
101 | return Promise.resolve(walletProvider);
102 | },
103 | async isAuthorized() {
104 | const recentConnectorId = await config.storage?.getItem(
105 | 'recentConnectorId'
106 | );
107 | if (recentConnectorId !== this.id) return false;
108 |
109 | const accounts = await this.getAccounts();
110 | return !!accounts.length;
111 | },
112 | async switchChain({ chainId }) {
113 | try {
114 | const provider = await this.getProvider();
115 | const id = numberToHex(chainId);
116 | const chain = config.chains.find((x) => x.id === chainId);
117 | const networks = await provider.supportChainList();
118 | const evmSupportMap = networks.reduce(
119 | (a: any, v: any) => ({ ...a, [v.chainId]: v }),
120 | {}
121 | );
122 | const isBloctoSupportChain = evmSupportMap[`${chainId}`];
123 |
124 | if (!chain) {
125 | throw new SwitchChainError(new Error(`Chain not in config: ${id}`));
126 | }
127 |
128 | if (!isBloctoSupportChain) {
129 | throw new SwitchChainError(
130 | new Error(`Blocto unsupported chain: ${id}`)
131 | );
132 | }
133 |
134 | await provider.request({
135 | method: 'wallet_addEthereumChain',
136 | params: [{ chainId: id, rpcUrls: chain?.rpcUrls.default.http }],
137 | });
138 | await provider.request({
139 | method: 'wallet_switchEthereumChain',
140 | params: [{ chainId: id }],
141 | });
142 |
143 | return chain;
144 | } catch (err) {
145 | const error = err as RpcError;
146 | if (error.code === UserRejectedRequestError.code)
147 | throw new UserRejectedRequestError(error);
148 |
149 | throw new SwitchChainError(error as Error);
150 | }
151 | },
152 | // eslint-disable-next-line @typescript-eslint/no-empty-function
153 | onAccountsChanged() {},
154 | async onChainChanged(chainId: string) {
155 | const accounts = await this.getAccounts();
156 | config.emitter.emit('change', {
157 | chainId: normalizeChainId(chainId),
158 | accounts,
159 | });
160 | },
161 | async onDisconnect() {
162 | config.emitter.emit('disconnect');
163 | },
164 | }));
165 | }
166 |
167 | export { createConnector };
--------------------------------------------------------------------------------
/adapters/aptos-wallet-adapter-plugin/src/index.ts:
--------------------------------------------------------------------------------
1 | import type { AptosProviderInterface as IBloctoAptos } from '@blocto/sdk';
2 | import BloctoSDK from '@blocto/sdk';
3 | import type {
4 | AccountInfo,
5 | AdapterPlugin,
6 | NetworkInfo,
7 | SignMessagePayload,
8 | SignMessageResponse,
9 | WalletName,
10 | } from '@aptos-labs/wallet-adapter-core';
11 | import { NetworkName, WalletReadyState } from '@aptos-labs/wallet-adapter-core';
12 | import type { Types } from 'aptos';
13 |
14 | interface BloctoWindow extends Window {
15 | bloctoAptos?: IBloctoAptos;
16 | }
17 |
18 | declare const window: BloctoWindow;
19 |
20 | export const BloctoWalletName = 'Blocto' as WalletName<'Blocto'>;
21 |
22 | export interface BloctoWalletAdapterConfig {
23 | network?: NetworkName.Mainnet | NetworkName.Testnet;
24 | bloctoAppId: string;
25 | }
26 |
27 | export const APTOS_NETWORK_CHAIN_ID_MAPPING = {
28 | // MAINNET
29 | [NetworkName.Mainnet]: 1,
30 | // TESTNET
31 | [NetworkName.Testnet]: 2,
32 | };
33 |
34 | export class BloctoWallet implements AdapterPlugin {
35 | readonly name = BloctoWalletName;
36 | readonly url = 'https://blocto.app';
37 | readonly icon =
38 | '';
39 |
40 | readonly providerName = 'bloctoAptos';
41 |
42 | provider: IBloctoAptos | undefined =
43 | typeof window !== 'undefined' ? window.bloctoAptos : undefined;
44 |
45 | readyState?: WalletReadyState = WalletReadyState.Loadable;
46 |
47 | protected _network: NetworkName.Mainnet | NetworkName.Testnet;
48 |
49 | constructor({
50 | network = NetworkName.Mainnet,
51 | bloctoAppId,
52 | }: BloctoWalletAdapterConfig) {
53 | const sdk = new BloctoSDK({
54 | aptos: {
55 | chainId: APTOS_NETWORK_CHAIN_ID_MAPPING[network],
56 | },
57 | appId: bloctoAppId,
58 | });
59 |
60 | this.provider = sdk.aptos;
61 | this._network = network;
62 | }
63 |
64 | async connect(): Promise {
65 | const accountInfo = await this.provider?.connect();
66 | if (!accountInfo) throw `${BloctoWalletName} Address Info Error`;
67 | if (!accountInfo.address) throw `${BloctoWalletName} address null`;
68 | if (!accountInfo.publicKey) throw `${BloctoWalletName} publicKey null`;
69 | if (!accountInfo.minKeysRequired)
70 | throw `${BloctoWalletName} minKeysRequired null`;
71 | return {
72 | address: accountInfo.address,
73 | publicKey: accountInfo.publicKey,
74 | minKeysRequired: accountInfo.minKeysRequired,
75 | };
76 | }
77 |
78 | async account(): Promise {
79 | const response = await this.provider?.publicAccount;
80 | if (!response) throw `${BloctoWalletName} Account Error`;
81 | if (!response.address) throw `${BloctoWalletName} address null`;
82 | if (!response.publicKey) throw `${BloctoWalletName} publicKey null`;
83 | if (!response.minKeysRequired)
84 | throw `${BloctoWalletName} minKeysRequired null`;
85 | return {
86 | address: response.address,
87 | publicKey: response.publicKey,
88 | minKeysRequired: response.minKeysRequired,
89 | };
90 | }
91 |
92 | async disconnect(): Promise {
93 | await this.provider?.disconnect();
94 | }
95 |
96 | async signAndSubmitTransaction(
97 | transaction: Types.TransactionPayload,
98 | options?: any
99 | ): Promise<{ hash: Types.HexEncodedBytes }> {
100 | const provider = this.provider;
101 | const response = await provider?.signAndSubmitTransaction(
102 | transaction,
103 | options
104 | );
105 | if (response) {
106 | return { hash: response.hash };
107 | } else {
108 | throw new Error('Transaction failed');
109 | }
110 | }
111 |
112 | async signMessage(message: SignMessagePayload): Promise {
113 | if (typeof message !== 'object' || !message.nonce) {
114 | `${BloctoWalletName} Invalid signMessage Payload`;
115 | }
116 | const response = await this.provider?.signMessage(message);
117 | if (response) {
118 | return response;
119 | } else {
120 | throw `${BloctoWalletName} Sign Message failed`;
121 | }
122 | }
123 |
124 | async network(): Promise {
125 | const response = await this.provider?.network();
126 | if (!response) throw `${BloctoWalletName} Network Error`;
127 | const name = response.name as unknown;
128 | return {
129 | name: name as NetworkName,
130 | chainId: response.chainId,
131 | };
132 | }
133 |
134 | async onNetworkChange(): Promise {
135 | // not supported yet
136 | return Promise.resolve();
137 | }
138 |
139 | async onAccountChange(): Promise {
140 | // not supported yet
141 | return Promise.resolve();
142 | }
143 | }
144 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | /* Basic Options */
4 | // "incremental": true, /* Enable incremental compilation */
5 | "target": "ES6" /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019', 'ES2020', 'ES2021', or 'ESNEXT'. */,
6 | // "lib": [], /* Specify library files to be included in the compilation. */
7 | // "allowJs": true, /* Allow javascript files to be compiled. */
8 | // "checkJs": true, /* Report errors in .js files. */
9 | // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', 'react', 'react-jsx' or 'react-jsxdev'. */
10 | "declaration": true,
11 | "declarationDir": "./dist/types" /* Generates corresponding '.d.ts' file. */,
12 | // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */
13 | // "sourceMap": true, /* Generates corresponding '.map' file. */
14 | // "outFile": "./", /* Concatenate and emit output to single file. */
15 | // "outDir": "./", /* Redirect output structure to the directory. */
16 | // "rootDir": "./", /* Specify the root directory of input files. Use to control the output directory structure with --outDir. */
17 | // "composite": true, /* Enable project compilation */
18 | // "tsBuildInfoFile": "./", /* Specify file to store incremental compilation information */
19 | // "removeComments": true, /* Do not emit comments to output. */
20 | // "noEmit": true, /* Do not emit outputs. */
21 | // "importHelpers": true, /* Import emit helpers from 'tslib'. */
22 | // "downlevelIteration": true, /* Provide full support for iterables in 'for-of', spread, and destructuring when targeting 'ES5' or 'ES3'. */
23 | // "isolatedModules": true, /* Transpile each file as a separate module (similar to 'ts.transpileModule'). */
24 |
25 | /* Strict Type-Checking Options */
26 | "strict": true /* Enable all strict type-checking options. */,
27 | // "noImplicitAny": true, /* Raise error on expressions and declarations with an implied 'any' type. */
28 | // "strictNullChecks": true, /* Enable strict null checks. */
29 | // "strictFunctionTypes": true, /* Enable strict checking of function types. */
30 | // "strictBindCallApply": true, /* Enable strict 'bind', 'call', and 'apply' methods on functions. */
31 | // "strictPropertyInitialization": true, /* Enable strict checking of property initialization in classes. */
32 | // "noImplicitThis": true, /* Raise error on 'this' expressions with an implied 'any' type. */
33 | // "alwaysStrict": true, /* Parse in strict mode and emit "use strict" for each source file. */
34 |
35 | /* Additional Checks */
36 | // "noUnusedLocals": true, /* Report errors on unused locals. */
37 | // "noUnusedParameters": true, /* Report errors on unused parameters. */
38 | // "noImplicitReturns": true, /* Report error when not all code paths in function return a value. */
39 | // "noFallthroughCasesInSwitch": true, /* Report errors for fallthrough cases in switch statement. */
40 | // "noUncheckedIndexedAccess": true, /* Include 'undefined' in index signature results */
41 | // "noImplicitOverride": true, /* Ensure overriding members in derived classes are marked with an 'override' modifier. */
42 | // "noPropertyAccessFromIndexSignature": true, /* Require undeclared properties from index signatures to use element accesses. */
43 |
44 | /* Module Resolution Options */
45 | "moduleResolution": "node" /* Specify module resolution strategy: 'node' (Node.js) or 'classic' (TypeScript pre-1.6). */,
46 | // "baseUrl": "./", /* Base directory to resolve non-absolute module names. */
47 | // "paths": {}, /* A series of entries which re-map imports to lookup locations relative to the 'baseUrl'. */
48 | // "rootDirs": [], /* List of root folders whose combined content represents the structure of the project at runtime. */
49 | // "typeRoots": [], /* List of folders to include type definitions from. */
50 | // "types": [], /* Type declaration files to be included in compilation. */
51 | // "allowSyntheticDefaultImports": true, /* Allow default imports from modules with no default export. This does not affect code emit, just typechecking. */
52 | "esModuleInterop": true /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */,
53 | // "preserveSymlinks": true, /* Do not resolve the real path of symlinks. */
54 | // "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */
55 |
56 | /* Source Map Options */
57 | // "sourceRoot": "", /* Specify the location where debugger should locate TypeScript files instead of source locations. */
58 | // "mapRoot": "", /* Specify the location where debugger should locate map files instead of generated locations. */
59 | // "inlineSourceMap": true, /* Emit a single file with source maps instead of having a separate file. */
60 | // "inlineSources": true, /* Emit the source alongside the sourcemaps within a single file; requires '--inlineSourceMap' or '--sourceMap' to be set. */
61 |
62 | /* Experimental Options */
63 | // "experimentalDecorators": true, /* Enables experimental support for ES7 decorators. */
64 | // "emitDecoratorMetadata": true, /* Enables experimental support for emitting type metadata for decorators. */
65 |
66 | /* Advanced Options */
67 | "skipLibCheck": true /* Skip type checking of declaration files. */,
68 | "forceConsistentCasingInFileNames": true /* Disallow inconsistently-cased references to the same file. */
69 | },
70 | "include": ["./src/**/*.ts"]
71 | }
72 |
--------------------------------------------------------------------------------
/adapters/connectkit-connector/CHANGELOG.md:
--------------------------------------------------------------------------------
1 | # @blocto/connectkit-connector
2 |
3 | ## 0.2.3
4 |
5 | ### Patch Changes
6 |
7 | - 267f154: fix(wagmi): correct typo with chainId
8 | - 48fa07a: fix: handle existedSDK handle window not defined
9 | - Updated dependencies [267f154]
10 | - Updated dependencies [48fa07a]
11 | - @blocto/wagmi-connector@1.3.1
12 |
13 | ## 0.2.3-beta.2
14 |
15 | ### Patch Changes
16 |
17 | - 267f154: fix(wagmi): correct typo with chainId
18 | - Updated dependencies [267f154]
19 | - @blocto/wagmi-connector@1.3.1-beta.2
20 |
21 | ## 0.2.3-beta.1
22 |
23 | ### Patch Changes
24 |
25 | - 48fa07a: fix: handle existedSDK handle window not defined
26 | - Updated dependencies [48fa07a]
27 | - @blocto/wagmi-connector@1.3.1-beta.1
28 |
29 | ## 0.2.3-beta.0
30 |
31 | ### Patch Changes
32 |
33 | - @blocto/wagmi-connector@1.3.1-beta.0
34 |
35 | ## 0.2.2
36 |
37 | ### Patch Changes
38 |
39 | - b9dac7d: support web3js v4
40 | - 8d0b5bf: enhance in-app-sdk event support
41 | - 00a3832: sendAsync can send another requests
42 | - Updated dependencies [b9dac7d]
43 | - Updated dependencies [8d0b5bf]
44 | - Updated dependencies [e075c22]
45 | - Updated dependencies [00a3832]
46 | - @blocto/wagmi-connector@1.3.0
47 |
48 | ## 0.2.2-beta.2
49 |
50 | ### Patch Changes
51 |
52 | - Updated dependencies [fce0e50]
53 | - @blocto/wagmi-connector@1.3.0-beta.2
54 |
55 | ## 0.2.2-beta.1
56 |
57 | ### Patch Changes
58 |
59 | - b9dac7d: support web3js v4
60 | - 00a3832: sendAsync can send another requests
61 | - Updated dependencies [b9dac7d]
62 | - Updated dependencies [00a3832]
63 | - @blocto/wagmi-connector@1.2.4-beta.1
64 |
65 | ## 0.2.2-beta.0
66 |
67 | ### Patch Changes
68 |
69 | - 8d0b5bf: enhance in-app-sdk event support
70 | - Updated dependencies [8d0b5bf]
71 | - @blocto/wagmi-connector@1.2.4-beta.0
72 |
73 | ## 0.2.1
74 |
75 | ### Patch Changes
76 |
77 | - 4835522: Fix export unreconize type for old typescript
78 | - Updated dependencies [4835522]
79 | - @blocto/wagmi-connector@1.2.2
80 |
81 | ## 0.2.1-beta.0
82 |
83 | ### Patch Changes
84 |
85 | - 4835522: Fix export unreconize type for old typescript
86 | - Updated dependencies [4835522]
87 | - @blocto/wagmi-connector@1.2.2-beta.0
88 |
89 | ## 0.2.0
90 |
91 | ### Minor Changes
92 |
93 | - 464a645: enhance the connecting experience with connectkit using @blocto/connectkit-connector
94 |
95 | ### Patch Changes
96 |
97 | - 2515a8f: Fix wrongly disconnect chain when account not changed
98 | - b40fbed: Fix skip calling enable() when request wallet_disconnect method
99 | - 4c769f9: Clear error message when add not supported chain
100 | - bbf2160: Make loadSwitchableNetwork rpcUrls optional
101 | - a3b0243: Seperate evm accounts from other chains in storage data structure
102 | - Updated dependencies [2515a8f]
103 | - Updated dependencies [b40fbed]
104 | - Updated dependencies [4c769f9]
105 | - Updated dependencies [bbf2160]
106 | - Updated dependencies [a3b0243]
107 | - @blocto/wagmi-connector@1.2.1
108 |
109 | ## 0.1.2-beta.1
110 |
111 | ### Patch Changes
112 |
113 | - 2515a8f: Fix wrongly disconnect chain when account not changed
114 | - Updated dependencies [2515a8f]
115 | - @blocto/sdk@0.7.0-beta.1
116 |
117 | ## 0.1.2-beta.0
118 |
119 | ### Patch Changes
120 |
121 | - b40fbed: Fix skip calling enable() when request wallet_disconnect method
122 | - 4c769f9: Clear error message when add not supported chain
123 | - bbf2160: Make loadSwitchableNetwork rpcUrls optional
124 | - a3b0243: Seperate evm accounts from other chains in storage data structure
125 | - Updated dependencies [b40fbed]
126 | - Updated dependencies [4c769f9]
127 | - Updated dependencies [bbf2160]
128 | - Updated dependencies [a3b0243]
129 | - Updated dependencies [1288bd1]
130 | - @blocto/sdk@0.7.0-beta.0
131 |
132 | ## 0.1.1
133 |
134 | ### Patch Changes
135 |
136 | - 21097d9: Fix switch chain is connected logic
137 | - 55f3395: Fix go login flow when switching to a different blocto server
138 | - 3ceb547: Fix skip switch chain if provide same id as current
139 | - 21097d9: Fix disconnect all evm chains when disconnect
140 | - 21097d9: Fix wrong switch chain url
141 | - 21097d9: Fix switch chain login logic
142 | - 21097d9: Fix emit disconnect event when switch chain approved but failed
143 | - Updated dependencies [21097d9]
144 | - Updated dependencies [e0e8fb7]
145 | - Updated dependencies [6e427b1]
146 | - Updated dependencies [34a23e4]
147 | - Updated dependencies [55f3395]
148 | - Updated dependencies [3ceb547]
149 | - Updated dependencies [21097d9]
150 | - Updated dependencies [21097d9]
151 | - Updated dependencies [21097d9]
152 | - Updated dependencies [21097d9]
153 | - Updated dependencies [494ded1]
154 | - @blocto/sdk@0.6.0
155 |
156 | ## 0.1.1-beta.7
157 |
158 | ### Patch Changes
159 |
160 | - 55f3395: Fix go login flow when switching to a different blocto server
161 | - Updated dependencies [6e427b1]
162 | - Updated dependencies [55f3395]
163 | - Updated dependencies [494ded1]
164 | - @blocto/sdk@0.6.0-beta.7
165 |
166 | ## 0.1.1-beta.6
167 |
168 | ### Patch Changes
169 |
170 | - 3ceb547: Fix skip switch chain if provide same id as current
171 | - Updated dependencies [34a23e4]
172 | - Updated dependencies [3ceb547]
173 | - @blocto/sdk@0.6.0-beta.6
174 |
175 | ## 0.1.1-beta.5
176 |
177 | ### Patch Changes
178 |
179 | - 39754e5: Fix disconnect all evm chains when disconnect
180 | - Updated dependencies [39754e5]
181 | - @blocto/sdk@0.6.0-beta.5
182 |
183 | ## 0.1.1-beta.4
184 |
185 | ### Patch Changes
186 |
187 | - 6acf18d: Fix switch chain is connected logic
188 | - Updated dependencies [6acf18d]
189 | - @blocto/sdk@0.6.0-beta.4
190 |
191 | ## 0.1.1-beta.3
192 |
193 | ### Patch Changes
194 |
195 | - 84d6727: Fix emit disconnect event when switch chain approved but failed
196 | - Updated dependencies [84d6727]
197 | - @blocto/sdk@0.6.0-beta.3
198 |
199 | ## 0.1.1-beta.2
200 |
201 | ### Patch Changes
202 |
203 | - a0cc54c: Fix switch chain login logic
204 | - Updated dependencies [a0cc54c]
205 | - @blocto/sdk@0.6.0-beta.2
206 |
207 | ## 0.1.1-beta.1
208 |
209 | ### Patch Changes
210 |
211 | - 0d5bfd5: Fix wrong switch chain url
212 | - Updated dependencies [0d5bfd5]
213 | - @blocto/sdk@0.6.0-beta.1
214 |
215 | ## 0.1.1-beta.0
216 |
217 | ### Patch Changes
218 |
219 | - Updated dependencies [e0e8fb7]
220 | - @blocto/sdk@0.6.0-beta.0
221 |
222 | ## 0.1.0
223 |
224 | ### Minor Changes
225 |
226 | - 9b82b57: add connectkit support
227 |
228 | ### Patch Changes
229 |
230 | - ac63830: wagmi three-shake config and rainbowkit use wagmi-connector
231 | - Updated dependencies [abd96b2]
232 | - Updated dependencies [fec7693]
233 | - Updated dependencies [dca87b8]
234 | - Updated dependencies [b9f1eca]
235 | - Updated dependencies [2cacc63]
236 | - @blocto/sdk@0.5.5
237 |
--------------------------------------------------------------------------------
/adapters/wagmi-connector/src/connector.test.ts:
--------------------------------------------------------------------------------
1 | import { expect, test, describe, vi, beforeEach, afterEach } from 'vitest';
2 | import { createConfig } from '@wagmi/core';
3 | import { polygonMumbai, arbitrumGoerli } from '@wagmi/chains';
4 | import { SwitchChainError, http, numberToHex } from 'viem';
5 | import { normalizeChainId } from '@wagmi/core';
6 | import { blocto } from './';
7 |
8 | const walletProvider = {
9 | on: vi.fn(),
10 | };
11 |
12 | describe('blocto-connector', () => {
13 | let connector: any;
14 | beforeEach(() => {
15 | const config = createConfig({
16 | chains: [polygonMumbai, arbitrumGoerli],
17 | pollingInterval: 100,
18 | storage: null,
19 | transports: {
20 | [polygonMumbai.id]: http(),
21 | [arbitrumGoerli.id]: http(),
22 | },
23 | });
24 | connector = config._internal.connectors.setup(blocto());
25 | });
26 |
27 | afterEach(() => {
28 | connector = null;
29 | });
30 |
31 | test('setup', () => {
32 | expect(connector.name).toEqual('Blocto');
33 | });
34 |
35 | test('connect', async () => {
36 | const chainId = 1;
37 | const accounts = ['0xc61B4Aa62E5FD40cceB08C602Eb5D157b257b49a'];
38 | const provider = {
39 | request: vi.fn().mockResolvedValue(accounts),
40 | };
41 | connector.getProvider = vi.fn().mockResolvedValue(provider);
42 | connector.getAccounts = vi.fn().mockResolvedValue(accounts);
43 | connector.getChainId = vi.fn().mockResolvedValue(chainId);
44 |
45 | const result = await connector.connect({ chainId });
46 |
47 | expect(result).toEqual({ accounts, chainId });
48 | expect(connector.getProvider).toHaveBeenCalledWith({ chainId });
49 | expect(provider.request).toHaveBeenCalledWith({
50 | method: 'eth_requestAccounts',
51 | });
52 | });
53 |
54 | test('catch error when user decline to connect', () => {
55 | const chainId = 1;
56 | const userRejectedRequest = new Error('User rejected request');
57 | const provider = {
58 | request: vi.fn().mockImplementation(() => {
59 | throw userRejectedRequest;
60 | }),
61 | };
62 | connector.getProvider = vi.fn().mockResolvedValue(provider);
63 |
64 | expect(connector.connect({ chainId })).rejects.toThrow(userRejectedRequest);
65 | });
66 |
67 | test('disconnect', async () => {
68 | const provider = {
69 | request: vi.fn().mockResolvedValue(undefined),
70 | };
71 | connector.getProvider = vi.fn().mockResolvedValue(provider);
72 |
73 | await connector.disconnect();
74 |
75 | expect(connector.getProvider).toHaveBeenCalled();
76 | expect(provider.request).toHaveBeenCalledWith({
77 | method: 'wallet_disconnect',
78 | });
79 | });
80 |
81 | test('getAccounts', async () => {
82 | const accounts = ['0xc61B4Aa62E5FD40cceB08C602Eb5D157b257b49a'];
83 | const provider = {
84 | request: vi.fn().mockResolvedValue(accounts),
85 | };
86 | connector.getProvider = vi.fn().mockResolvedValue(provider);
87 |
88 | const result = await connector.getAccounts();
89 |
90 | expect(result).toEqual(['0xc61B4Aa62E5FD40cceB08C602Eb5D157b257b49a']);
91 | expect(connector.getProvider).toHaveBeenCalled();
92 | expect(provider.request).toHaveBeenCalledWith({ method: 'eth_accounts' });
93 | });
94 |
95 | test('getChainId', async () => {
96 | const chainId = '0x1';
97 | const provider = {
98 | chainId: undefined,
99 | request: vi.fn().mockResolvedValue(chainId),
100 | };
101 | connector.getProvider = vi.fn().mockResolvedValue(provider);
102 |
103 | const result = await connector.getChainId();
104 |
105 | expect(result).toEqual(normalizeChainId(chainId));
106 | expect(connector.getProvider).toHaveBeenCalled();
107 | expect(provider.request).toHaveBeenCalledWith({ method: 'eth_chainId' });
108 | });
109 |
110 | test('getProvider', async () => {
111 | vi.mock('@blocto/sdk', () => ({
112 | default: vi.fn().mockImplementation(() => ({
113 | ethereum: walletProvider,
114 | })),
115 | }));
116 |
117 | const chainId = 1;
118 | const result = await connector.getProvider({ chainId });
119 |
120 | expect(result).toEqual(walletProvider);
121 | expect(walletProvider.on).toHaveBeenCalledWith(
122 | 'accountsChanged',
123 | expect.any(Function)
124 | );
125 | expect(walletProvider.on).toHaveBeenCalledWith(
126 | 'chainChanged',
127 | expect.any(Function)
128 | );
129 | expect(walletProvider.on).toHaveBeenCalledWith(
130 | 'disconnect',
131 | expect.any(Function)
132 | );
133 | });
134 | test('isAuthorized', async () => {
135 | const accounts = ['0xc61B4Aa62E5FD40cceB08C602Eb5D157b257b49a'];
136 | connector.getAccounts = vi.fn().mockResolvedValue(accounts);
137 |
138 | const result = await connector.isAuthorized();
139 |
140 | expect(result).toEqual(false);
141 | });
142 |
143 | test('switchChain', async () => {
144 | const chainId = arbitrumGoerli.id;
145 | const provider = {
146 | request: vi.fn().mockResolvedValue(undefined),
147 | supportChainList: vi.fn().mockResolvedValue(
148 | [polygonMumbai, arbitrumGoerli].map(({ id, name }) => ({
149 | chainId: id,
150 | chainName: name,
151 | }))
152 | ),
153 | };
154 | connector.getProvider = vi.fn().mockResolvedValue(provider);
155 |
156 | const chain = await connector.switchChain({ chainId });
157 |
158 | expect(connector.getProvider).toHaveBeenCalled();
159 | expect(provider.request).toHaveBeenCalledWith({
160 | method: 'wallet_addEthereumChain',
161 | params: [
162 | {
163 | chainId: numberToHex(chainId),
164 | rpcUrls: arbitrumGoerli.rpcUrls.default.http,
165 | },
166 | ],
167 | });
168 | expect(provider.request).toHaveBeenCalledWith({
169 | method: 'wallet_switchEthereumChain',
170 | params: [{ chainId: numberToHex(chainId) }],
171 | });
172 | expect(chain.id).toEqual(chainId);
173 | });
174 |
175 | test('catch error when switching to unConfigured chain', async () => {
176 | const unConfiguredChainId = 111111;
177 | const provider = {
178 | request: vi.fn().mockResolvedValue(undefined),
179 | supportChainList: vi.fn().mockResolvedValue(
180 | [polygonMumbai, arbitrumGoerli].map(({ id, name }) => ({
181 | chainId: id,
182 | chainName: name,
183 | }))
184 | ),
185 | };
186 | connector.getProvider = vi.fn().mockResolvedValue(provider);
187 | const expectError = new SwitchChainError(
188 | new Error(`Chain not in config: ${numberToHex(unConfiguredChainId)}`)
189 | );
190 |
191 | expect(
192 | connector.switchChain({ chainId: unConfiguredChainId })
193 | ).rejects.toThrow(expectError);
194 | });
195 |
196 | test('catch error when switching to blocto unsupported chain', async () => {
197 | const unsupportedChainId = arbitrumGoerli.id;
198 | const provider = {
199 | request: vi.fn().mockResolvedValue(undefined),
200 | supportChainList: vi.fn().mockResolvedValue(
201 | [polygonMumbai].map(({ id, name }) => ({
202 | chainId: id,
203 | chainName: name,
204 | }))
205 | ),
206 | };
207 | connector.getProvider = vi.fn().mockResolvedValue(provider);
208 | const expectError = new SwitchChainError(
209 | new Error(`Blocto unsupported chain: ${numberToHex(unsupportedChainId)}`)
210 | );
211 |
212 | expect(
213 | connector.switchChain({ chainId: unsupportedChainId })
214 | ).rejects.toThrow(expectError);
215 | });
216 | });
217 |
--------------------------------------------------------------------------------
/packages/dappauth/test/unit/index.test.js:
--------------------------------------------------------------------------------
1 | import { privateToPublic, stripHexPrefix } from 'ethereumjs-util';
2 | import { equal } from 'assert';
3 | import DappAuth from '../../src/index.js';
4 | import { removeHexPrefix } from '../../src/utils/index.js';
5 | import ProviderMock from './provider-mock.js';
6 | import ContractMock from './contract-mock.js';
7 | import { generateRandomKey, keyToAddress, signEOAPersonalMessage, signERC1654PersonalMessage } from './test-utils.js';
8 |
9 | describe('DappAuth', function() {
10 | const keyA = generateRandomKey();
11 | const keyB = generateRandomKey();
12 | const keyC = generateRandomKey();
13 |
14 | const testCases = [
15 | {
16 | title: 'External wallets should be authorized signers over their address',
17 | isEOA: true,
18 | challenge: 'foo',
19 | challengeSign: 'foo',
20 | signingKeys: [keyA],
21 | authAddr: keyToAddress(keyA),
22 | mockContract: {
23 | authorizedKey: null,
24 | address: null,
25 | errorIsValidSignature: false,
26 | },
27 | expectedAuthorizedSignerError: false,
28 | expectedAuthorizedSigner: true,
29 | },
30 | {
31 | title:
32 | 'External wallets should NOT be authorized signers when signing the wrong challenge',
33 | isEOA: true,
34 | challenge: 'foo',
35 | challengeSign: 'bar',
36 | signingKeys: [keyA],
37 | authAddr: keyToAddress(keyA),
38 | mockContract: {
39 | authorizedKey: privateToPublic(keyC),
40 | address: keyToAddress(keyA),
41 | errorIsValidSignature: false,
42 | },
43 | expectedAuthorizedSignerError: false,
44 | expectedAuthorizedSigner: false,
45 | },
46 | {
47 | title:
48 | 'External wallets should NOT be authorized signers over OTHER addresses',
49 | isEOA: true,
50 | challenge: 'foo',
51 | challengeSign: 'foo',
52 | signingKeys: [keyA],
53 | authAddr: keyToAddress(keyB),
54 | mockContract: {
55 | authorizedKey: privateToPublic(keyC),
56 | address: keyToAddress(keyB),
57 | errorIsValidSignature: false,
58 | },
59 | expectedAuthorizedSignerError: false,
60 | expectedAuthorizedSigner: false,
61 | },
62 | {
63 | title:
64 | 'Smart-contract wallets with a 1-of-1 correct internal key should be authorized signers over their address',
65 | isEOA: false,
66 | challenge: 'foo',
67 | challengeSign: 'foo',
68 | signingKeys: [keyB],
69 | authAddr: keyToAddress(keyA),
70 | mockContract: {
71 | authorizedKey: privateToPublic(keyB),
72 | address: keyToAddress(keyA),
73 | errorIsValidSignature: false,
74 | },
75 | expectedAuthorizedSignerError: false,
76 | expectedAuthorizedSigner: true,
77 | },
78 | {
79 | title:
80 | 'Smart-contract wallets with a 1-of-2 (multi-sig) correct internal key should be authorized signers over their address',
81 | isEOA: false,
82 | challenge: 'foo',
83 | challengeSign: 'foo',
84 | signingKeys: [keyB, keyC],
85 | authAddr: keyToAddress(keyA),
86 | mockContract: {
87 | authorizedKey: privateToPublic(keyB),
88 | address: keyToAddress(keyA),
89 | errorIsValidSignature: false,
90 | },
91 | expectedAuthorizedSignerError: false,
92 | expectedAuthorizedSigner: true,
93 | },
94 | {
95 | title:
96 | 'Smart-contract wallets with a 1-of-1 incorrect internal key should NOT be authorized signers over their address',
97 | isEOA: false,
98 | challenge: 'foo',
99 | challengeSign: 'foo',
100 | signingKeys: [keyB],
101 | authAddr: keyToAddress(keyA),
102 | mockContract: {
103 | authorizedKey: privateToPublic(keyC),
104 | address: keyToAddress(keyA),
105 | errorIsValidSignature: false,
106 | },
107 | expectedAuthorizedSignerError: false,
108 | expectedAuthorizedSigner: false,
109 | },
110 | {
111 | title: 'isAuthorizedSigner should error when smart-contract call errors',
112 | isEOA: false,
113 | challenge: 'foo',
114 | challengeSign: 'foo',
115 | signingKeys: [keyB],
116 | authAddr: keyToAddress(keyA),
117 | mockContract: {
118 | authorizedKey: privateToPublic(keyB),
119 | address: keyToAddress(keyA),
120 | errorIsValidSignature: true,
121 | },
122 | expectedAuthorizedSignerError: true,
123 | expectedAuthorizedSigner: false,
124 | },
125 | ];
126 |
127 | testCases.forEach((test) =>
128 | it(test.title, async () => {
129 | const dappAuth = new DappAuth(
130 | new ProviderMock(new ContractMock(test.mockContract)),
131 | );
132 |
133 | const signatureFunc = test.isEOA
134 | ? signEOAPersonalMessage
135 | : signERC1654PersonalMessage;
136 |
137 | const signatures = `0x${test.signingKeys
138 | .map((signingKey) =>
139 | stripHexPrefix(
140 | signatureFunc(test.challengeSign, signingKey, test.authAddr),
141 | ),
142 | )
143 | .join('')}`;
144 |
145 | let isError = false;
146 | let isAuthorizedSigner = false;
147 | try {
148 | isAuthorizedSigner = await dappAuth.isAuthorizedSigner(
149 | test.challenge,
150 | signatures,
151 | test.authAddr,
152 | );
153 | } catch (error) {
154 | isError = true;
155 | }
156 |
157 | equal(isError, test.expectedAuthorizedSignerError);
158 | equal(isAuthorizedSigner, test.expectedAuthorizedSigner);
159 | }),
160 | );
161 |
162 | it('It should decode challenge as utf8 by default when decoding challenges', async function() {
163 | const dappAuth = new DappAuth(
164 | new ProviderMock(
165 | new ContractMock({
166 | authorizedKey: null,
167 | address: null,
168 | errorIsValidSignature: false,
169 | }),
170 | ),
171 | );
172 |
173 | const eoaHash = dappAuth._hashEOAPersonalMessage('foo');
174 | equal(
175 | `0x${eoaHash.toString('hex')}`,
176 | '0x76b2e96714d3b5e6eb1d1c509265430b907b44f72b2a22b06fcd4d96372b8565',
177 | );
178 |
179 | const scHash = dappAuth._hashSCMessage('foo');
180 | equal(
181 | `0x${scHash.toString('hex')}`,
182 | '0x41b1a0649752af1b28b3dc29a1556eee781e4a4c3a1f7f53f90fa834de098c4d',
183 | );
184 | });
185 |
186 | // See https://github.com/MetaMask/eth-sig-util/issues/60
187 | it('It should decode challenge as hex if hex is detected when decoding challenges', async function() {
188 | const dappAuth = new DappAuth(
189 | new ProviderMock(
190 | new ContractMock({
191 | authorizedKey: null,
192 | address: null,
193 | errorIsValidSignature: false,
194 | }),
195 | ),
196 | );
197 |
198 | // result if 0xffff is decoded as hex: 13a6aa3102b2d639f36804a2d7c31469618fd7a7907c658a7b2aa91a06e31e47
199 | // result if 0xffff is decoded as utf8: 247aefb5d2e5b17fca61f786c779f7388485460c13e51308f88b2ff84ffa6851
200 | const eoaHash = dappAuth._hashEOAPersonalMessage('0xffff');
201 | equal(
202 | `0x${eoaHash.toString('hex')}`,
203 | '0x13a6aa3102b2d639f36804a2d7c31469618fd7a7907c658a7b2aa91a06e31e47',
204 | );
205 |
206 | // result if 0xffff is decoded as hex: 06d41322d79dfed27126569cb9a80eb0967335bf2f3316359d2a93c779fcd38a
207 | // result if 0xffff is decoded as utf8: f0443ea82539c5136844b0a175f544b7ee7bc0fc5ce940bad19f08eaf618af71
208 | const scHash = dappAuth._hashSCMessage('0xffff');
209 | equal(
210 | `0x${scHash.toString('hex')}`,
211 | '0x06d41322d79dfed27126569cb9a80eb0967335bf2f3316359d2a93c779fcd38a',
212 | );
213 | });
214 |
215 | // This test is needed for 100% coverage
216 | it('Invalid signature should fail', async function() {
217 | const dappAuth = new DappAuth(
218 | new ProviderMock(
219 | new ContractMock({
220 | authorizedKey: null,
221 | address: null,
222 | errorIsValidSignature: false,
223 | }),
224 | ),
225 | );
226 |
227 | const signatures = '0xinvalid-signature';
228 |
229 | let isError = false;
230 | let isAuthorizedSigner = false;
231 | try {
232 | isAuthorizedSigner = await dappAuth.isAuthorizedSigner(
233 | 'foo',
234 | signatures,
235 | keyToAddress(keyA),
236 | );
237 | } catch (error) {
238 | isError = true;
239 | }
240 |
241 | equal(isError, true);
242 | equal(isAuthorizedSigner, false);
243 | });
244 | });
245 |
246 | describe('utils', function() {
247 | it('Should remove hex prefix if value is hex prefixed', function() {
248 | const value = 'foo';
249 | equal(removeHexPrefix(value), 'foo');
250 | });
251 | });
252 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/__tests__/fixtures/getEvmSupport.ts:
--------------------------------------------------------------------------------
1 | export const getEvmSupportList = {
2 | '1': {
3 | chain_id: 1,
4 | name: 'ethereum',
5 | display_name: 'Ethereum',
6 | network_type: 'mainnet',
7 | blocto_service_environment: 'prod',
8 | rpc_endpoint_domains: [
9 | 'mainnet.infura.io',
10 | 'rpc.ankr.com',
11 | 'cloudflare-eth.com',
12 | ],
13 | },
14 | '5': {
15 | chain_id: 5,
16 | name: 'ethereum',
17 | display_name: 'Goerli',
18 | network_type: 'testnet',
19 | blocto_service_environment: 'dev',
20 | rpc_endpoint_domains: [
21 | 'goerli.infura.io',
22 | 'goerli.blockpi.network',
23 | 'rpc.ankr.com',
24 | ],
25 | },
26 | '10': {
27 | chain_id: 10,
28 | name: 'optimism',
29 | display_name: 'Optimism',
30 | network_type: 'mainnet',
31 | blocto_service_environment: 'prod',
32 | rpc_endpoint_domains: [
33 | 'optimism-mainnet.infura.io',
34 | 'mainnet.optimism.io',
35 | 'opt-mainnet.g.alchemy.com',
36 | 'optimism.blockpi.network',
37 | 'rpc.ankr.com',
38 | ],
39 | },
40 | '56': {
41 | chain_id: 56,
42 | name: 'bsc',
43 | display_name: 'BNB Smart Chain',
44 | network_type: 'mainnet',
45 | blocto_service_environment: 'prod',
46 | rpc_endpoint_domains: [
47 | 'bsc-dataseed.binance.org',
48 | 'bsc-dataseed1.binance.org',
49 | 'bsc-dataseed2.binance.org',
50 | 'bsc-dataseed3.binance.org',
51 | 'bsc-dataseed4.binance.org',
52 | 'bsc-dataseed1.defibit.io',
53 | 'bsc-dataseed2.defibit.io',
54 | 'bsc-dataseed3.defibit.io',
55 | 'bsc-dataseed4.defibit.io',
56 | 'bsc-dataseed1.ninicoin.io',
57 | 'bsc-dataseed2.ninicoin.io',
58 | 'bsc-dataseed3.ninicoin.io',
59 | 'bsc-dataseed4.ninicoin.io',
60 | 'rpc.ankr.com',
61 | ],
62 | },
63 | '97': {
64 | chain_id: 97,
65 | name: 'bsc',
66 | display_name: 'BNB Smart Chain Testnet',
67 | network_type: 'testnet',
68 | blocto_service_environment: 'dev',
69 | rpc_endpoint_domains: [
70 | 'data-seed-prebsc-1-s1.binance.org',
71 | 'data-seed-prebsc-2-s1.binance.org',
72 | 'data-seed-prebsc-1-s2.binance.org',
73 | 'data-seed-prebsc-2-s2.binance.org',
74 | 'data-seed-prebsc-1-s3.binance.org',
75 | 'data-seed-prebsc-2-s3.binance.org',
76 | 'rpc.ankr.com',
77 | ],
78 | },
79 | '137': {
80 | chain_id: 137,
81 | name: 'polygon',
82 | display_name: 'Polygon',
83 | network_type: 'mainnet',
84 | blocto_service_environment: 'prod',
85 | rpc_endpoint_domains: [
86 | 'polygon-mainnet.infura.io',
87 | 'rpc.ankr.com',
88 | 'polygon-rpc.com',
89 | 'rpc-mainnet.maticvigil.com',
90 | ],
91 | },
92 | '4200': {
93 | chain_id: 4200,
94 | name: 'merlin',
95 | display_name: 'Merlin',
96 | network_type: 'mainnet',
97 | blocto_service_environment: 'prod',
98 | rpc_endpoint_domains: ['rpc.merlinchain.io', 'merlin.blockpi.network'],
99 | },
100 | '8453': {
101 | chain_id: 8453,
102 | name: 'base',
103 | display_name: 'Base',
104 | network_type: 'mainnet',
105 | blocto_service_environment: 'prod',
106 | rpc_endpoint_domains: [
107 | 'base-mainnet.infura.io',
108 | 'base-mainnet.g.alchemy.com',
109 | 'mainnet.base.org',
110 | 'rpc.ankr.com',
111 | 'base.blockpi.network',
112 | '1rpc.io',
113 | ],
114 | },
115 | '42161': {
116 | chain_id: 42161,
117 | name: 'arbitrum',
118 | display_name: 'Arbitrum',
119 | network_type: 'mainnet',
120 | blocto_service_environment: 'prod',
121 | rpc_endpoint_domains: [
122 | 'arbitrum-mainnet.infura.io',
123 | 'arb1.arbitrum.io',
124 | 'arb-mainnet.g.alchemy.com',
125 | 'arbitrum.blockpi.network',
126 | 'rpc.ankr.com',
127 | ],
128 | },
129 | '43113': {
130 | chain_id: 43113,
131 | name: 'avalanche',
132 | display_name: 'Avalanche Fuji',
133 | network_type: 'testnet',
134 | blocto_service_environment: 'dev',
135 | rpc_endpoint_domains: [
136 | 'api.avax-test.network',
137 | 'avalanche-fuji.infura.io',
138 | 'rpc.ankr.com',
139 | 'ava-testnet.public.blastapi.io',
140 | ],
141 | },
142 | '43114': {
143 | chain_id: 43114,
144 | name: 'avalanche',
145 | display_name: 'Avalanche',
146 | network_type: 'mainnet',
147 | blocto_service_environment: 'prod',
148 | rpc_endpoint_domains: [
149 | 'api.avax.network',
150 | 'avalanche-mainnet.infura.io',
151 | 'rpc.ankr.com',
152 | 'ava-mainnet.public.blastapi.io',
153 | 'snowtrace.io',
154 | ],
155 | },
156 | '80002': {
157 | chain_id: 80002,
158 | name: 'polygon_amoy',
159 | display_name: 'Amoy Testnet',
160 | network_type: 'testnet',
161 | blocto_service_environment: 'dev',
162 | rpc_endpoint_domains: ['rpc-amoy.polygon.technology', 'rpc.ankr.com'],
163 | },
164 | '81457': {
165 | chain_id: 81457,
166 | name: 'blast',
167 | display_name: 'Blast',
168 | network_type: 'mainnet',
169 | blocto_service_environment: 'prod',
170 | rpc_endpoint_domains: [
171 | 'rpc.blast.io',
172 | 'rpc.ankr.com',
173 | 'blast.din.dev',
174 | 'blastl2-mainnet.public.blastapi.io',
175 | 'blast.blockpi.network',
176 | ],
177 | },
178 | '84532': {
179 | chain_id: 84532,
180 | name: 'base_sepolia',
181 | display_name: 'Base Sepolia',
182 | network_type: 'testnet',
183 | blocto_service_environment: 'dev',
184 | rpc_endpoint_domains: [
185 | 'sepolia.base.org',
186 | 'base-sepolia.blockpi.network',
187 | 'base-sepolia.g.alchemy.com',
188 | 'base-sepolia.infura.io',
189 | 'rpc.ankr.com',
190 | '1rpc.io',
191 | ],
192 | },
193 | '421614': {
194 | chain_id: 421614,
195 | name: 'arbitrum_sepolia',
196 | display_name: 'Arbitrum Sepolia',
197 | network_type: 'testnet',
198 | blocto_service_environment: 'dev',
199 | rpc_endpoint_domains: [
200 | 'sepolia-rollup.arbitrum.io/rpc',
201 | 'arbitrum-sepolia.infura.io',
202 | 'arb-sepolia.g.alchemy.com',
203 | 'rpc.ankr.com',
204 | ],
205 | },
206 | '534351': {
207 | chain_id: 534351,
208 | name: 'scroll_sepolia',
209 | display_name: 'Scroll Sepolia',
210 | network_type: 'testnet',
211 | blocto_service_environment: 'dev',
212 | rpc_endpoint_domains: [
213 | 'sepolia-rpc.scroll.io',
214 | 'scroll-testnet-public.unifra.io',
215 | 'scroll-sepolia.blockpi.network',
216 | 'rpc.ankr.com',
217 | 'scroll-sepolia.chainstacklabs.com',
218 | 'scroll-public.scroll-testnet.quiknode.pro',
219 | ],
220 | },
221 | '534352': {
222 | chain_id: 534352,
223 | name: 'scroll',
224 | display_name: 'Scroll',
225 | network_type: 'mainnet',
226 | blocto_service_environment: 'prod',
227 | rpc_endpoint_domains: [
228 | 'rpc.scroll.io',
229 | 'rpc.ankr.com',
230 | 'rpc-scroll.icecreamswap.com',
231 | 'scroll-mainnet.public.blastapi.io',
232 | 'scroll-mainnet-public.unifra.io',
233 | 'scroll.blockpi.network',
234 | '1rpc.io',
235 | 'scroll-mainnet.chainstacklabs.com',
236 | ],
237 | },
238 | '686868': {
239 | chain_id: 686868,
240 | name: 'merlin_testnet',
241 | display_name: 'Merlin Testnet',
242 | network_type: 'testnet',
243 | blocto_service_environment: 'dev',
244 | rpc_endpoint_domains: ['testnet-rpc.merlinchain.io'],
245 | },
246 | '1261120': {
247 | chain_id: 1261120,
248 | name: 'zkatana_sepolia',
249 | display_name: 'Astar zKatana',
250 | network_type: 'testnet',
251 | blocto_service_environment: 'dev',
252 | rpc_endpoint_domains: [
253 | 'rpc.startale.com/zkatana',
254 | 'rpc.zkatana.gelato.digital',
255 | 'astar-zkatana-rpc.dwellir.com',
256 | ],
257 | },
258 | '7777777': {
259 | chain_id: 7777777,
260 | name: 'zora',
261 | display_name: 'Zora',
262 | network_type: 'mainnet',
263 | blocto_service_environment: 'prod',
264 | rpc_endpoint_domains: ['rpc.zora.energy'],
265 | },
266 | '11155111': {
267 | chain_id: 11155111,
268 | name: 'ethereum_sepolia',
269 | display_name: 'Sepolia',
270 | network_type: 'testnet',
271 | blocto_service_environment: 'dev',
272 | rpc_endpoint_domains: [
273 | 'sepolia.infura.io',
274 | 'rpc.sepolia.dev',
275 | 'rpc.sepolia.org',
276 | 'rpc.sepolia.online',
277 | 'www.sepoliarpc.space',
278 | 'rpc-sepolia.rockx.com',
279 | 'rpc.bordel.wtf/sepolia',
280 | ],
281 | },
282 | '11155420': {
283 | chain_id: 11155420,
284 | name: 'optimism_sepolia',
285 | display_name: 'Optimism Sepolia',
286 | network_type: 'testnet',
287 | blocto_service_environment: 'dev',
288 | rpc_endpoint_domains: [
289 | 'optimism-sepolia.infura.io',
290 | 'rpc.ankr.com',
291 | 'sepolia.optimism.io',
292 | 'opt-sepolia.g.alchemy.com',
293 | 'optimism-sepolia.blockpi.network',
294 | ],
295 | },
296 | '168587773': {
297 | chain_id: 168587773,
298 | name: 'blast_sepolia',
299 | display_name: 'Blast Sepolia',
300 | network_type: 'testnet',
301 | blocto_service_environment: 'dev',
302 | rpc_endpoint_domains: ['sepolia.blast.io', 'blast-sepolia.blockpi.network'],
303 | },
304 | '999999999': {
305 | chain_id: 999999999,
306 | name: 'zora_sepolia',
307 | display_name: 'Zora Sepolia',
308 | network_type: 'testnet',
309 | blocto_service_environment: 'dev',
310 | rpc_endpoint_domains: ['sepolia.rpc.zora.energy'],
311 | },
312 | };
313 |
--------------------------------------------------------------------------------
/packages/blocto-sdk/src/__tests__/index.test.js:
--------------------------------------------------------------------------------
1 | import BloctoSDK from '../main';
2 | import { enableFetchMocks } from 'jest-fetch-mock';
3 | import { setAccountStorage } from '../lib/storage';
4 |
5 | beforeAll(() => {
6 | enableFetchMocks();
7 | fetch.mockResponse((req) => {
8 | if (req.url === 'https://api.blocto.app/networks/evm') {
9 | return new Promise((resolve) => {
10 | resolve({
11 | status: 200,
12 | body: JSON.stringify({
13 | networks: [
14 | {
15 | chain_id: 5,
16 | name: 'ethereum',
17 | display_name: 'Ethereum',
18 | network_type: 'testnet',
19 | blocto_service_environment: 'dev',
20 | rpc_endpoint_domains: [],
21 | },
22 | {
23 | chain_id: 56,
24 | name: 'bsc',
25 | display_name: 'Smart Chain',
26 | network_type: 'mainnet',
27 | blocto_service_environment: 'prod',
28 | rpc_endpoint_domains: [],
29 | },
30 | ],
31 | }),
32 | });
33 | });
34 | }
35 | return new Promise((resolve) => {
36 | resolve({
37 | status: 200,
38 | body: JSON.stringify({
39 | test: 'pass',
40 | }),
41 | });
42 | });
43 | });
44 | setAccountStorage('BLOCTO_SDK', {
45 | code: '123',
46 | accounts: {
47 | aptos: ['0x123'],
48 | },
49 | evm: {
50 | bsc: ['0x123'],
51 | },
52 | });
53 | });
54 |
55 | describe('Testing BloctoSDK', () => {
56 | test('should be a class', () => {
57 | expect(BloctoSDK).toBeInstanceOf(Function);
58 | });
59 | });
60 |
61 | describe('Testing BloctoSDK ethereum provider', () => {
62 | const bloctoSDK = new BloctoSDK({
63 | ethereum: {
64 | chainId: 56,
65 | rpc: 'https://bsc-dataseed.binance.org',
66 | },
67 | });
68 | const ethereum = bloctoSDK.ethereum;
69 | ethereum.setIframe = jest.fn();
70 | ethereum.responseListener = jest.fn();
71 |
72 | test('should be a object', () => {
73 | expect(bloctoSDK).toBeInstanceOf(Object);
74 | });
75 | test('should have ethereum provider', () => {
76 | expect(ethereum).toBeInstanceOf(Object);
77 | });
78 | test('should have isBlocto', () => {
79 | expect(ethereum.isBlocto).toBe(true);
80 | });
81 | test('converted chainId should be hexdecimal', () => {
82 | expect(ethereum.chainId).toBe('0x38');
83 | });
84 | test('should setup rpc', () => {
85 | expect(ethereum.rpc).toBe('https://bsc-dataseed.binance.org');
86 | });
87 | test('should setup _blocto', async () => {
88 | await ethereum.request({
89 | method:"eth_addEthereumChain",
90 | params: [
91 | {
92 | chainId: '0xaa36a7',
93 | rpcUrls: ['https://ethereum-sepolia.blockpi.network/v1/rpc/public']
94 | }
95 | ]
96 | });
97 | expect(ethereum._blocto.networkType).toBe('mainnet');
98 | });
99 | test('should request chainId work', async () => {
100 | return ethereum.request({ method: 'eth_chainId' }).then((chainId) => {
101 | expect(chainId).toBe('0x38');
102 | });
103 | });
104 | test('should request eth_estimateUserOperationGas works', async () => {
105 | return ethereum
106 | .request({ method: 'eth_estimateUserOperationGas' })
107 | .then((response) => {
108 | expect(JSON.parse(fetch.mock.lastCall[1].body).method).toBe(
109 | 'eth_estimateUserOperationGas'
110 | );
111 | expect(response.test).toBe('pass');
112 | });
113 | });
114 | test('should request personal_sign send right param', async () => {
115 | const exampleMessage = 'Test `personal_sign` message.';
116 | return ethereum
117 | .request({
118 | method: 'personal_sign',
119 | params: [exampleMessage, '0x123'],
120 | })
121 | .then(() => {
122 | expect(fetch.mock.lastCall[0]).toBe(
123 | 'https://wallet-v2.blocto.app/api/bsc/user-signature'
124 | );
125 | expect(fetch.mock.lastCall[1].body).toBe(
126 | JSON.stringify({
127 | method: 'personal_sign',
128 | message:
129 | '546573742060706572736f6e616c5f7369676e60206d6573736167652e',
130 | })
131 | );
132 | });
133 | });
134 | test('should request eth_signTypedData call right api', async () => {
135 | const msgParams = JSON.stringify({
136 | domain: {
137 | chainId: 56,
138 | name: 'Ether Mail',
139 | verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC',
140 | version: '1',
141 | },
142 | message: {
143 | contents: 'Hello, Bob!',
144 | },
145 | });
146 | return ethereum
147 | .request({
148 | method: 'eth_signTypedData',
149 | params: ['0x123', msgParams],
150 | })
151 | .then(() => {
152 | expect(fetch.mock.lastCall[0]).toBe(
153 | 'https://wallet-v2.blocto.app/api/bsc/user-signature'
154 | );
155 | expect(fetch.mock.lastCall[1].body).toBe(
156 | JSON.stringify({ method: 'eth_signTypedData', message: msgParams })
157 | );
158 | });
159 | });
160 | test('should request eth_sendTransaction call right api', async () => {
161 | return ethereum
162 | .request({
163 | method: 'eth_sendTransaction',
164 | params: [
165 | {
166 | from: '0x123',
167 | to: '0x123',
168 | value: '0x29a2241af62c0000',
169 | },
170 | ],
171 | })
172 | .then(() => {
173 | expect(fetch.mock.lastCall[0]).toBe(
174 | 'https://wallet-v2.blocto.app/api/bsc/authz'
175 | );
176 | expect(fetch.mock.lastCall[1].body).toBe(
177 | '[[{"from":"0x123","to":"0x123","value":"0x29a2241af62c0000"}],true]'
178 | );
179 | });
180 | });
181 | test('should request wallet_sendMultiCallTransaction in EIP-1193 way has right payload', async () => {
182 | return ethereum
183 | .request({
184 | method: 'wallet_sendMultiCallTransaction',
185 | params: [
186 | [{
187 | from: '0x123',
188 | to: '0x123',
189 | value: '0x01abc',
190 | },
191 | {
192 | from: '0x456',
193 | to: '0x456',
194 | value: '0x02def',
195 | }],
196 | false
197 | ],
198 | })
199 | .then(() => {
200 | expect(fetch.mock.lastCall[0]).toBe(
201 | 'https://wallet-v2.blocto.app/api/bsc/authz'
202 | );
203 | expect(fetch.mock.lastCall[1].body).toBe(
204 | '[[{"from":"0x123","to":"0x123","value":"0x01abc"},{"from":"0x456","to":"0x456","value":"0x02def"}],false]'
205 | );
206 | });
207 | });
208 | test('should request wallet_sendMultiCallTransaction in Web3 Batch Request way has right payload', async () => {
209 | return ethereum
210 | .sendAsync([
211 | {
212 | method: 'eth_sendTransaction',
213 | params: [
214 | {
215 | from: '0x123',
216 | to: '0x123',
217 | value: '0x01abc',
218 | },
219 | ],
220 | },
221 | {
222 | method: 'eth_sendTransaction',
223 | params: [
224 | {
225 | from: '0x456',
226 | to: '0x456',
227 | value: '0x02def',
228 | },
229 | ],
230 | },
231 | ])
232 | .then(() => {
233 | expect(fetch.mock.lastCall[0]).toBe(
234 | 'https://wallet-v2.blocto.app/api/bsc/authz'
235 | );
236 | expect(fetch.mock.lastCall[1].body).toBe(
237 | '[[{"from":"0x123","to":"0x123","value":"0x01abc"},{"from":"0x456","to":"0x456","value":"0x02def"}],false]'
238 | );
239 | });
240 | });
241 | test('should request eth_sendUserOperation call right api', async () => {
242 | return ethereum
243 | .request({
244 | method: 'eth_sendUserOperation',
245 | params: [
246 | {
247 | callData: '0x123',
248 | },
249 | ],
250 | })
251 | .then(() => {
252 | expect(fetch.mock.lastCall[0]).toBe(
253 | 'https://wallet-v2.blocto.app/api/bsc/user-operation'
254 | );
255 | expect(fetch.mock.lastCall[1].body).toBe('[{"callData":"0x123"}]');
256 | });
257 | });
258 | });
259 |
260 | describe('Testing BloctoSDK providers with appId', () => {
261 | const bloctoSDK = new BloctoSDK({
262 | ethereum: {
263 | chainId: 56,
264 | rpc: 'https://bsc-dataseed.binance.org',
265 | },
266 | appId: '6f6b97c5-d97b-4799-8ad7-d7e8426d3369',
267 | });
268 | const ethereum = bloctoSDK.ethereum;
269 | ethereum.setIframe = jest.fn();
270 | ethereum.responseListener = jest.fn();
271 | test('should be a object', () => {
272 | expect(bloctoSDK).toBeInstanceOf(Object);
273 | });
274 | test('should eth_sendTransaction has right appId in header', async () => {
275 | return ethereum
276 | .request({
277 | method: 'eth_sendTransaction',
278 | params: [
279 | {
280 | from: '0x123',
281 | to: '0x123',
282 | value: '0x29a2241af62c0000',
283 | },
284 | ],
285 | })
286 | .then(() => {
287 | expect(
288 | fetch.mock.lastCall[1].headers['Blocto-Application-Identifier']
289 | ).toBe('6f6b97c5-d97b-4799-8ad7-d7e8426d3369');
290 | expect(fetch.mock.lastCall[0]).toBe(
291 | 'https://wallet-v2.blocto.app/api/bsc/authz'
292 | );
293 | expect(fetch.mock.lastCall[1].body).toBe(
294 | '[[{"from":"0x123","to":"0x123","value":"0x29a2241af62c0000"}],true]'
295 | );
296 | });
297 | });
298 | });
299 |
--------------------------------------------------------------------------------
/adapters/aptos-wallet-adapter-plugin/CHANGELOG.md:
--------------------------------------------------------------------------------
1 | # @blocto/aptos-wallet-adapter-plugin
2 |
3 | ## 0.2.11
4 |
5 | ### Patch Changes
6 |
7 | - Updated dependencies [79dade9]
8 | - Updated dependencies [ec87223]
9 | - Updated dependencies [2e2164e]
10 | - @blocto/sdk@0.10.4
11 |
12 | ## 0.2.11-beta.0
13 |
14 | ### Patch Changes
15 |
16 | - Updated dependencies [ec87223]
17 | - @blocto/sdk@0.10.4-beta.0
18 |
19 | ## 0.2.10
20 |
21 | ### Patch Changes
22 |
23 | - Updated dependencies [e066144]
24 | - @blocto/sdk@0.10.3
25 |
26 | ## 0.2.10-beta.0
27 |
28 | ### Patch Changes
29 |
30 | - Updated dependencies [e066144]
31 | - @blocto/sdk@0.10.3-beta.0
32 |
33 | ## 0.2.9
34 |
35 | ### Patch Changes
36 |
37 | - Updated dependencies [bd6b11c]
38 | - Updated dependencies [74abbd7]
39 | - @blocto/sdk@0.10.0
40 |
41 | ## 0.2.9-beta.0
42 |
43 | ### Patch Changes
44 |
45 | - Updated dependencies [74abbd7]
46 | - @blocto/sdk@0.10.0-beta.0
47 |
48 | ## 0.2.8
49 |
50 | ### Patch Changes
51 |
52 | - 48fa07a: fix: handle existedSDK handle window not defined
53 | - Updated dependencies [48fa07a]
54 | - Updated dependencies [4ea8a07]
55 | - @blocto/sdk@0.9.1
56 |
57 | ## 0.2.8-beta.1
58 |
59 | ### Patch Changes
60 |
61 | - 48fa07a: fix: handle existedSDK handle window not defined
62 | - Updated dependencies [48fa07a]
63 | - @blocto/sdk@0.9.1-beta.1
64 |
65 | ## 0.2.8-beta.0
66 |
67 | ### Patch Changes
68 |
69 | - Updated dependencies [4ea8a07]
70 | - @blocto/sdk@0.9.1-beta.0
71 |
72 | ## 0.2.7
73 |
74 | ### Patch Changes
75 |
76 | - b9dac7d: support web3js v4
77 | - 8d0b5bf: enhance in-app-sdk event support
78 | - 00a3832: sendAsync can send another requests
79 | - Updated dependencies [b9dac7d]
80 | - Updated dependencies [8d0b5bf]
81 | - Updated dependencies [55afe21]
82 | - Updated dependencies [3017eb1]
83 | - Updated dependencies [00a3832]
84 | - Updated dependencies [aa28bf6]
85 | - Updated dependencies [d34fca2]
86 | - @blocto/sdk@0.9.0
87 |
88 | ## 0.2.7-beta.1
89 |
90 | ### Patch Changes
91 |
92 | - b9dac7d: support web3js v4
93 | - 00a3832: sendAsync can send another requests
94 | - Updated dependencies [b9dac7d]
95 | - Updated dependencies [55afe21]
96 | - Updated dependencies [3017eb1]
97 | - Updated dependencies [00a3832]
98 | - @blocto/sdk@0.9.0-beta.2
99 |
100 | ## 0.2.7-beta.0
101 |
102 | ### Patch Changes
103 |
104 | - 8d0b5bf: enhance in-app-sdk event support
105 | - Updated dependencies [8d0b5bf]
106 | - @blocto/sdk@0.8.1-beta.0
107 |
108 | ## 0.2.6
109 |
110 | ### Patch Changes
111 |
112 | - Updated dependencies [a4d3fbd]
113 | - @blocto/sdk@0.8.0
114 |
115 | ## 0.2.6-beta.0
116 |
117 | ### Patch Changes
118 |
119 | - Updated dependencies [4835522]
120 | - Updated dependencies [e742f66]
121 | - @blocto/sdk@0.7.1-beta.0
122 |
123 | ## 0.2.5
124 |
125 | ### Patch Changes
126 |
127 | - Updated dependencies [2515a8f]
128 | - Updated dependencies [b40fbed]
129 | - Updated dependencies [4c769f9]
130 | - Updated dependencies [bbf2160]
131 | - Updated dependencies [a3b0243]
132 | - Updated dependencies [1288bd1]
133 | - @blocto/sdk@0.7.0
134 |
135 | ## 0.2.5-beta.0
136 |
137 | ### Patch Changes
138 |
139 | - Updated dependencies [b40fbed]
140 | - Updated dependencies [4c769f9]
141 | - Updated dependencies [bbf2160]
142 | - Updated dependencies [a3b0243]
143 | - Updated dependencies [1288bd1]
144 | - @blocto/sdk@0.7.0-beta.0
145 |
146 | ## 0.2.4
147 |
148 | ### Patch Changes
149 |
150 | - Updated dependencies [21097d9]
151 | - Updated dependencies [e0e8fb7]
152 | - Updated dependencies [6e427b1]
153 | - Updated dependencies [34a23e4]
154 | - Updated dependencies [55f3395]
155 | - Updated dependencies [3ceb547]
156 | - Updated dependencies [21097d9]
157 | - Updated dependencies [21097d9]
158 | - Updated dependencies [21097d9]
159 | - Updated dependencies [21097d9]
160 | - Updated dependencies [494ded1]
161 | - @blocto/sdk@0.6.0
162 |
163 | ## 0.2.4-beta.0
164 |
165 | ### Patch Changes
166 |
167 | - Updated dependencies [e0e8fb7]
168 | - @blocto/sdk@0.6.0-beta.0
169 |
170 | ## 0.2.3
171 |
172 | ### Patch Changes
173 |
174 | - abd96b2: Update aptos dependency to 1.15.0
175 | - Updated dependencies [abd96b2]
176 | - Updated dependencies [fec7693]
177 | - Updated dependencies [dca87b8]
178 | - Updated dependencies [b9f1eca]
179 | - Updated dependencies [2cacc63]
180 | - @blocto/sdk@0.5.5
181 |
182 | ## 0.2.3-beta.1
183 |
184 | ### Patch Changes
185 |
186 | - Update aptos dependency to 1.15.0
187 | - Updated dependencies
188 | - @blocto/sdk@0.5.5-beta.3
189 |
190 | ## 0.2.3-beta.0
191 |
192 | ### Patch Changes
193 |
194 | - Updated dependencies [fec7693]
195 | - @blocto/sdk@0.5.5-beta.0
196 |
197 | ## 0.2.2
198 |
199 | ### Patch Changes
200 |
201 | - Updated dependencies [6ae5c63]
202 | - Updated dependencies [438b66e]
203 | - Updated dependencies [a21d79b]
204 | - @blocto/sdk@0.5.4
205 |
206 | ## 0.2.2-beta.0
207 |
208 | ### Patch Changes
209 |
210 | - Updated dependencies [6ae5c63]
211 | - Updated dependencies [438b66e]
212 | - Updated dependencies [a21d79b]
213 | - @blocto/sdk@0.5.4-beta.0
214 |
215 | ## 0.2.1
216 |
217 | ### Patch Changes
218 |
219 | - f019a52: feat: help disconnect providers when get "incorrect_session_id" event
220 | - 80590d9: Emit accountChanged event if switch chain return different account
221 | - Updated dependencies [f019a52]
222 | - Updated dependencies [80590d9]
223 | - Updated dependencies [2c598f5]
224 | - @blocto/sdk@0.5.1
225 |
226 | ## 0.2.0
227 |
228 | ### Minor Changes
229 |
230 | - 9329e0a: Update core SDK dependency
231 |
232 | ### Patch Changes
233 |
234 | - 22ed496: fix: handle if sendUserOp call before enable
235 | - 2f87cbf: Update dependencies
236 | - 37e7db3: Add support for native ES6 import
237 | - c8a3b32: Fix eth_sendTransaction function
238 | - 8a5b40b: fix: wrong chainId after switch chain
239 | - cd1f781: Move session id to session storage
240 | - 05848f8: Replace rollup polyfill plugin with rollup-plugin-polyfill-node
241 | - Updated dependencies [22ed496]
242 | - Updated dependencies [68e1e82]
243 | - Updated dependencies [2f87cbf]
244 | - Updated dependencies [a83376b]
245 | - Updated dependencies [448c667]
246 | - Updated dependencies [9285453]
247 | - Updated dependencies [d495e61]
248 | - Updated dependencies [a4d99b1]
249 | - Updated dependencies [37e7db3]
250 | - Updated dependencies [652eefb]
251 | - Updated dependencies [17340c0]
252 | - Updated dependencies [c8a3b32]
253 | - Updated dependencies [5ea9bdd]
254 | - Updated dependencies [f078a5f]
255 | - Updated dependencies [e180c58]
256 | - Updated dependencies [c3485b6]
257 | - Updated dependencies [9c2d4bc]
258 | - Updated dependencies [a94fdba]
259 | - Updated dependencies [26cc3e9]
260 | - Updated dependencies [8a5b40b]
261 | - Updated dependencies [9630860]
262 | - Updated dependencies [cd1f781]
263 | - Updated dependencies [7830573]
264 | - Updated dependencies [17cf594]
265 | - Updated dependencies [05848f8]
266 | - @blocto/sdk@0.5.0
267 |
268 | ## 0.2.0-beta.5
269 |
270 | ### Patch Changes
271 |
272 | - 22ed496: fix: handle if sendUserOp call before enable
273 | - Updated dependencies [22ed496]
274 | - @blocto/sdk@0.5.0-beta.14
275 |
276 | ## 0.2.0-beta.4
277 |
278 | ### Patch Changes
279 |
280 | - 8a5b40b: fix: wrong chainId after switch chain
281 | - Updated dependencies [8a5b40b]
282 | - @blocto/sdk@0.5.0-beta.12
283 |
284 | ## 0.2.0-beta.3
285 |
286 | ### Patch Changes
287 |
288 | - Replace rollup polyfill plugin with rollup-plugin-polyfill-node
289 | - Updated dependencies
290 | - @blocto/sdk@0.5.0-beta.11
291 |
292 | ## 0.2.0-beta.2
293 |
294 | ### Patch Changes
295 |
296 | - Add support for native ES6 import
297 | - Updated dependencies
298 | - @blocto/sdk@0.5.0-beta.10
299 |
300 | ## 0.2.0-beta.1
301 |
302 | ### Minor Changes
303 |
304 | - Update core SDK dependency
305 |
306 | ## 0.1.9-beta.0
307 |
308 | ### Patch Changes
309 |
310 | - 2f87cbf: Update dependencies
311 | - c8a3b32: Fix eth_sendTransaction function
312 | - cd1f781: Move session id to session storage
313 | - Updated dependencies [68e1e82]
314 | - Updated dependencies [2f87cbf]
315 | - Updated dependencies [448c667]
316 | - Updated dependencies [d495e61]
317 | - Updated dependencies [c8a3b32]
318 | - Updated dependencies [f078a5f]
319 | - Updated dependencies [e180c58]
320 | - Updated dependencies [26cc3e9]
321 | - Updated dependencies [0351fca]
322 | - Updated dependencies [cd1f781]
323 | - Updated dependencies [7830573]
324 | - @blocto/sdk@0.5.0-beta.0
325 |
326 | ## 0.1.9
327 |
328 | ### Patch Changes
329 |
330 | - 17ead9c: aptos login condition
331 | - Updated dependencies [17ead9c]
332 | - @blocto/sdk@0.4.9
333 |
334 | ## 0.1.8
335 |
336 | ### Patch Changes
337 |
338 | - ebfb0da: Fix TypeScript types and eslint errors
339 | - Updated dependencies [ebfb0da]
340 | - Updated dependencies [0570b1e]
341 | - @blocto/sdk@0.4.7
342 |
343 | ## 0.1.8-beta.0
344 |
345 | ### Patch Changes
346 |
347 | - ebfb0da: Fix TypeScript types and eslint errors
348 | - Updated dependencies [ebfb0da]
349 | - @blocto/sdk@0.4.7-beta.0
350 |
351 | ## 0.1.7
352 |
353 | ### Patch Changes
354 |
355 | - 76a1b11: Add goerli support
356 | - Updated dependencies [76a1b11]
357 | - @blocto/sdk@0.4.6
358 |
359 | ## 0.1.6
360 |
361 | ### Patch Changes
362 |
363 | - 8313b19: Format signing message payload for Aptos
364 | - 8188f4f: Remove invariant lib prevent process undefined
365 | - 4da58d8: Fix incorrect session id error when the user logs in/out on the different chains
366 | - 655d1ca: Fix the formatting funciton for Aptos signing message
367 | - Updated dependencies [8313b19]
368 | - Updated dependencies [6e2cd2c]
369 | - Updated dependencies [8188f4f]
370 | - Updated dependencies [4da58d8]
371 | - Updated dependencies [655d1ca]
372 | - @blocto/sdk@0.4.5
373 |
374 | ## 0.1.6-beta.0
375 |
376 | ### Patch Changes
377 |
378 | - 8188f4f: Remove invariant lib prevent process undefined
379 | - Updated dependencies [8188f4f]
380 | - @blocto/sdk@0.4.5-beta.0
381 |
382 | ## 0.1.5
383 |
384 | ### Patch Changes
385 |
386 | - 8285743: Migrate with monorepo and changeset
387 | - 53fbf7f: Changed to newest @blocto/sdk dependencies
388 | - Updated dependencies [f45f39e]
389 | - Updated dependencies [8285743]
390 | - Updated dependencies [4cd7b21]
391 | - @blocto/sdk@0.4.4
392 |
393 | ## 0.1.5-beta.1
394 |
395 | ### Patch Changes
396 |
397 | - 53fbf7f: Changed to newest @blocto/sdk dependencies
398 |
399 | ## 0.1.5-beta.0
400 |
401 | ### Patch Changes
402 |
403 | - 8285743: Migrate with monorepo and changeset
404 | - Updated dependencies [8285743]
405 | - @blocto/sdk@0.4.4-beta.0
406 |
--------------------------------------------------------------------------------