Skip to content

Commit

Permalink
calculate object address and token address
Browse files Browse the repository at this point in the history
  • Loading branch information
0xmaayan committed Mar 25, 2024
1 parent 2bf4f02 commit cea70ad
Show file tree
Hide file tree
Showing 5 changed files with 95 additions and 0 deletions.
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,8 @@ All notable changes to the Aptos TypeScript SDK will be captured in this file. T
- Add support to allow setting per-backend (fullnode, indexer, faucet) configuration
- [`Breaking`] `AUTH_TOKEN` client config moved to be under `faucetConfig` property
- Handle `Unauthorized` server error
- Add function to create object address locally
- Add function to create token object address locally

# 1.10.0 (2024-03-11)

Expand Down
1 change: 1 addition & 0 deletions src/core/account/index.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
export * from "./Ed25519Account";
export * from "./Account";
export * from "./SingleKeyAccount";
export * from "./utils";
51 changes: 51 additions & 0 deletions src/core/account/utils/address.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
import { sha3_256 } from "@noble/hashes/sha3";
import { AccountAddress } from "../../accountAddress";
import { Hex } from "../../hex";

/**
* Scheme identifier used to derive addresses of objects
* {@link https://github.com/aptos-labs/aptos-core/blob/main/types/src/transaction/authenticator.rs#L460}
*/
export const DERIVE_OBJECT_ADDRESS_FROM_SEED_SCHEME = 254;

/**
* Creates an object address from creator address and seed
*
* @param creatorAddress The object creator account address
* @param seed The seed in either Uint8Array | string type
*
* @returns The object address
*/
export const createObjectAddress = (creatorAddress: AccountAddress, seed: Uint8Array | string): string => {
const creatorBytes = creatorAddress.bcsToBytes();

let seedBytes = seed;
if (typeof seedBytes === "string") {
seedBytes = new TextEncoder().encode(seedBytes);
}

const bytes = new Uint8Array([...creatorBytes, ...seedBytes, DERIVE_OBJECT_ADDRESS_FROM_SEED_SCHEME]);

const hash = sha3_256.create();
hash.update(bytes);

return Hex.fromHexInput(hash.digest()).toString();
};

/**
* Creates a token object address from creator address, collection name and token name
*
* @param creatorAddress The token creator account address
* @param collectionName The collection name
* @param tokenName The token name
*
* @returns The token address
*/
export const createTokenAddress = (
creatorAddress: AccountAddress,
collectionName: string,
tokenName: string,
): string => {
const seed = new TextEncoder().encode(`${collectionName}::${tokenName}`);
return createObjectAddress(creatorAddress, seed);
};
1 change: 1 addition & 0 deletions src/core/account/utils/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export * from "./address";
40 changes: 40 additions & 0 deletions tests/unit/address.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
/* eslint-disable max-len */
import { AccountAddress } from "../../src";
import { createTokenAddress, createObjectAddress } from "../../src/core/account/utils/address";

describe("address", () => {
/**
* Reference: {@link https://explorer.aptoslabs.com/account/0xf0995d360365587c500cc171d1416bad10a331b9c71871a1aec5f2c37ff43124/modules/view/migration_helper/migration_object_address?network=testnet}
* creatorAddr = 0x120e79e45d21ef439963580c77a023e2729db799e96e61f878fac98fde5b9cc9
* seed = "migration::migration_contract"
* Expect = 0xbe376272a5c4361ee96bc147525b26b3bf2ee25f433cbd410a7b3b4b881ffcbf
*/
test("create an object address from creator address and seed as Uint8Array type", () => {
const creatorAddress = AccountAddress.from("0x120e79e45d21ef439963580c77a023e2729db799e96e61f878fac98fde5b9cc9");
const seed = new TextEncoder().encode("migration::migration_contract");
const address = createObjectAddress(creatorAddress, seed);
expect(address).toEqual("0xbe376272a5c4361ee96bc147525b26b3bf2ee25f433cbd410a7b3b4b881ffcbf");
});

test("create an object address from creator address and seed as string type", () => {
const creatorAddress = AccountAddress.from("0x120e79e45d21ef439963580c77a023e2729db799e96e61f878fac98fde5b9cc9");
const seed = "migration::migration_contract";
const address = createObjectAddress(creatorAddress, seed);
expect(address).toEqual("0xbe376272a5c4361ee96bc147525b26b3bf2ee25f433cbd410a7b3b4b881ffcbf");
});

/**
* Reference: {@link https://explorer.aptoslabs.com/txn/473081244/userTxnOverview?network=mainnet}
* creatorAddr = 0x9d518b9b84f327eafc5f6632200ea224a818a935ffd6be5d78ada250bbc44a6
* collectionName = SuperV Villains
* tokenName = Nami #5962
* Expect = 0x44697f48d1e1a899953b4ea6c03a92c567f3741f0b415a74d1c23cdf141368be
*/
test("create token object address", () => {
const creatorAddress = AccountAddress.from("0x9d518b9b84f327eafc5f6632200ea224a818a935ffd6be5d78ada250bbc44a6");
const collectionName = "SuperV Villains";
const tokenName = "Nami #5962";
const address = createTokenAddress(creatorAddress, collectionName, tokenName);
expect(address).toEqual("0x44697f48d1e1a899953b4ea6c03a92c567f3741f0b415a74d1c23cdf141368be");
});
});

0 comments on commit cea70ad

Please sign in to comment.