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
use std::fs;
use std::io::Write;
use std::str::FromStr;
use std::sync::Arc;

use clap::ArgMatches;
use serde::{Deserialize, Serialize};
use solana_clap_utils::keypair::DefaultSigner;
use solana_client::rpc_client::RpcClient;
use solana_program::pubkey::Pubkey;
use solana_remote_wallet::remote_wallet::RemoteWalletManager;
use solana_sdk::signature::{Keypair, Signer};
use tabled::Tabled;

use crate::check_and_update_err;
use crate::command::{CliCommand, CliCommandInfo, CliConfig, CliError, ProcessResult};
use crate::contract::instructions::create_partner::new_create_partner;
use crate::program::get_pubkey_for_program_with_seeds;
use crate::utils::file::{get_template_dir, read_for_str};
use crate::utils::send::send_tx;

pub const PARTNER_TEMPLATE_DIR: &str = "./partner-template.yaml";

#[derive(Debug, Serialize, Deserialize, Tabled)]
pub struct PartnerTemplate {
    pub authority: String,
    pub partner_fee_claim_authority: String,
    pub fee_rate: f64,
    pub start_time: u64,
    pub end_time: u64,
    pub name: String,
}


pub fn parse_create_partner<'a>(matches: &'a ArgMatches, default_signer: &DefaultSigner, wallet_manager: &mut Option<Arc<RemoteWalletManager>>) -> Result<CliCommandInfo<'a>, CliError> {
    let entry_file = get_template_dir(matches, "entry_file", PARTNER_TEMPLATE_DIR);
    Ok(CliCommandInfo {
        command: CliCommand::PairPartnerCreate {
            entry_file: entry_file.unwrap(),
        },
        signers: vec![check_and_update_err!(default_signer.signer_from_path(matches, wallet_manager), CliError::RpcRequestError("owner key is invalid".to_string()))?],
    })
}


pub fn process_create_partner(rpc_client: &RpcClient, config: &mut CliConfig, output: &str) -> ProcessResult {
    let partner: PartnerTemplate = read_for_str(output);

    let base_keypair = Keypair::new();
    let base_pubkey = base_keypair.pubkey();

    let (partner_pubkey, _) = get_pubkey_for_program_with_seeds(
        &[
            b"partner",
            base_keypair.pubkey().as_ref(),
        ]
    );

    config.signers.push(Box::new(base_keypair));

    let ixs = [new_create_partner(&partner, partner_pubkey, base_pubkey, config.pubkey().unwrap())];

    let res = send_tx(rpc_client, config, &ixs)?;

    println!("partner base key: {}", base_pubkey.to_string());
    println!("partner key: {}", partner_pubkey.to_string());

    Ok("signers : ".to_owned() + res.to_string().as_str())
}

pub fn parse_create_partner_template<'a>(matches: &'a ArgMatches) -> Result<CliCommandInfo<'a>, CliError> {
    let output_file = get_template_dir(matches, "output-file", PARTNER_TEMPLATE_DIR);
    Ok(CliCommandInfo {
        command: CliCommand::PairPartnerCreateTemplate {
            output_file: output_file.unwrap(),
        },
        signers: vec![],
    })
}

pub fn process_create_partner_template(output: &str) -> ProcessResult {
    let partner_template = PartnerTemplate {
        authority: Pubkey::from_str("BPFLoaderUpgradeab1e11111111111111111111111").unwrap().to_string(),
        partner_fee_claim_authority: Pubkey::from_str("BPFLoaderUpgradeab1e11111111111111111111111").unwrap().to_string(),
        fee_rate: 0.0001,
        start_time: 100,
        end_time: 100,
        name: "partner name".to_string(),
    };

    let mut file = fs::File::create(output).expect("create failed");
    file.write_all(serde_yaml::to_string(&partner_template).unwrap().as_bytes()).expect("write failed");

    Ok(output.to_string() + " file create success".to_string().as_str())
}