diff --git a/apps/walletd/config/providers.tsx b/apps/walletd/config/providers.tsx
index 46c37e9a4..7cec3aced 100644
--- a/apps/walletd/config/providers.tsx
+++ b/apps/walletd/config/providers.tsx
@@ -4,6 +4,7 @@ import { WalletsProvider } from '../contexts/wallets'
import { AddressesProvider } from '../contexts/addresses'
import { EventsProvider } from '../contexts/events'
import { LedgerProvider } from '../contexts/ledger'
+import { AppProvider } from '../contexts/app'
type Props = {
children: React.ReactNode
@@ -11,19 +12,21 @@ type Props = {
export function Providers({ children }: Props) {
return (
-
-
-
-
-
- {/* this is here so that dialogs can use all the other providers,
+
+
+
+
+
+
+ {/* this is here so that dialogs can use all the other providers,
and the other providers can trigger dialogs */}
-
- {children}
-
-
-
-
-
+
+ {children}
+
+
+
+
+
+
)
}
diff --git a/apps/walletd/contexts/app/index.tsx b/apps/walletd/contexts/app/index.tsx
new file mode 100644
index 000000000..a11cd151c
--- /dev/null
+++ b/apps/walletd/contexts/app/index.tsx
@@ -0,0 +1,27 @@
+import { createContext, useContext, useEffect } from 'react'
+import { initSDK } from '@siafoundation/sdk'
+
+function useAppMain() {
+ // Initialize the SDK on app load
+ useEffect(() => {
+ const func = async () => {
+ await initSDK()
+ }
+ func()
+ }, [])
+ return {}
+}
+
+type State = ReturnType
+
+const AppContext = createContext({} as State)
+export const useApp = () => useContext(AppContext)
+
+type Props = {
+ children: React.ReactNode
+}
+
+export function AppProvider({ children }: Props) {
+ const state = useAppMain()
+ return {children}
+}
diff --git a/libs/sdk/src/index.ts b/libs/sdk/src/index.ts
index e59a87889..6ab683a62 100644
--- a/libs/sdk/src/index.ts
+++ b/libs/sdk/src/index.ts
@@ -1,2 +1,3 @@
-export * from './init'
export * from './types'
+export { initSDK } from './init'
+export { getSDK, SDK } from './sdk'
diff --git a/libs/sdk/src/sdk.ts b/libs/sdk/src/sdk.ts
index 948d8a4c6..330af96d7 100644
--- a/libs/sdk/src/sdk.ts
+++ b/libs/sdk/src/sdk.ts
@@ -1,12 +1,22 @@
import { WebTransportClient } from './transport'
-import { WASM } from './types'
+import { WasmApi } from './types'
-export function getSDK() {
+export type SDK = WasmApi & {
+ WebTransportClient: typeof WebTransportClient
+}
+
+export function getWasmApi(): WasmApi {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
- const wasm = (global as any).sia as WASM
+ return (globalThis as any).sia as WasmApi
+}
+
+export function getSDK(): SDK {
+ const wasmApi = getWasmApi()
+ if (wasmApi === undefined) {
+ throw new Error('The Sia SDK has not been initialized')
+ }
return {
- rhp: wasm.rhp,
- wallet: wasm.wallet,
+ ...wasmApi,
WebTransportClient,
}
}
diff --git a/libs/sdk/src/transport.ts b/libs/sdk/src/transport.ts
index 372c4e848..c9c1735e5 100644
--- a/libs/sdk/src/transport.ts
+++ b/libs/sdk/src/transport.ts
@@ -1,3 +1,4 @@
+import { getWasmApi } from './sdk'
import {
RPCReadSectorResponse,
RPCSettingsResponse,
@@ -9,26 +10,27 @@ import {
RPCWriteSector,
RPCSettings,
} from './types'
-import { WASM } from './types'
export class WebTransportClient {
#url: string
#cert: string
- #wasm: WASM
#transport!: WebTransport
- constructor(url: string, cert: string, wasm: WASM) {
+ constructor(url: string, cert: string) {
this.#url = url
this.#cert = cert
- this.#wasm = wasm
- }
- async connect() {
+ if (!getWasmApi()) {
+ throw new Error('The Sia SDK has not been initialized.')
+ }
+
if (!('WebTransport' in window)) {
throw new Error('WebTransport is not supported in your browser.')
}
+ }
+ async connect() {
try {
this.#transport = new WebTransport(this.#url, {
serverCertificateHashes: this.#cert
@@ -101,8 +103,8 @@ export class WebTransportClient {
): Promise {
return this.sendRequest(
readSector,
- this.#wasm.rhp.encodeReadSectorRequest,
- this.#wasm.rhp.decodeReadSectorResponse
+ getWasmApi().rhp.encodeReadSectorRequest,
+ getWasmApi().rhp.decodeReadSectorResponse
)
}
@@ -111,16 +113,16 @@ export class WebTransportClient {
): Promise {
return this.sendRequest(
writeSector,
- this.#wasm.rhp.encodeWriteSectorRequest,
- this.#wasm.rhp.decodeWriteSectorResponse
+ getWasmApi().rhp.encodeWriteSectorRequest,
+ getWasmApi().rhp.decodeWriteSectorResponse
)
}
async sendRPCSettingsRequest(): Promise {
return this.sendRequest(
undefined,
- this.#wasm.rhp.encodeSettingsRequest,
- this.#wasm.rhp.decodeSettingsResponse
+ getWasmApi().rhp.encodeSettingsRequest,
+ getWasmApi().rhp.decodeSettingsResponse
)
}
}
diff --git a/libs/sdk/src/types.ts b/libs/sdk/src/types.ts
index 8d3458dc4..1008727aa 100644
--- a/libs/sdk/src/types.ts
+++ b/libs/sdk/src/types.ts
@@ -3,15 +3,14 @@ import {
UnlockConditions,
ConsensusNetwork,
ConsensusState,
+ Currency,
+ Signature,
+ Address,
+ Hash256,
+ PublicKey,
+ PrivateKey,
} from '@siafoundation/types'
-type Currency = string
-type Signature = string
-type Address = string
-type Hash256 = string // 32 bytes
-type PrivateKey = string
-type PublicKey = string // 32 bytes
-
type AccountToken = {
account: PublicKey
validUntil: string
@@ -92,7 +91,7 @@ export type RPCWriteSector = {
export type RPC = RPCSettings | RPCReadSector | RPCWriteSector
-export type WASM = {
+export type WasmApi = {
rhp: {
generateAccount: () => {
privateKey?: PrivateKey
@@ -189,7 +188,7 @@ export type WASM = {
encodedTransaction?: string
error?: string
}
- signTransaction: (
+ signTransactionV1: (
cs: ConsensusState,
cn: ConsensusNetwork,
txn: Transaction,
diff --git a/libs/sdk/src/wallet.spec.ts b/libs/sdk/src/wallet.spec.ts
index 01961d8fc..bac4011d2 100644
--- a/libs/sdk/src/wallet.spec.ts
+++ b/libs/sdk/src/wallet.spec.ts
@@ -151,7 +151,7 @@ describe('wallet', () => {
it('signs a valid transaction', async () => {
const sdk = await initSDKTest()
const { privateKey } = sdk.wallet.keyPairFromSeedPhrase(mockPhrase!, 0)
- const { error, signature } = sdk.wallet.signTransaction(
+ const { error, signature } = sdk.wallet.signTransactionV1(
getConsensusState(),
getConsensusNetwork(),
getTransaction(),
@@ -160,13 +160,13 @@ describe('wallet', () => {
)
expect(error).toBeUndefined()
expect(signature).toEqual(
- 'sig:c58f8fe1ee5a08147484a53af7d3a64eca8039794b6c475342f0d8927b04d3172b3ed72861c183c73e87d719b782fb291dbfe8b3e0b1088095a9264bc97b6f06'
+ 'xY+P4e5aCBR0hKU699OmTsqAOXlLbEdTQvDYknsE0xcrPtcoYcGDxz6H1xm3gvspHb/os+CxCICVqSZLyXtvBg=='
)
})
it('errors if the signature index is invalid', async () => {
const sdk = await initSDKTest()
const { privateKey } = sdk.wallet.keyPairFromSeedPhrase(mockPhrase!, 0)
- const { error, signature } = sdk.wallet.signTransaction(
+ const { error, signature } = sdk.wallet.signTransactionV1(
getConsensusState(),
getConsensusNetwork(),
getTransaction(),
@@ -178,7 +178,7 @@ describe('wallet', () => {
})
it('errors if the private key is invalid', async () => {
const sdk = await initSDKTest()
- const { error, signature } = sdk.wallet.signTransaction(
+ const { error, signature } = sdk.wallet.signTransactionV1(
getConsensusState(),
getConsensusNetwork(),
getTransaction(),
diff --git a/libs/sdk/src/wasm.ts b/libs/sdk/src/wasm.ts
index d257acab8..f004fe447 100644
--- a/libs/sdk/src/wasm.ts
+++ b/libs/sdk/src/wasm.ts
@@ -5,7 +5,7 @@ export async function initWASM(): Promise {
try {
const go = new window.Go()
const source = await wasm(go.importObject)
- await go.run(source.instance)
+ go.run(source.instance)
} catch (e) {
throw new Error(`failed to initialize WASM: ${(e as Error).message}`)
}
diff --git a/libs/types/src/core.ts b/libs/types/src/core.ts
index aff0d7e63..3a297a5bb 100644
--- a/libs/types/src/core.ts
+++ b/libs/types/src/core.ts
@@ -8,6 +8,7 @@ export type OutputID = string
export type EncryptionKey = string
export type FileContractID = string
export type PublicKey = string
+export type PrivateKey = string
export type TransactionID = Hash256
export type SiacoinOutputID = Hash256
export type SiafundOutputID = Hash256
diff --git a/sdk/main.go b/sdk/main.go
index 8759cce1e..5ebf343d3 100644
--- a/sdk/main.go
+++ b/sdk/main.go
@@ -34,7 +34,7 @@ func main() {
"addressFromSpendPolicy": jsFunc(addressFromSpendPolicy),
"encodeTransaction": jsFunc(encodeTransaction),
"transactionId": jsFunc(transactionID),
- "signTransaction": jsFunc(signTransaction),
+ "signTransactionV1": jsFunc(signTransactionV1),
},
})
c := make(chan bool, 1)
diff --git a/sdk/wallet.go b/sdk/wallet.go
index 1d9cb04a4..0f9ad4611 100644
--- a/sdk/wallet.go
+++ b/sdk/wallet.go
@@ -3,6 +3,7 @@ package main
import (
"bytes"
"crypto/ed25519"
+ "encoding/base64"
"encoding/hex"
"fmt"
"syscall/js"
@@ -146,7 +147,7 @@ func encodeTransaction(this js.Value, args []js.Value) result {
}
// SignTransaction returns the signature of a transaction.
-func signTransaction(this js.Value, args []js.Value) result {
+func signTransactionV1(this js.Value, args []js.Value) result {
if err := checkArgs(args, js.TypeObject, js.TypeObject, js.TypeObject, js.TypeNumber, js.TypeString); err != nil {
return resultErr(err)
}
@@ -187,8 +188,9 @@ func signTransaction(this js.Value, args []js.Value) result {
} else {
sigHash = cs.PartialSigHash(txn, tsig.CoveredFields)
}
+ sig := privateKey.SignHash(sigHash)
return result(map[string]any{
- "signature": privateKey.SignHash(sigHash).String(),
+ "signature": base64.StdEncoding.EncodeToString(sig[:]),
})
}