2021-08-24 16:14:15 +03:00
|
|
|
/*
|
|
|
|
* Copyright 2021 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 super::ExecutedState;
|
2021-11-24 17:57:14 +03:00
|
|
|
use super::GlobalStreamGens;
|
|
|
|
use super::RestrictedStreamGens;
|
2021-08-24 16:14:15 +03:00
|
|
|
use super::DATA_FORMAT_VERSION;
|
|
|
|
use serde::Deserialize;
|
|
|
|
use serde::Serialize;
|
|
|
|
use std::ops::Deref;
|
|
|
|
|
|
|
|
pub type ExecutionTrace = Vec<ExecutedState>;
|
|
|
|
|
|
|
|
/// The AIR interpreter could be considered as a function
|
|
|
|
/// f(prev_data: InterpreterData, current_data: InterpreterData, ... ) -> (result_data: InterpreterData, ...).
|
|
|
|
/// This function receives prev and current data and produces a result data. All these data
|
|
|
|
/// have the following format.
|
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
|
|
pub struct InterpreterData {
|
2022-04-15 22:25:03 +03:00
|
|
|
/// Trace of AIR execution, which contains executed call, par, fold, and ap states.
|
2021-08-24 16:14:15 +03:00
|
|
|
pub trace: ExecutionTrace,
|
|
|
|
|
2021-11-24 17:57:14 +03:00
|
|
|
/// Contains maximum generation for each global stream. This info will be used while merging
|
|
|
|
/// values in streams. This field is also needed for backward compatibility with
|
|
|
|
/// <= 0.2.1 versions.
|
|
|
|
#[serde(rename = "streams")] // for compatibility with versions <= 0.2.1
|
|
|
|
pub global_streams: GlobalStreamGens,
|
2021-08-24 16:14:15 +03:00
|
|
|
|
|
|
|
/// Version of this data format.
|
|
|
|
pub version: semver::Version,
|
2021-10-04 10:58:00 +03:00
|
|
|
|
|
|
|
/// Last exposed to a peer call request id. All next call request ids will be bigger than this.
|
|
|
|
#[serde(default)]
|
|
|
|
#[serde(rename = "lcid")]
|
|
|
|
pub last_call_request_id: u32,
|
2021-11-24 17:57:14 +03:00
|
|
|
|
|
|
|
/// Contains maximum generation for each private stream. This info will be used while merging
|
|
|
|
/// values in streams.
|
|
|
|
#[serde(default)]
|
|
|
|
#[serde(rename = "r_streams")]
|
|
|
|
pub restricted_streams: RestrictedStreamGens,
|
2021-08-24 16:14:15 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl InterpreterData {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Self {
|
|
|
|
trace: <_>::default(),
|
2021-11-24 17:57:14 +03:00
|
|
|
global_streams: <_>::default(),
|
2021-08-24 16:14:15 +03:00
|
|
|
version: DATA_FORMAT_VERSION.deref().clone(),
|
2021-10-04 10:58:00 +03:00
|
|
|
last_call_request_id: 0,
|
2021-11-24 17:57:14 +03:00
|
|
|
restricted_streams: <_>::default(),
|
2021-08-24 16:14:15 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-04 10:58:00 +03:00
|
|
|
pub fn from_execution_result(
|
|
|
|
trace: ExecutionTrace,
|
2021-11-24 17:57:14 +03:00
|
|
|
streams: GlobalStreamGens,
|
|
|
|
restricted_streams: RestrictedStreamGens,
|
2021-10-04 10:58:00 +03:00
|
|
|
last_call_request_id: u32,
|
|
|
|
) -> Self {
|
2021-08-24 16:14:15 +03:00
|
|
|
Self {
|
|
|
|
trace,
|
2021-11-24 17:57:14 +03:00
|
|
|
global_streams: streams,
|
2021-08-24 16:14:15 +03:00
|
|
|
version: DATA_FORMAT_VERSION.deref().clone(),
|
2021-10-04 10:58:00 +03:00
|
|
|
last_call_request_id,
|
2021-11-24 17:57:14 +03:00
|
|
|
restricted_streams,
|
2021-08-24 16:14:15 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Tries to de InterpreterData from slice according to the data version.
|
|
|
|
pub fn try_from_slice(slice: &[u8]) -> Result<Self, serde_json::Error> {
|
|
|
|
// treat empty slice as an empty interpreter data allows abstracting from
|
|
|
|
// the internal format for empty data.
|
|
|
|
if slice.is_empty() {
|
|
|
|
return Ok(Self::default());
|
|
|
|
}
|
|
|
|
|
|
|
|
serde_json::from_slice(slice)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for InterpreterData {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::new()
|
|
|
|
}
|
|
|
|
}
|
2021-10-04 10:58:00 +03:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
use serde::Deserialize;
|
|
|
|
use serde::Serialize;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn compatible_with_0_2_0_version() {
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
struct InterpreterData0_2_0 {
|
|
|
|
pub trace: ExecutionTrace,
|
2021-11-24 17:57:14 +03:00
|
|
|
pub streams: GlobalStreamGens,
|
2021-10-04 10:58:00 +03:00
|
|
|
pub version: semver::Version,
|
|
|
|
}
|
|
|
|
|
2021-11-24 17:57:14 +03:00
|
|
|
// test 0.2.0 to 0.2.2 conversion
|
2021-10-04 10:58:00 +03:00
|
|
|
let data_0_2_0 = InterpreterData0_2_0 {
|
|
|
|
trace: ExecutionTrace::default(),
|
2021-11-24 17:57:14 +03:00
|
|
|
streams: GlobalStreamGens::default(),
|
2021-10-04 10:58:00 +03:00
|
|
|
version: semver::Version::new(0, 2, 0),
|
|
|
|
};
|
|
|
|
|
|
|
|
let data_0_2_0_se = serde_json::to_vec(&data_0_2_0).unwrap();
|
|
|
|
let data_0_2_1 = serde_json::from_slice::<InterpreterData>(&data_0_2_0_se);
|
|
|
|
assert!(data_0_2_1.is_ok());
|
|
|
|
|
2021-11-24 17:57:14 +03:00
|
|
|
// test 0.2.2 to 0.2.0 conversion
|
|
|
|
let data_0_2_2 = InterpreterData::default();
|
|
|
|
let data_0_2_2_se = serde_json::to_vec(&data_0_2_2).unwrap();
|
|
|
|
let data_0_2_0 = serde_json::from_slice::<InterpreterData0_2_0>(&data_0_2_2_se);
|
|
|
|
assert!(data_0_2_0.is_ok());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn compatible_with_0_2_1_version() {
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
struct InterpreterData0_2_1 {
|
|
|
|
pub trace: ExecutionTrace,
|
|
|
|
pub streams: GlobalStreamGens,
|
|
|
|
pub version: semver::Version,
|
|
|
|
#[serde(default)]
|
|
|
|
#[serde(rename = "lcid")]
|
|
|
|
pub last_call_request_id: u32,
|
|
|
|
}
|
|
|
|
|
|
|
|
// test 0.2.1 to 0.2.2 conversion
|
|
|
|
let data_0_2_1 = InterpreterData0_2_1 {
|
|
|
|
trace: ExecutionTrace::default(),
|
|
|
|
streams: GlobalStreamGens::default(),
|
|
|
|
version: semver::Version::new(0, 2, 1),
|
|
|
|
last_call_request_id: 1,
|
|
|
|
};
|
|
|
|
|
2021-10-04 10:58:00 +03:00
|
|
|
let data_0_2_1_se = serde_json::to_vec(&data_0_2_1).unwrap();
|
2021-11-24 17:57:14 +03:00
|
|
|
let data_0_2_2 = serde_json::from_slice::<InterpreterData>(&data_0_2_1_se);
|
|
|
|
assert!(data_0_2_2.is_ok());
|
|
|
|
|
|
|
|
// test 0.2.2 to 0.2.1 conversion
|
|
|
|
let data_0_2_2 = InterpreterData::default();
|
|
|
|
let data_0_2_2_se = serde_json::to_vec(&data_0_2_2).unwrap();
|
|
|
|
let data_0_2_0 = serde_json::from_slice::<InterpreterData0_2_1>(&data_0_2_2_se);
|
2021-10-04 10:58:00 +03:00
|
|
|
assert!(data_0_2_0.is_ok());
|
|
|
|
}
|
|
|
|
}
|