mirror of
https://github.com/fluencelabs/marine.git
synced 2024-12-12 14:55:32 +00:00
parent
16959d128a
commit
043e87623e
1
Cargo.lock
generated
1
Cargo.lock
generated
@ -712,6 +712,7 @@ version = "0.1.10"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"fce-wit-interfaces",
|
"fce-wit-interfaces",
|
||||||
|
"serde",
|
||||||
"walrus",
|
"walrus",
|
||||||
"wasmer-interface-types-fl",
|
"wasmer-interface-types-fl",
|
||||||
"wasmer-runtime-core-fl",
|
"wasmer-runtime-core-fl",
|
||||||
|
@ -17,3 +17,5 @@ anyhow = "1.0.31"
|
|||||||
walrus = "0.17.0"
|
walrus = "0.17.0"
|
||||||
wasmer-core = { package = "wasmer-runtime-core-fl", version = "0.17.0"}
|
wasmer-core = { package = "wasmer-runtime-core-fl", version = "0.17.0"}
|
||||||
wasmer-wit = { package = "wasmer-interface-types-fl", version = "=0.17.19" }
|
wasmer-wit = { package = "wasmer-interface-types-fl", version = "=0.17.19" }
|
||||||
|
|
||||||
|
serde = "1.0.117"
|
||||||
|
@ -32,6 +32,9 @@ pub enum WITParserError {
|
|||||||
/// An error occurred while parsing WIT section.
|
/// An error occurred while parsing WIT section.
|
||||||
CorruptedWITSection,
|
CorruptedWITSection,
|
||||||
|
|
||||||
|
// An error related to incorrect wit section.
|
||||||
|
IncorrectWIT(String),
|
||||||
|
|
||||||
/// An error occurred while parsing file in Wat format.
|
/// An error occurred while parsing file in Wat format.
|
||||||
CorruptedWATFile(WATError),
|
CorruptedWATFile(WATError),
|
||||||
|
|
||||||
@ -59,6 +62,7 @@ impl std::fmt::Display for WITParserError {
|
|||||||
f,
|
f,
|
||||||
"WIT section remainder isn't empty - WIT section possibly corrupted"
|
"WIT section remainder isn't empty - WIT section possibly corrupted"
|
||||||
),
|
),
|
||||||
|
WITParserError::IncorrectWIT(err_msg) => write!(f, "{}", err_msg),
|
||||||
WITParserError::CorruptedWITSection => write!(f, "WIT section is corrupted"),
|
WITParserError::CorruptedWITSection => write!(f, "WIT section is corrupted"),
|
||||||
WITParserError::CorruptedWATFile(err) => {
|
WITParserError::CorruptedWATFile(err) => {
|
||||||
write!(f, "an error occurred while parsing wat file: {}", err)
|
write!(f, "an error occurred while parsing wat file: {}", err)
|
||||||
|
@ -14,61 +14,21 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
use super::custom::WIT_SECTION_NAME;
|
mod functions;
|
||||||
use super::errors::WITParserError;
|
mod wit;
|
||||||
|
|
||||||
use walrus::{IdsToIndices, ModuleConfig};
|
pub use functions::*;
|
||||||
use wasmer_wit::ast::Interfaces;
|
pub use wit::*;
|
||||||
use wasmer_core::Module as WasmerModule;
|
|
||||||
|
|
||||||
|
use crate::Result;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
/// Extracts WIT section of provided Wasm binary and converts it to a string.
|
pub fn module_interface(module_path: PathBuf) -> Result<ServiceInterface> {
|
||||||
pub fn extract_text_wit(wasm_file_path: PathBuf) -> Result<String, WITParserError> {
|
use fce_wit_interfaces::FCEWITInterfaces;
|
||||||
let wit_section_bytes = extract_wit_section_bytes(wasm_file_path)?;
|
|
||||||
|
let wit_section_bytes = extract_wit_section_bytes(module_path)?;
|
||||||
let wit = extract_wit_with_fn(&wit_section_bytes)?;
|
let wit = extract_wit_with_fn(&wit_section_bytes)?;
|
||||||
Ok((&wit).to_string())
|
let fce_interface = FCEWITInterfaces::new(wit);
|
||||||
}
|
|
||||||
|
get_interface(&fce_interface)
|
||||||
/// Extracts WIT section of provided Wasm binary and converts it to a FCEWITInterfaces.
|
|
||||||
pub fn extract_wit(wasmer_module: &WasmerModule) -> Result<Interfaces<'_>, WITParserError> {
|
|
||||||
let wit_sections = wasmer_module
|
|
||||||
.custom_sections(WIT_SECTION_NAME)
|
|
||||||
.ok_or(WITParserError::NoWITSection)?;
|
|
||||||
|
|
||||||
if wit_sections.len() > 1 {
|
|
||||||
return Err(WITParserError::MultipleWITSections);
|
|
||||||
}
|
|
||||||
|
|
||||||
extract_wit_with_fn(&wit_sections[0])
|
|
||||||
}
|
|
||||||
|
|
||||||
fn extract_wit_with_fn(wit_section_bytes: &[u8]) -> Result<Interfaces<'_>, WITParserError> {
|
|
||||||
match wasmer_wit::decoders::binary::parse::<()>(&wit_section_bytes) {
|
|
||||||
Ok((remainder, wit)) if remainder.is_empty() => Ok(wit),
|
|
||||||
Ok(_) => Err(WITParserError::WITRemainderNotEmpty),
|
|
||||||
Err(_) => Err(WITParserError::CorruptedWITSection),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn extract_wit_section_bytes(wasm_file_path: PathBuf) -> Result<Vec<u8>, WITParserError> {
|
|
||||||
let module = ModuleConfig::new()
|
|
||||||
.parse_file(wasm_file_path)
|
|
||||||
.map_err(WITParserError::CorruptedWasmFile)?;
|
|
||||||
|
|
||||||
let sections = module
|
|
||||||
.customs
|
|
||||||
.iter()
|
|
||||||
.filter(|(_, section)| section.name() == WIT_SECTION_NAME)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
if sections.is_empty() {
|
|
||||||
return Err(WITParserError::NoWITSection);
|
|
||||||
}
|
|
||||||
if sections.len() > 1 {
|
|
||||||
return Err(WITParserError::MultipleWITSections);
|
|
||||||
}
|
|
||||||
|
|
||||||
let default_ids = IdsToIndices::default();
|
|
||||||
Ok(sections[0].1.data(&default_ids).into_owned())
|
|
||||||
}
|
}
|
||||||
|
211
crates/wit-parser/src/extractor/functions.rs
Normal file
211
crates/wit-parser/src/extractor/functions.rs
Normal file
@ -0,0 +1,211 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2020 Fluence Labs Limited
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
use crate::Result;
|
||||||
|
use crate::WITParserError;
|
||||||
|
use fce_wit_interfaces::FCEWITInterfaces;
|
||||||
|
|
||||||
|
use wasmer_wit::types::RecordType as IRecordType;
|
||||||
|
use wasmer_wit::ast::FunctionArg as IFunctionArg;
|
||||||
|
use wasmer_wit::types::InterfaceType as IType;
|
||||||
|
use serde::Serialize;
|
||||||
|
use serde::Deserialize;
|
||||||
|
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::rc::Rc;
|
||||||
|
|
||||||
|
pub type RecordTypes = HashMap<u64, Rc<IRecordType>>;
|
||||||
|
|
||||||
|
#[derive(PartialEq, Eq, Debug, Clone, Hash, Serialize, Deserialize)]
|
||||||
|
pub struct FCEFunctionSignature {
|
||||||
|
pub name: Rc<String>,
|
||||||
|
pub arguments: Rc<Vec<IFunctionArg>>,
|
||||||
|
pub outputs: Rc<Vec<IType>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Eq, Debug, Clone, Serialize)]
|
||||||
|
pub struct FCEModuleInterface {
|
||||||
|
pub record_types: RecordTypes,
|
||||||
|
pub function_signatures: Vec<FCEFunctionSignature>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_interface(wit: &FCEWITInterfaces<'_>) -> Result<ServiceInterface> {
|
||||||
|
let function_signatures = get_exports(wit)?;
|
||||||
|
let record_types = extract_record_types(wit);
|
||||||
|
|
||||||
|
let fce_interface = FCEModuleInterface {
|
||||||
|
record_types,
|
||||||
|
function_signatures,
|
||||||
|
};
|
||||||
|
|
||||||
|
let service_interface = into_service_interface(fce_interface);
|
||||||
|
|
||||||
|
Ok(service_interface)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_exports(wit: &FCEWITInterfaces<'_>) -> Result<Vec<FCEFunctionSignature>> {
|
||||||
|
use fce_wit_interfaces::WITAstType;
|
||||||
|
|
||||||
|
wit.implementations()
|
||||||
|
.filter_map(|(adapter_function_type, core_function_type)| {
|
||||||
|
match wit.exports_by_type(*core_function_type) {
|
||||||
|
Some(export_function_name) => Some((adapter_function_type, export_function_name)),
|
||||||
|
// pass functions that aren't export
|
||||||
|
None => None,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.map(|(adapter_function_type, export_function_names)| {
|
||||||
|
export_function_names
|
||||||
|
.iter()
|
||||||
|
.map(move |export_function_name| (*adapter_function_type, export_function_name))
|
||||||
|
})
|
||||||
|
.flatten()
|
||||||
|
.map(|(adapter_function_type, export_function_name)| {
|
||||||
|
let wit_type = wit.type_by_idx_r(adapter_function_type).unwrap();
|
||||||
|
|
||||||
|
match wit_type {
|
||||||
|
WITAstType::Function {
|
||||||
|
arguments,
|
||||||
|
output_types,
|
||||||
|
} => {
|
||||||
|
let signature = FCEFunctionSignature {
|
||||||
|
name: Rc::new(export_function_name.to_string()),
|
||||||
|
arguments: arguments.clone(),
|
||||||
|
outputs: output_types.clone(),
|
||||||
|
};
|
||||||
|
Ok(signature)
|
||||||
|
}
|
||||||
|
_ => Err(WITParserError::IncorrectWIT(format!(
|
||||||
|
"type with idx = {} isn't a function type",
|
||||||
|
adapter_function_type
|
||||||
|
))),
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect::<Result<Vec<FCEFunctionSignature>>>()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn extract_record_types(wit: &FCEWITInterfaces<'_>) -> RecordTypes {
|
||||||
|
use fce_wit_interfaces::WITAstType;
|
||||||
|
|
||||||
|
let (record_types_by_id, _) = wit.types().fold(
|
||||||
|
(HashMap::new(), 0u64),
|
||||||
|
|(mut record_types_by_id, id), ty| {
|
||||||
|
match ty {
|
||||||
|
WITAstType::Record(record_type) => {
|
||||||
|
record_types_by_id.insert(id, record_type.clone());
|
||||||
|
}
|
||||||
|
WITAstType::Function { .. } => {}
|
||||||
|
};
|
||||||
|
(record_types_by_id, id + 1)
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
record_types_by_id
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
pub struct FunctionSignature {
|
||||||
|
pub name: String,
|
||||||
|
pub arguments: Vec<(String, String)>,
|
||||||
|
pub output_types: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
pub struct RecordType {
|
||||||
|
pub name: String,
|
||||||
|
pub id: u64,
|
||||||
|
pub fields: Vec<(String, String)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
pub struct ServiceInterface {
|
||||||
|
pub function_signatures: Vec<FunctionSignature>,
|
||||||
|
pub record_types: Vec<RecordType>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn into_service_interface(fce_interface: FCEModuleInterface) -> ServiceInterface {
|
||||||
|
let record_types = fce_interface.record_types;
|
||||||
|
|
||||||
|
let function_signatures = fce_interface
|
||||||
|
.function_signatures
|
||||||
|
.into_iter()
|
||||||
|
.map(|sign| serialize_function_signature(sign, &record_types))
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let record_types = record_types
|
||||||
|
.iter()
|
||||||
|
.map(|(id, record)| serialize_record_type(*id, record.clone(), &record_types))
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
ServiceInterface {
|
||||||
|
record_types,
|
||||||
|
function_signatures,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn serialize_function_signature(
|
||||||
|
signature: FCEFunctionSignature,
|
||||||
|
record_types: &RecordTypes,
|
||||||
|
) -> FunctionSignature {
|
||||||
|
let arguments = signature
|
||||||
|
.arguments
|
||||||
|
.iter()
|
||||||
|
.map(|arg| (arg.name.clone(), itype_text_view(&arg.ty, record_types)))
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let output_types = signature
|
||||||
|
.outputs
|
||||||
|
.iter()
|
||||||
|
.map(|itype| itype_text_view(itype, record_types))
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
FunctionSignature {
|
||||||
|
name: signature.name.to_string(),
|
||||||
|
arguments,
|
||||||
|
output_types,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn serialize_record_type<'a, 'b>(
|
||||||
|
id: u64,
|
||||||
|
record: Rc<IRecordType>,
|
||||||
|
record_types: &RecordTypes,
|
||||||
|
) -> RecordType {
|
||||||
|
let fields = record
|
||||||
|
.fields
|
||||||
|
.iter()
|
||||||
|
.map(|field| (field.name.clone(), itype_text_view(&field.ty, record_types)))
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
RecordType {
|
||||||
|
name: record.name.clone(),
|
||||||
|
id,
|
||||||
|
fields,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn itype_text_view(arg_ty: &IType, record_types: &RecordTypes) -> String {
|
||||||
|
match arg_ty {
|
||||||
|
IType::Record(record_type_id) => {
|
||||||
|
// unwrap is safe because FaaSInterface here is well-formed
|
||||||
|
// (it was checked on the module startup stage)
|
||||||
|
let record = record_types.get(record_type_id).unwrap();
|
||||||
|
record.name.clone()
|
||||||
|
}
|
||||||
|
IType::Array(array_ty) => format!("Array<{}>", itype_text_view(array_ty, record_types)),
|
||||||
|
t => format!("{:?}", t),
|
||||||
|
}
|
||||||
|
}
|
78
crates/wit-parser/src/extractor/wit.rs
Normal file
78
crates/wit-parser/src/extractor/wit.rs
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2020 Fluence Labs Limited
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
use crate::custom::WIT_SECTION_NAME;
|
||||||
|
use crate::errors::WITParserError;
|
||||||
|
|
||||||
|
use walrus::{IdsToIndices, ModuleConfig};
|
||||||
|
use wasmer_wit::ast::Interfaces;
|
||||||
|
use wasmer_core::Module as WasmerModule;
|
||||||
|
|
||||||
|
use std::path::PathBuf;
|
||||||
|
|
||||||
|
/// Extracts WIT section of provided Wasm binary and converts it to a string.
|
||||||
|
pub fn extract_text_wit(wasm_file_path: PathBuf) -> Result<String, WITParserError> {
|
||||||
|
let wit_section_bytes = extract_wit_section_bytes(wasm_file_path)?;
|
||||||
|
let wit = extract_wit_with_fn(&wit_section_bytes)?;
|
||||||
|
Ok((&wit).to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Extracts WIT section of provided Wasm binary and converts it to a FCEWITInterfaces.
|
||||||
|
pub fn extract_wit(wasmer_module: &WasmerModule) -> Result<Interfaces<'_>, WITParserError> {
|
||||||
|
let wit_sections = wasmer_module
|
||||||
|
.custom_sections(WIT_SECTION_NAME)
|
||||||
|
.ok_or(WITParserError::NoWITSection)?;
|
||||||
|
|
||||||
|
if wit_sections.len() > 1 {
|
||||||
|
return Err(WITParserError::MultipleWITSections);
|
||||||
|
}
|
||||||
|
|
||||||
|
extract_wit_with_fn(&wit_sections[0])
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn extract_wit_with_fn(
|
||||||
|
wit_section_bytes: &[u8],
|
||||||
|
) -> Result<Interfaces<'_>, WITParserError> {
|
||||||
|
match wasmer_wit::decoders::binary::parse::<()>(&wit_section_bytes) {
|
||||||
|
Ok((remainder, wit)) if remainder.is_empty() => Ok(wit),
|
||||||
|
Ok(_) => Err(WITParserError::WITRemainderNotEmpty),
|
||||||
|
Err(_) => Err(WITParserError::CorruptedWITSection),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn extract_wit_section_bytes(
|
||||||
|
wasm_file_path: PathBuf,
|
||||||
|
) -> Result<Vec<u8>, WITParserError> {
|
||||||
|
let module = ModuleConfig::new()
|
||||||
|
.parse_file(wasm_file_path)
|
||||||
|
.map_err(WITParserError::CorruptedWasmFile)?;
|
||||||
|
|
||||||
|
let sections = module
|
||||||
|
.customs
|
||||||
|
.iter()
|
||||||
|
.filter(|(_, section)| section.name() == WIT_SECTION_NAME)
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
if sections.is_empty() {
|
||||||
|
return Err(WITParserError::NoWITSection);
|
||||||
|
}
|
||||||
|
if sections.len() > 1 {
|
||||||
|
return Err(WITParserError::MultipleWITSections);
|
||||||
|
}
|
||||||
|
|
||||||
|
let default_ids = IdsToIndices::default();
|
||||||
|
Ok(sections[0].1.data(&default_ids).into_owned())
|
||||||
|
}
|
@ -38,3 +38,6 @@ pub use embedder::embed_wit;
|
|||||||
pub use embedder::embed_text_wit;
|
pub use embedder::embed_text_wit;
|
||||||
pub use extractor::extract_wit;
|
pub use extractor::extract_wit;
|
||||||
pub use extractor::extract_text_wit;
|
pub use extractor::extract_text_wit;
|
||||||
|
pub use extractor::module_interface;
|
||||||
|
|
||||||
|
pub(crate) type Result<T> = std::result::Result<T, WITParserError>;
|
||||||
|
Loading…
Reference in New Issue
Block a user