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
use anchor_lang::{system_program, InstructionData, ToAccountMetas};
use reqwest::multipart::{Form, Part};
use serde_json::Value;
use shadow_drive_user_staking::accounts as shdw_drive_accounts;
use shadow_drive_user_staking::instruction as shdw_drive_instructions;
use solana_client::rpc_client::serialize_and_encode;
use solana_sdk::{
instruction::Instruction, pubkey::Pubkey, signer::Signer, transaction::Transaction,
};
use solana_transaction_status::UiTransactionEncoding;
use std::str::FromStr;
use super::ShadowDriveClient;
use crate::{
constants::{PROGRAM_ADDRESS, SHDW_DRIVE_ENDPOINT, STORAGE_CONFIG_PDA, TOKEN_MINT, UPLOADER},
error::Error,
models::*,
};
impl<T> ShadowDriveClient<T>
where
T: Signer + Send + Sync,
{
pub async fn edit_file(
&self,
storage_account_key: &Pubkey,
url: &str,
data: ShadowFile,
) -> ShadowDriveResult<ShadowUploadResponse> {
let upload_data = data
.prepare_upload(storage_account_key)
.await
.map_err(Error::FileValidationError)?;
let selected_account = self.get_storage_account(storage_account_key).await?;
let existing_file_data = self.get_object_data(url).await?;
let file_owner_on_chain =
Pubkey::from_str(&existing_file_data.file_data.owner_account_pubkey)?;
if file_owner_on_chain != self.wallet.pubkey() {
return Err(Error::NotFileOwner);
}
let file_acct = Pubkey::from_str(&existing_file_data.file_data.file_account_pubkey)?;
let form = Form::new().part("file", upload_data.to_form_part().await?);
let accounts = shdw_drive_accounts::EditFile {
storage_config: *STORAGE_CONFIG_PDA,
storage_account: *storage_account_key,
file: file_acct,
owner: selected_account.owner_1,
uploader: UPLOADER,
token_mint: TOKEN_MINT,
system_program: system_program::ID,
};
let args = shdw_drive_instructions::EditFile {
sha256_hash: hex::encode(upload_data.sha256_hash.into_bytes()),
size: upload_data.size,
};
let instruction = Instruction {
program_id: PROGRAM_ADDRESS,
accounts: accounts.to_account_metas(None),
data: args.data(),
};
let mut txn = Transaction::new_with_payer(&[instruction], Some(&self.wallet.pubkey()));
txn.try_partial_sign(&[&self.wallet], self.rpc_client.get_latest_blockhash()?)?;
let txn_encoded = serialize_and_encode(&txn, UiTransactionEncoding::Base64)?;
let form = form.part("transaction", Part::text(txn_encoded));
let response = self
.http_client
.post(format!("{}/upload", SHDW_DRIVE_ENDPOINT))
.multipart(form)
.send()
.await?;
if !response.status().is_success() {
return Err(Error::ShadowDriveServerError {
status: response.status().as_u16(),
message: response.json::<Value>().await?,
});
}
let response = response.json::<ShadowUploadResponse>().await?;
Ok(response)
}
}