Skip to content

Commit d6f5170

Browse files
[thirdweb] feat: Support account overrides in engineAccount()
1 parent 56bc4ae commit d6f5170

File tree

3 files changed

+76
-3
lines changed

3 files changed

+76
-3
lines changed

.changeset/fifty-hoops-build.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"thirdweb": patch
3+
---
4+
5+
Support account overrides in engineAccount()

packages/thirdweb/src/wallets/engine/engine-account.test.ts

+42
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,8 @@ import { sepolia } from "../../chains/chain-definitions/sepolia.js";
66
import { getContract } from "../../contract/contract.js";
77
import { claimTo } from "../../extensions/erc1155/drops/write/claimTo.js";
88
import { sendTransaction } from "../../transaction/actions/send-transaction.js";
9+
import { smartWallet } from "../smart/smart-wallet.js";
10+
import { generateAccount } from "../utils/generateAccount.js";
911
import { engineAccount } from "./index.js";
1012

1113
describe.runIf(
@@ -66,4 +68,44 @@ describe.runIf(
6668
});
6769
expect(tx).toBeDefined();
6870
});
71+
72+
it.skip("should send a session key tx", async () => {
73+
const personalAccount = await generateAccount({
74+
client: TEST_CLIENT,
75+
});
76+
const smart = smartWallet({
77+
chain: sepolia,
78+
sponsorGas: true,
79+
sessionKey: {
80+
address: process.env.ENGINE_WALLET_ADDRESS_EOA as string,
81+
permissions: {
82+
approvedTargets: "*",
83+
},
84+
},
85+
});
86+
const smartAccount = await smart.connect({
87+
client: TEST_CLIENT,
88+
personalAccount,
89+
});
90+
91+
const engineAcc = engineAccount({
92+
engineUrl: process.env.ENGINE_URL as string,
93+
authToken: process.env.ENGINE_AUTH_TOKEN as string,
94+
walletAddress: process.env.ENGINE_WALLET_ADDRESS_EOA as string,
95+
chain: sepolia,
96+
overrides: {
97+
accountAddress: smartAccount.address,
98+
},
99+
});
100+
const tx = await sendTransaction({
101+
account: engineAcc,
102+
transaction: {
103+
client: TEST_CLIENT,
104+
chain: sepolia,
105+
to: TEST_ACCOUNT_B.address,
106+
value: 0n,
107+
},
108+
});
109+
expect(tx).toBeDefined();
110+
});
69111
});

packages/thirdweb/src/wallets/engine/index.ts

+29-3
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,20 @@ export type EngineAccountOptions = {
1919
* The backend wallet to use for sending transactions inside engine.
2020
*/
2121
walletAddress: string;
22+
overrides?: {
23+
/**
24+
* The address of the smart account to act on behalf of. Requires your backend wallet to be a valid signer on that smart account.
25+
*/
26+
accountAddress?: string;
27+
/**
28+
* The address of the smart account factory to use for creating smart accounts.
29+
*/
30+
accountFactoryAddress?: string;
31+
/**
32+
* The salt to use for creating the smart account.
33+
*/
34+
accountSalt?: Hex;
35+
};
2236
/**
2337
* The chain to use for signing messages and typed data (smart backend wallet only).
2438
*/
@@ -55,7 +69,7 @@ export type EngineAccountOptions = {
5569
* ```
5670
*/
5771
export function engineAccount(options: EngineAccountOptions): Account {
58-
const { engineUrl, authToken, walletAddress, chain } = options;
72+
const { engineUrl, authToken, walletAddress, chain, overrides } = options;
5973

6074
// these are shared across all methods
6175
const headers: HeadersInit = {
@@ -64,6 +78,16 @@ export function engineAccount(options: EngineAccountOptions): Account {
6478
"Content-Type": "application/json",
6579
};
6680

81+
if (overrides?.accountAddress) {
82+
headers["x-account-address"] = overrides.accountAddress;
83+
}
84+
if (overrides?.accountFactoryAddress) {
85+
headers["x-account-factory-address"] = overrides.accountFactoryAddress;
86+
}
87+
if (overrides?.accountSalt) {
88+
headers["x-account-salt"] = overrides.accountSalt;
89+
}
90+
6791
return {
6892
address: walletAddress,
6993
sendTransaction: async (transaction: SendTransactionOption) => {
@@ -181,12 +205,14 @@ export function engineAccount(options: EngineAccountOptions): Account {
181205
domain: _typedData.domain,
182206
types: _typedData.types,
183207
value: _typedData.message,
208+
primaryType: _typedData.primaryType,
209+
chainId: chain?.id,
184210
}),
185211
});
186212
if (!engineRes.ok) {
187-
engineRes.body?.cancel();
213+
const body = await engineRes.text();
188214
throw new Error(
189-
`Engine request failed with status ${engineRes.status}`,
215+
`Engine request failed with status ${engineRes.status} - ${body}`,
190216
);
191217
}
192218
const engineJson = (await engineRes.json()) as {

0 commit comments

Comments
 (0)