Skip to content

Commit

Permalink
Merge branch 'master' into gn/update-readme
Browse files Browse the repository at this point in the history
  • Loading branch information
German authored Nov 28, 2023
2 parents b2321ab + 6cd6feb commit 6f2d393
Show file tree
Hide file tree
Showing 19 changed files with 594 additions and 87 deletions.
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -22,9 +22,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Add workspace support -[#1358](https://github.com/paritytech/cargo-contract/pull/1358)
- Add `Storage Total Deposit` to `info` command output - [#1347](https://github.com/paritytech/cargo-contract/pull/1347)
- Add dynamic types support - [#1399](https://github.com/paritytech/cargo-contract/pull/1399)
- Basic storage inspection command - [#1395](https://github.com/paritytech/cargo-contract/pull/1395)

### Fixed
- Do not allow to execute calls on immutable contract messages - [#1397](https://github.com/paritytech/cargo-contract/pull/1397)
- Improve JSON Output for Upload and Remove Commands - [#1389](https://github.com/paritytech/cargo-contract/pull/1389)

## [4.0.0-alpha]

Expand Down
25 changes: 13 additions & 12 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 5 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -165,6 +165,11 @@ Generate schema and print it to STDOUT.

Verify a metadata file or a contract bundle containing metadata against the schema file.

##### `cargo contract storage`

Fetch and display the storage of a contract on chain.


## Publishing

In order to publish a new version of `cargo-contract`:
Expand Down
4 changes: 2 additions & 2 deletions crates/build/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ anyhow = "1.0.75"
blake2 = "0.10.6"
cargo_metadata = "0.18.1"
colored = "2.0.4"
clap = { version = "4.4.8", features = ["derive", "env"] }
clap = { version = "4.4.10", features = ["derive", "env"] }
duct = "0.13.6"
heck = "0.4.0"
hex = "0.4.3"
Expand All @@ -34,7 +34,7 @@ serde = { version = "1", default-features = false, features = ["derive"] }
serde_json = "1.0.108"
tempfile = "3.8.1"
term_size = "0.3.2"
url = { version = "2.4.1", features = ["serde"] }
url = { version = "2.5.0", features = ["serde"] }
wasm-opt = { version = "=0.116.0", deafult-features = false }
which = "5.0.0"
zip = { version = "0.6.6", default-features = false }
Expand Down
4 changes: 2 additions & 2 deletions crates/cargo-contract/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -25,15 +25,15 @@ contract-metadata = { version = "4.0.0-alpha", path = "../metadata" }
contract-analyze = { version = "0.1.0", path = "../analyze" }

anyhow = "1.0.75"
clap = { version = "4.4.8", features = ["derive", "env"] }
clap = { version = "4.4.10", features = ["derive", "env"] }
primitive-types = { version = "0.12.2", default-features = false, features = ["codec", "scale-info", "serde"] }
tracing = "0.1.40"
tracing-subscriber = { version = "0.3.18", features = ["env-filter"] }
which = "5.0.0"
colored = "2.0.4"
serde_json = "1.0.108"
serde = { version = "1.0.193", default-features = false, features = ["derive"] }
url = { version = "2.4.1", features = ["serde"] }
url = { version = "2.5.0", features = ["serde"] }
semver = "1.0"
jsonschema = "0.17"
schemars = "0.8"
Expand Down
5 changes: 3 additions & 2 deletions crates/cargo-contract/src/cmd/info.rs
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ use contract_extrinsics::{
CodeHash,
ContractInfo,
ErrorVariant,
TrieId,
};
use std::{
fmt::Debug,
Expand Down Expand Up @@ -139,7 +140,7 @@ impl InfoCommand {

#[derive(serde::Serialize)]
pub struct ExtendedContractInfo {
pub trie_id: String,
pub trie_id: TrieId,
pub code_hash: CodeHash,
pub storage_items: u32,
pub storage_items_deposit: Balance,
Expand All @@ -154,7 +155,7 @@ impl ExtendedContractInfo {
None => "Unknown".to_string(),
};
ExtendedContractInfo {
trie_id: contract_info.trie_id().to_string(),
trie_id: contract_info.trie_id().clone(),
code_hash: *contract_info.code_hash(),
storage_items: contract_info.storage_items(),
storage_items_deposit: contract_info.storage_items_deposit(),
Expand Down
4 changes: 3 additions & 1 deletion crates/cargo-contract/src/cmd/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ pub mod info;
pub mod instantiate;
pub mod remove;
pub mod schema;
pub mod storage;
pub mod upload;
pub mod verify;

Expand All @@ -42,6 +43,7 @@ pub(crate) use self::{
GenerateSchemaCommand,
VerifySchemaCommand,
},
storage::StorageCommand,
upload::UploadCommand,
verify::VerifyCommand,
};
Expand Down Expand Up @@ -223,7 +225,7 @@ pub fn print_gas_required_success(gas: Weight) {

/// Display contract information in a formatted way
pub fn basic_display_format_extended_contract_info(info: &ExtendedContractInfo) {
name_value_println!("TrieId", format!("{}", info.trie_id), MAX_KEY_COL_WIDTH);
name_value_println!("TrieId", info.trie_id, MAX_KEY_COL_WIDTH);
name_value_println!(
"Code Hash",
format!("{:?}", info.code_hash),
Expand Down
12 changes: 9 additions & 3 deletions crates/cargo-contract/src/cmd/remove.rs
Original file line number Diff line number Diff line change
Expand Up @@ -62,21 +62,27 @@ impl RemoveCommand {
.await?;
let remove_result = remove_exec.remove_code().await?;
let display_events = remove_result.display_events;
let output = if self.output_json() {
let output_events = if self.output_json() {
display_events.to_json()?
} else {
display_events.display_events(
self.extrinsic_cli_opts.verbosity().unwrap(),
remove_exec.token_metadata(),
)?
};
println!("{output}");
if let Some(code_removed) = remove_result.code_removed {
let remove_result = code_removed.code_hash;

if self.output_json() {
println!("{}", &remove_result);
// Create a JSON object with the events and the removed code hash.
let json_object = serde_json::json!({
"events": serde_json::from_str::<serde_json::Value>(&output_events)?,
"code_hash": remove_result,
});
let json_object = serde_json::to_string_pretty(&json_object)?;
println!("{}", json_object);
} else {
println!("{}", output_events);
name_value_println!("Code hash", format!("{remove_result:?}"));
}
Result::<(), ErrorVariant>::Ok(())
Expand Down
108 changes: 108 additions & 0 deletions crates/cargo-contract/src/cmd/storage.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
// Copyright 2018-2023 Parity Technologies (UK) Ltd.
// This file is part of cargo-contract.
//
// cargo-contract is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// cargo-contract is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with cargo-contract. If not, see <http://www.gnu.org/licenses/>.

use super::DefaultConfig;
use anyhow::Result;
use colored::Colorize;
use contract_extrinsics::{
ContractArtifacts,
ContractStorage,
ContractStorageRpc,
ErrorVariant,
};
use std::{
fmt::Debug,
path::PathBuf,
};
use subxt::Config;

#[derive(Debug, clap::Args)]
#[clap(name = "storage", about = "Inspect contract storage")]
pub struct StorageCommand {
/// The address of the contract to inspect storage of.
#[clap(name = "contract", long, env = "CONTRACT")]
contract: <DefaultConfig as Config>::AccountId,
/// Fetch the "raw" storage keys and values for the contract.
#[clap(long)]
raw: bool,
/// Path to a contract build artifact file: a raw `.wasm` file, a `.contract` bundle,
/// or a `.json` metadata file.
#[clap(value_parser, conflicts_with = "manifest_path")]
file: Option<PathBuf>,
/// Path to the `Cargo.toml` of the contract.
#[clap(long, value_parser)]
manifest_path: Option<PathBuf>,
/// Websockets url of a substrate node.
#[clap(
name = "url",
long,
value_parser,
default_value = "ws://localhost:9944"
)]
url: url::Url,
}

impl StorageCommand {
pub async fn run(&self) -> Result<(), ErrorVariant> {
let rpc = ContractStorageRpc::<DefaultConfig>::new(&self.url).await?;
let storage_layout = ContractStorage::<DefaultConfig>::new(rpc);

if self.raw {
let storage_data = storage_layout
.load_contract_storage_data(&self.contract)
.await?;
println!(
"{json}",
json = serde_json::to_string_pretty(&storage_data)?
);
return Ok(())
}

let contract_artifacts = ContractArtifacts::from_manifest_or_file(
self.manifest_path.as_ref(),
self.file.as_ref(),
);

match contract_artifacts {
Ok(contract_artifacts) => {
let ink_metadata = contract_artifacts.ink_project_metadata()?;
let contract_storage = storage_layout
.load_contract_storage_with_layout(&ink_metadata, &self.contract)
.await?;
println!(
"{json}",
json = serde_json::to_string_pretty(&contract_storage)?
);
}
Err(_) => {
eprintln!(
"{} Displaying raw storage: no valid contract metadata artifacts found",
"Info:".cyan().bold(),
);
let storage_data = storage_layout
.load_contract_storage_data(&self.contract)
.await?;
println!(
"{json}",
json = serde_json::to_string_pretty(&storage_data)?
);
return Ok(())
}
}

Ok(())
}
}
32 changes: 10 additions & 22 deletions crates/cargo-contract/src/cmd/upload.rs
Original file line number Diff line number Diff line change
Expand Up @@ -88,23 +88,26 @@ impl UploadCommand {
} else {
let upload_result = upload_exec.upload_code().await?;
let display_events = upload_result.display_events;
let output = if self.output_json() {
let output_events = if self.output_json() {
display_events.to_json()?
} else {
display_events.display_events(
self.extrinsic_cli_opts.verbosity()?,
upload_exec.token_metadata(),
)?
};
println!("{output}");
if let Some(code_stored) = upload_result.code_stored {
let upload_result = CodeHashResult {
code_hash: format!("{:?}", code_stored.code_hash),
};
let code_hash = code_stored.code_hash;
if self.output_json() {
println!("{}", upload_result.to_json()?);
// Create a JSON object with the events and the code hash.
let json_object = serde_json::json!({
"events": serde_json::from_str::<serde_json::Value>(&output_events)?,
"code_hash": code_hash,
});
println!("{}", serde_json::to_string_pretty(&json_object)?);
} else {
upload_result.print();
println!("{}", output_events);
name_value_println!("Code hash", format!("{:?}", code_hash));
}
} else {
let code_hash = hex::encode(code_hash);
Expand All @@ -118,21 +121,6 @@ impl UploadCommand {
}
}

#[derive(serde::Serialize)]
pub struct CodeHashResult {
pub code_hash: String,
}

impl CodeHashResult {
pub fn to_json(&self) -> Result<String> {
Ok(serde_json::to_string_pretty(self)?)
}

pub fn print(&self) {
name_value_println!("Code hash", format!("{:?}", self.code_hash));
}
}

#[derive(serde::Serialize)]
pub struct UploadDryRunResult {
pub result: String,
Expand Down
Loading

0 comments on commit 6f2d393

Please sign in to comment.