Skip to content

Commit

Permalink
Merge pull-request #446
Browse files Browse the repository at this point in the history
  • Loading branch information
omkarshanbhag committed Dec 14, 2024
2 parents df1b6e7 + 12cded4 commit cc90919
Show file tree
Hide file tree
Showing 10 changed files with 690 additions and 8 deletions.
135 changes: 133 additions & 2 deletions examples/with-solana/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ This example walks through the following:
- Construction of a transaction sending the funds out with the `@turnkey/solana` signer
- Creating, minting, and transferring a SPL token using Turnkey

You can try this example quickly on Stackblitz. Follow the instructions below --> [Stackblitz Instructions](#6-stackblitz-example)
You can try this example quickly on Stackblitz. Follow the instructions below --> [Stackblitz Instructions](#7-stackblitz-example)

## Getting started

Expand Down Expand Up @@ -166,7 +166,138 @@ Token balance for warchest: 0.0001

Enjoy!

### 6/ Stackblitz Example
### 6/ Running the "Create SPL token transfer with policy" example

This example shows the flow of using our policy engine's support for SPL token transfers! Specifically we show how to allow a user (non root user) to send a Solana SPL token transfer by calculating the "associated token address" of the receiving account, given the mint address of the token that we are transferring.

To best see how policies can be used effectively, we will go through example in the following steps

1. Setup - this step creates the on-chain state and SOME of the Turnkey setup state required to make this transfer (notably, it leaves out the creation of the policy that allows the created non root user to sign the SPL transfer)
2. Attempt Transfer - since we're attempting the transfer WITHOUT having created the policy allowing it, this will fail
3. Create Token Policy - this will create a policy that allows Solana transactions initiated by the correct user to be signed if they contain a single instruction which is an SPL token transfer to the correct token account
4. Attempt Transfer (again) - this will succeed, now that the appropriate policy has been created!

NOTE: A very key piece of understanding to glean from this example is the use of "associated token addresses (or "associated token account address") in the creation of SPL related policies. For further context on ATA's look here: https://spl.solana.com/associated-token-account

#### Step 1. Setup

First we will run the setup command

```bash
$ pnpm token-transfer-policy setup
```

This will create a Solana wallet address that you'll have to fund with devnet SOL via a faucet or cli. A commonly used SOL devnet faucet is available here --> https://faucet.solana.com/

Funding your devnet and hitting `y` in the terminal to proceed will set up the necessary on-chain state and Turnkey setup to start, and will output the following information to be used in coming steps

- Turnkey Solana wallet address
- Token Mint public key
- Non root user created with user id

```bash
✔ Ready to Continue? … yes
Broadcasting token creation transaction...
Transaction broadcast and confirmed! 🎉
https://explorer.solana.com/tx/35SJqgdy2nWBvuECXrqRVkuYdRf6kkmWCuxNFd3TcmdMsXfrEns6nLjQqhKx5zvUwYPZ8xHnsiYCRtuhS8a9a5XX?cluster=devnet

Broadcasting token account creation transaction...
Transaction broadcast and confirmed! 🎉
https://explorer.solana.com/tx/3ffN8Goe4P3TSbJZGrmBADfGZWVdnrHNTsbNqce8AAPbANFmhWVHhd5MMhinXhppccMMvjqPqj8vz1PbFQUaV8Ns?cluster=devnet

Broadcasting token account creation transaction...
Transaction broadcast and confirmed! 🎉
https://explorer.solana.com/tx/4sTgMDSMvf1oTJenJWiGRapf6mhoghv85juqdTv5VmZuyQ48ykgf22RZPxdCACSEEg8ige6GarSEQc2Nr4NkfzY1?cluster=devnet

Broadcasting mint transaction...
Transaction broadcast and confirmed! 🎉
https://explorer.solana.com/tx/FMBBXrXYGb2nGQqbHs1j17GJG5owzSMnZ1RqN9R7cSVn8Q2JMbZKMik9MLzvG1LNmdrsmSV16yvFdyLHukZw5KJ?cluster=devnet

New user created!
- Name: Non Root User
- User ID: 13c4609b-8818-40be-aeac-c8b63da0de4a

Setup complete -- token mint and token accounts created, non root user created
Turnkey Solana wallet address: 2bwUkGZ72SNyzBWFwfzKVrVWJ9JNZggjqFPDM9X9mrp9
Token Mint public key: FA7umztm6eYcENE22ndEZhd8MCf9C7myJ2KjfCwobj8F
Non root user created with user id: 13c4609b-8818-40be-aeac-c8b63da0de4a
```

#### Step 2. Attempt to Transfer Tokens (This is expected to FAIL)

We will now attempt to transfer these SPL tokens that we've created

```bash
$ pnpm token-transfer-policy attempt-transfer
```

This will prompt you first to enter in the Solana wallet address that is originating the transfer (use the address labeled "Turnkey Solana wallet address" output at the end of the setup step)

```bash
? Enter Solana wallet address originating transfer (created during setup stage):
```

Once you've entered your wallet address you will then be prompted to enter in the Mint address/public key for the token being transferred (use the address labeled "Token Mint public key" output at the end of the setup step)

```bash
? Enter Mint account address of token being transferred (created during setup stage):
```

Since this is being initiated by a non root user, and the policy allowing this has not been created, this is EXPECTED TO FAIL. You will get an error that includes the following information saying that no policies evaluated to the outcome: Allow

```bash
details: [
{
'@type': 'type.googleapis.com/errors.v1.PolicyEnginePermissionError',
message: 'No policies evaluated to outcome: Allow',
policyEvaluations: []
}
],
code: 7
```

#### Step 3. Create Policy allowing the correct user to make SPL transfers to receiving Associated Token Address

This is the most important step! In this step we are taking our receiving address (the Turnkey Warchest), calculating it's "associated token address", given the Mint account address for the token we've created, and creating a policy that allows the correct non root user to sign Solana transactions if and only if they contain a single instruction that is an SPL token transfer to this associated token address.

```bash
$ pnpm token-transfer-policy create-token-policy
```

This will prompt you first to enter in the user ID of the non root user who's credentials are being used to sign the transfer (use the value labeled "Non root user created with user id" output at the end of the setup step)

```bash
? Enter non-root user ID originating the transfer (created during setup stage):
```

Once you've entered this user id, you will then be prompted to enter in the Mint address/public key for the token being transferred (use the address labeled "Token Mint public key" output at the end of the setup step)

```bash
? Enter Mint account address of token being transferred (created during setup stage):
```

This should successfully create your policy!

```bash
New policy created!
- Name: Let non root user send SPL transfers to the associated token account of WARCHEST: 6nDvKk6emwskFLtEpbQgFX6rsMtzbNY4LkvaNnzkvBaq given the mint address for the token just created
- Policy ID: f715fda0-2236-4904-b4be-9098f7c192b8
- Effect: EFFECT_ALLOW
- Consensus: approvers.any(user, user.id == '96a10eb7-0389-447e-bc50-b768a76c0d7f')
- Condition: solana.tx.instructions.count() == 1 && solana.tx.spl_transfers.any(transfer, transfer.to == '6nDvKk6emwskFLtEpbQgFX6rsMtzbNY4LkvaNnzkvBaq')
```

#### Step 4. Attempt to Transfer Tokens (This is expected to SUCCEED)

Following the instructions for step 2 above should find success this time -- showing that the policy has successfully allowed the user to transfer tokens!

```bash
Broadcasting token transfer transaction...
Transaction broadcast and confirmed! 🎉
https://explorer.solana.com/tx/GQa4uqoS2zU9uGsGuzMBAq4UmR4dYPXAWgfS4cHQ5Lmqdwb3pKwXrEc4jGHEaZLCyZmCZWErsaPWecN8udkBXaT?cluster=devnet
```

### 7/ Stackblitz Example

Example Link: https://stackblitz.com/edit/stackblitz-starters-xeb93i

Expand Down
1 change: 1 addition & 0 deletions examples/with-solana/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
"start": "pnpm tsx src/index.ts",
"advanced": "pnpm tsx src/advanced.ts",
"token-transfer": "pnpm tsx src/tokenTransfer.ts",
"token-transfer-policy": "pnpm tsx src/tokenTransferPolicy.ts",
"faucet": "pnpm tsx src/faucet.ts",
"jupiter-swap": "pnpm tsx src/jupiterSwap.ts",
"with-fee-payer": "pnpm tsx src/withFeePayer.ts",
Expand Down
13 changes: 13 additions & 0 deletions examples/with-solana/src/keys.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
// NOTE: All of these keys should be considered compromised!!!!
// NOTE: Only used for demo purposes

const keys: { [key: string]: { [key: string]: string } } = {
nonRootUser: {
publicKey:
"03178b1cb727bd1ab23c3c5725d633f9ad3c7c0502efda2371e4e4bd88aea8a94d",
privateKey:
"26e53957146d5d5b5612f09518b8bb40deddcf28d378270101227cb0a231969e",
},
};

export default keys;
52 changes: 52 additions & 0 deletions examples/with-solana/src/requests/createPolicy.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
import {
type TurnkeyServerClient,
TurnkeyActivityError,
} from "@turnkey/sdk-server";

import { refineNonNull } from "../utils";

export default async function createPolicy(
turnkeyClient: TurnkeyServerClient,
policyName: string,
effect: "EFFECT_ALLOW" | "EFFECT_DENY",
consensus: string,
condition: string,
notes: ""
): Promise<string> {
try {
const { policyId } = await turnkeyClient.createPolicy({
policyName,
condition,
consensus,
effect,
notes,
});

const newPolicyId = refineNonNull(policyId);

// Success!
console.log(
[
`New policy created!`,
`- Name: ${policyName}`,
`- Policy ID: ${newPolicyId}`,
`- Effect: ${effect}`,
`- Consensus: ${consensus}`,
`- Condition: ${condition}`,
``,
].join("\n")
);

return newPolicyId;
} catch (error) {
// If needed, you can read from `TurnkeyActivityError` to find out why the activity didn't succeed
if (error instanceof TurnkeyActivityError) {
throw error;
}

throw new TurnkeyActivityError({
message: "Failed to create a new policy",
cause: error as Error,
});
}
}
55 changes: 55 additions & 0 deletions examples/with-solana/src/requests/createUser.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
import {
type TurnkeyServerClient,
TurnkeyActivityError,
} from "@turnkey/sdk-server";

import { refineNonNull } from "../utils";

export default async function createUser(
turnkeyClient: TurnkeyServerClient,
userName: string,
apiKeyName: string,
publicKey: string
): Promise<string> {
let userTags: string[] = new Array();
try {
const { userIds } = await turnkeyClient.createApiOnlyUsers({
apiOnlyUsers: [
{
userName,
userTags,
apiKeys: [
{
apiKeyName,
publicKey,
},
],
},
],
});

const userId = refineNonNull(userIds?.[0]);

// Success!
console.log(
[
`New user created!`,
`- Name: ${userName}`,
`- User ID: ${userId}`,
``,
].join("\n")
);

return userId;
} catch (error) {
// If needed, you can read from `TurnkeyActivityError` to find out why the activity didn't succeed
if (error instanceof TurnkeyActivityError) {
throw error;
}

throw new TurnkeyActivityError({
message: "Failed to create a new user",
cause: error as Error,
});
}
}
2 changes: 2 additions & 0 deletions examples/with-solana/src/requests/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export { default as createUser } from "./createUser";
export { default as createPolicy } from "./createPolicy";
4 changes: 2 additions & 2 deletions examples/with-solana/src/tokenTransfer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ import {
createNewSolanaWallet,
createToken,
createTokenAccount,
createTokenTransfer,
createTokenTransferAddSignature,
solanaNetwork,
TURNKEY_WAR_CHEST,
} from "./utils";
Expand Down Expand Up @@ -125,7 +125,7 @@ async function main() {
);

// Transfer token from primary to Warchest
await createTokenTransfer(
await createTokenTransferAddSignature(
turnkeySigner,
connection,
solAddress,
Expand Down
Loading

0 comments on commit cc90919

Please sign in to comment.