forked from use-ink/cargo-contract
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcrate_metadata.rs
174 lines (157 loc) · 6.02 KB
/
crate_metadata.rs
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
// Copyright 2018-2021 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 crate::ManifestPath;
use anyhow::{Context, Result};
use cargo_metadata::{Metadata as CargoMetadata, MetadataCommand, Package};
use semver::Version;
use serde_json::{Map, Value};
use std::{fs, path::PathBuf};
use toml::value;
use url::Url;
/// Relevant metadata obtained from Cargo.toml.
#[derive(Debug)]
pub struct CrateMetadata {
pub manifest_path: ManifestPath,
pub cargo_meta: cargo_metadata::Metadata,
pub package_name: String,
pub root_package: Package,
pub original_wasm: PathBuf,
pub dest_wasm: PathBuf,
pub ink_version: Version,
pub documentation: Option<Url>,
pub homepage: Option<Url>,
pub user: Option<Map<String, Value>>,
pub target_directory: PathBuf,
}
impl CrateMetadata {
/// Parses the contract manifest and returns relevant metadata.
pub fn collect(manifest_path: &ManifestPath) -> Result<Self> {
let (metadata, root_package) = get_cargo_metadata(manifest_path)?;
let mut target_directory = metadata.target_directory.as_path().join("ink");
// Normalize the package name.
let package_name = root_package.name.replace("-", "_");
let absolute_manifest_path = manifest_path.absolute_directory()?;
let absolute_workspace_root = metadata.workspace_root.canonicalize()?;
if absolute_manifest_path != absolute_workspace_root {
target_directory = target_directory.join(package_name.clone());
}
// {target_dir}/wasm32-unknown-unknown/release/{package_name}.wasm
let mut original_wasm = target_directory.clone();
original_wasm.push("wasm32-unknown-unknown");
original_wasm.push("release");
original_wasm.push(package_name.clone());
original_wasm.set_extension("wasm");
// {target_dir}/{package_name}.wasm
let mut dest_wasm = target_directory.clone();
dest_wasm.push(package_name.clone());
dest_wasm.set_extension("wasm");
let ink_version = metadata
.packages
.iter()
.find_map(|package| {
if package.name == "ink_lang" {
Some(
Version::parse(&package.version.to_string())
.expect("Invalid ink_lang version string"),
)
} else {
None
}
})
.ok_or_else(|| anyhow::anyhow!("No 'ink_lang' dependency found"))?;
let ExtraMetadata {
documentation,
homepage,
user,
} = get_cargo_toml_metadata(manifest_path)?;
let crate_metadata = CrateMetadata {
manifest_path: manifest_path.clone(),
cargo_meta: metadata,
root_package,
package_name,
original_wasm: original_wasm.into(),
dest_wasm: dest_wasm.into(),
ink_version,
documentation,
homepage,
user,
target_directory: target_directory.into(),
};
Ok(crate_metadata)
}
}
/// Get the result of `cargo metadata`, together with the root package id.
fn get_cargo_metadata(manifest_path: &ManifestPath) -> Result<(CargoMetadata, Package)> {
let mut cmd = MetadataCommand::new();
let metadata = cmd
.manifest_path(manifest_path.as_ref())
.exec()
.context("Error invoking `cargo metadata`")?;
let root_package_id = metadata
.resolve
.as_ref()
.and_then(|resolve| resolve.root.as_ref())
.context("Cannot infer the root project id")?
.clone();
// Find the root package by id in the list of packages. It is logical error if the root
// package is not found in the list.
let root_package = metadata
.packages
.iter()
.find(|package| package.id == root_package_id)
.expect("The package is not found in the `cargo metadata` output")
.clone();
Ok((metadata, root_package))
}
/// Extra metadata not available via `cargo metadata`.
struct ExtraMetadata {
documentation: Option<Url>,
homepage: Option<Url>,
user: Option<Map<String, Value>>,
}
/// Read extra metadata not available via `cargo metadata` directly from `Cargo.toml`
fn get_cargo_toml_metadata(manifest_path: &ManifestPath) -> Result<ExtraMetadata> {
let toml = fs::read_to_string(manifest_path)?;
let toml: value::Table = toml::from_str(&toml)?;
let get_url = |field_name| -> Result<Option<Url>> {
toml.get("package")
.ok_or_else(|| anyhow::anyhow!("package section not found"))?
.get(field_name)
.and_then(|v| v.as_str())
.map(Url::parse)
.transpose()
.context(format!("{} should be a valid URL", field_name))
.map_err(Into::into)
};
let documentation = get_url("documentation")?;
let homepage = get_url("homepage")?;
let user = toml
.get("package")
.and_then(|v| v.get("metadata"))
.and_then(|v| v.get("contract"))
.and_then(|v| v.get("user"))
.and_then(|v| v.as_table())
.map(|v| {
// convert user defined section from toml to json
serde_json::to_string(v).and_then(|json| serde_json::from_str(&json))
})
.transpose()?;
Ok(ExtraMetadata {
documentation,
homepage,
user,
})
}