1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
//! CLI for creating Aptos resource accounts.
//!
//! *Note: this CLI will be changed in the near future, since resource accounts are still being developed in aptos-core.*
//!
//! # Setup
//!
//! To use this CLI, you may install this crate via Cargo:
//!
//! ```bash
//! cargo install aptos-create-resource-account
//! ```
//!
//! You should also have an Aptos CLI configuration file, which can be created using:
//!
//! ```bash
//! aptos init
//! ```
//!
//! # Usage
//!
//! Run the following command to create a new resource account:
//!
//! ```bash
//! aptos-create-resource-account my-account
//! ```
//!
//! This will create a profile in your `.aptos/config.yaml` with the name `my-account`.
//! You can then use this profile to publish modules, run scripts, etc.

use anyhow::{Context, Result};
use aptos_cli_common::{
    utils::fund_account, FaucetOptions, TransactionOptions, TransactionSummary,
};
use aptos_cli_config::{CliConfig, ProfileConfig};
use aptos_crypto::{ed25519::Ed25519PublicKey, HashValue};
use aptos_keygen::KeyGen;
use aptos_rest_client::Transaction;
use aptos_types::{
    account_address::AccountAddress,
    transaction::{authenticator::AuthenticationKey, ScriptFunction, TransactionPayload},
};
use async_trait::async_trait;
use clap::Parser;
use clitool::CliTool;
use move_core_types::{ident_str, language_storage::ModuleId};
use serde::{Deserialize, Serialize};

pub const DEFAULT_FUNDED_COINS: u64 = 10000;

/// Module address of the deployer.
pub const DEPLOYER_ADDRESS: AccountAddress = static_address::static_address!(
    "0x1245d0cf838606de0efd8bdfcc80b80cb4198f589b14ecac66ccc83035102c00"
);

/// Command to create a new account on-chain
///
#[derive(Debug, Parser)]
#[clap(name = "aptos-create-resource-account", about, version)]
pub struct AptosCreateResourceAccountTool {
    #[clap(flatten)]
    pub transaction_options: TransactionOptions,
    /// Flag for using faucet to create the account
    #[clap(long)]
    pub use_faucet: bool,
    #[clap(flatten)]
    pub faucet_options: FaucetOptions,
    /// Initial coins to fund when using the faucet
    #[clap(long, default_value_t = DEFAULT_FUNDED_COINS)]
    pub initial_coins: u64,
    /// Name of the profile to create. This is also the seed.
    pub new_profile: String,
}

#[async_trait]
impl CliTool<AptosCreateResourceAccountResult> for AptosCreateResourceAccountTool {
    async fn execute(self) -> Result<AptosCreateResourceAccountResult> {
        let result = self.generate_profile().await?;

        if self.use_faucet {
            fund_account(
                self.faucet_options.faucet_url(&result.profile.name)?,
                self.initial_coins,
                &result.profile.account,
            )
            .await?;
        }

        Ok(result)
    }
}

/// An individual profile
#[derive(Clone, Debug, Serialize)]
pub struct AptosCreateResourceAccountResult {
    pub profile: ProfileInfo,
    pub tx: TransactionSummary,
}

/// An individual profile
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct ProfileInfo {
    /// Profile name.
    pub name: String,
    /// Public key for commands
    pub public_key: Ed25519PublicKey,
    /// Account created.
    pub account: AccountAddress,
}

/// Gets the address of a resource account.
pub fn get_resource_account_address(
    source: &AccountAddress,
    seed: &[u8],
) -> Result<AccountAddress> {
    AccountAddress::from_bytes(
        HashValue::sha3_256_of(&[bcs::to_bytes(source)?, seed.to_vec()].concat()).as_ref(),
    )
    .with_context(|| "Failed to generate resource account address")
}

impl AptosCreateResourceAccountTool {
    /// Creates the resource account using the provided public key.
    pub async fn create_resource_account_with_public_key(
        &self,
        public_key: &Ed25519PublicKey,
    ) -> Result<(Transaction, AccountAddress)> {
        let auth_key = AuthenticationKey::ed25519(public_key);
        let profile_bytes = &bcs::to_bytes(self.new_profile.as_bytes())?;
        let auth_key_bytes = &bcs::to_bytes(&auth_key)?;
        let create_account_fn = TransactionPayload::ScriptFunction(ScriptFunction::new(
            ModuleId::new(DEPLOYER_ADDRESS, ident_str!("deployer").to_owned()),
            ident_str!("create_resource_account").to_owned(),
            vec![],
            vec![profile_bytes.clone(), auth_key_bytes.clone()],
        ));

        let address = get_resource_account_address(
            &self.transaction_options.profile_options.account_address()?,
            self.new_profile.as_bytes(),
        )?;
        println!("Creating account at: {}", address);

        let tx = self
            .transaction_options
            .submit_transaction(create_account_fn)
            .await?;

        Ok((tx, address))
    }

    async fn generate_profile(&self) -> Result<AptosCreateResourceAccountResult> {
        // First we must generate a new keypair. This will be the keypair of the created account.
        let (module_private_key, module_auth_key) =
            KeyGen::from_os_rng().generate_ed25519_keypair();

        // Next we'll create the account as a resource account.
        let (create_tx, module_account_key) = self
            .create_resource_account_with_public_key(&module_auth_key)
            .await?;

        let rest_url = self
            .transaction_options
            .rest_options
            .url(&self.transaction_options.profile_options.profile)?;

        let new_profile = ProfileConfig {
            private_key: Some(module_private_key),
            public_key: Some(module_auth_key.clone()),
            account: Some(module_account_key),
            rest_url: Some(rest_url.to_string()),
            faucet_url: Some(
                self.faucet_options
                    .faucet_url(&self.transaction_options.profile_options.profile)?
                    .to_string(),
            ),
        };

        let config: &mut CliConfig = &mut CliConfig::load()?;
        config
            .profiles
            .as_mut()
            .unwrap()
            .insert(self.new_profile.clone(), new_profile);
        config.save()?;

        let profile_info = ProfileInfo {
            name: self.new_profile.clone(),
            public_key: module_auth_key.clone(),
            account: module_account_key,
        };
        Ok(AptosCreateResourceAccountResult {
            profile: profile_info,
            tx: create_tx.into(),
        })
    }
}