2021-12-15 12:46:09 +03:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2021-12-21 11:37:35 +03:00
|
|
|
use air::CatchableError;
|
|
|
|
use air::LambdaError;
|
2021-12-15 12:46:09 +03:00
|
|
|
use air_test_utils::prelude::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn lambda_not_allowed_for_non_objects_and_arrays() {
|
|
|
|
let set_variable_peer_id = "set_variable";
|
|
|
|
let mut set_variable_vm = create_avm(echo_call_service(), set_variable_peer_id);
|
|
|
|
|
|
|
|
let local_peer_id = "local_peer_id";
|
|
|
|
|
2021-12-21 11:37:35 +03:00
|
|
|
let some_string = "some_string";
|
|
|
|
let script = f!(r#"
|
2021-12-15 12:46:09 +03:00
|
|
|
(seq
|
2021-12-21 11:37:35 +03:00
|
|
|
(call "{set_variable_peer_id}" ("" "") ["{some_string}"] string_variable)
|
|
|
|
(call "{local_peer_id}" ("" "") [string_variable.$.some_lambda])
|
2021-12-15 12:46:09 +03:00
|
|
|
)
|
2021-12-21 11:37:35 +03:00
|
|
|
"#);
|
2021-12-15 12:46:09 +03:00
|
|
|
|
2022-02-16 23:34:15 +03:00
|
|
|
let result = call_vm!(set_variable_vm, "asd", &script, "", "");
|
2021-12-15 12:46:09 +03:00
|
|
|
|
2021-12-21 11:37:35 +03:00
|
|
|
let expected_error = CatchableError::LambdaApplierError(LambdaError::FieldAccessorNotMatchValue {
|
|
|
|
value: json!(some_string),
|
|
|
|
field_name: "some_lambda".to_string(),
|
|
|
|
});
|
|
|
|
assert!(check_error(&result, expected_error));
|
2021-12-15 12:46:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn lambda_with_string_scalar() {
|
|
|
|
let set_variable_peer_id = "set_variable";
|
|
|
|
let variables = maplit::hashmap! {
|
|
|
|
"string_accessor".to_string() => json!("some_field_name"),
|
|
|
|
"value".to_string() => json!({"other_name_1": 0, "some_field_name": 1, "other_name_2": 0})
|
|
|
|
};
|
|
|
|
let mut set_variable_vm = create_avm(
|
|
|
|
set_variables_call_service(variables, VariableOptionSource::FunctionName),
|
|
|
|
set_variable_peer_id,
|
|
|
|
);
|
|
|
|
|
|
|
|
let local_peer_id = "local_peer_id";
|
|
|
|
let mut local_vm = create_avm(echo_call_service(), local_peer_id);
|
|
|
|
|
|
|
|
let script = f!(r#"
|
|
|
|
(seq
|
|
|
|
(seq
|
|
|
|
(call "{set_variable_peer_id}" ("" "string_accessor") [] string_accessor)
|
|
|
|
(call "{set_variable_peer_id}" ("" "value") [] value)
|
|
|
|
)
|
|
|
|
(call "{local_peer_id}" ("" "") [value.$.[string_accessor]])
|
|
|
|
)
|
|
|
|
"#);
|
|
|
|
|
|
|
|
let result = checked_call_vm!(set_variable_vm, "asd", &script, "", "");
|
|
|
|
let result = checked_call_vm!(local_vm, "asd", script, "", result.data);
|
|
|
|
let trace = trace_from_result(&result);
|
|
|
|
|
|
|
|
assert_eq!(&trace[2], &executed_state::scalar_number(1u32));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn lambda_with_number_scalar() {
|
|
|
|
let set_variable_peer_id = "set_variable";
|
|
|
|
let variables = maplit::hashmap! {
|
2022-01-26 17:16:59 +03:00
|
|
|
"number_accessor".to_string() => json!(1u32),
|
2021-12-15 12:46:09 +03:00
|
|
|
"value".to_string() => json!([0, 1, 2])
|
|
|
|
};
|
|
|
|
let mut set_variable_vm = create_avm(
|
|
|
|
set_variables_call_service(variables, VariableOptionSource::FunctionName),
|
|
|
|
set_variable_peer_id,
|
|
|
|
);
|
|
|
|
|
|
|
|
let local_peer_id = "local_peer_id";
|
|
|
|
let mut local_vm = create_avm(echo_call_service(), local_peer_id);
|
|
|
|
|
|
|
|
let script = f!(r#"
|
|
|
|
(seq
|
|
|
|
(seq
|
2022-01-26 17:16:59 +03:00
|
|
|
(call "{set_variable_peer_id}" ("" "number_accessor") [] number_accessor)
|
2021-12-15 12:46:09 +03:00
|
|
|
(call "{set_variable_peer_id}" ("" "value") [] value)
|
|
|
|
)
|
2022-01-26 17:16:59 +03:00
|
|
|
(call "{local_peer_id}" ("" "") [value.$.[number_accessor]])
|
2021-12-15 12:46:09 +03:00
|
|
|
)
|
|
|
|
"#);
|
|
|
|
|
|
|
|
let result = checked_call_vm!(set_variable_vm, "asd", &script, "", "");
|
|
|
|
let result = checked_call_vm!(local_vm, "asd", script, "", result.data);
|
|
|
|
let trace = trace_from_result(&result);
|
|
|
|
|
|
|
|
assert_eq!(&trace[2], &executed_state::scalar_number(1u32));
|
|
|
|
}
|
|
|
|
|
2022-01-26 17:16:59 +03:00
|
|
|
#[test]
|
|
|
|
fn lambda_with_number_stream() {
|
|
|
|
let set_variable_peer_id = "set_variable";
|
|
|
|
let variables = maplit::hashmap! {
|
|
|
|
"number_accessor".to_string() => json!(1),
|
|
|
|
"iterable".to_string() => json!([1,2,3]),
|
|
|
|
};
|
|
|
|
let mut set_variable_vm = create_avm(
|
|
|
|
set_variables_call_service(variables, VariableOptionSource::FunctionName),
|
|
|
|
set_variable_peer_id,
|
|
|
|
);
|
|
|
|
|
|
|
|
let local_peer_id = "local_peer_id";
|
|
|
|
let mut local_vm = create_avm(echo_call_service(), local_peer_id);
|
|
|
|
|
|
|
|
let script = f!(r#"
|
|
|
|
(seq
|
|
|
|
(seq
|
|
|
|
(call "{set_variable_peer_id}" ("" "number_accessor") [] number_accessor)
|
|
|
|
(seq
|
|
|
|
(call "{set_variable_peer_id}" ("" "iterable") [] iterable)
|
|
|
|
(fold iterable iterator
|
|
|
|
(seq
|
|
|
|
(call "{local_peer_id}" ("" "") [iterator] $stream)
|
|
|
|
(next iterator)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
(call "{local_peer_id}" ("" "") [$stream.$.[number_accessor]])
|
|
|
|
)
|
|
|
|
"#);
|
|
|
|
|
|
|
|
let result = checked_call_vm!(set_variable_vm, "asd", &script, "", "");
|
|
|
|
let result = checked_call_vm!(local_vm, "asd", script, "", result.data);
|
|
|
|
let actual_trace = trace_from_result(&result);
|
|
|
|
|
|
|
|
assert_eq!(&actual_trace[5], &executed_state::scalar_number(2));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn lambda_with_number_stream_and_followed_scalar() {
|
|
|
|
let set_variable_peer_id = "set_variable";
|
|
|
|
let checkable_value = 1337;
|
|
|
|
let variables = maplit::hashmap! {
|
|
|
|
"number_accessor".to_string() => json!(1),
|
|
|
|
"iterable".to_string() => json!([1,2,3]),
|
|
|
|
"value".to_string() => json!({"field_1": checkable_value, "field_2": 31337}),
|
|
|
|
};
|
|
|
|
let mut set_variable_vm = create_avm(
|
|
|
|
set_variables_call_service(variables, VariableOptionSource::FunctionName),
|
|
|
|
set_variable_peer_id,
|
|
|
|
);
|
|
|
|
|
|
|
|
let local_peer_id = "local_peer_id";
|
|
|
|
let mut local_vm = create_avm(echo_call_service(), local_peer_id);
|
|
|
|
|
|
|
|
let script = f!(r#"
|
|
|
|
(seq
|
|
|
|
(seq
|
|
|
|
(seq
|
|
|
|
(call "{set_variable_peer_id}" ("" "number_accessor") [] number_accessor)
|
|
|
|
(call "{set_variable_peer_id}" ("" "value") [] value)
|
|
|
|
)
|
|
|
|
(seq
|
|
|
|
(call "{set_variable_peer_id}" ("" "iterable") [] iterable)
|
|
|
|
(fold iterable iterator
|
|
|
|
(seq
|
|
|
|
(call "{local_peer_id}" ("" "") [value] $stream) ;; place 3 complex values in a stream
|
|
|
|
(next iterator)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
(call "{local_peer_id}" ("" "") [$stream.$.[number_accessor].field_1]) ;; get the 2nd value and then access its field
|
|
|
|
)
|
|
|
|
"#);
|
|
|
|
|
|
|
|
let result = checked_call_vm!(set_variable_vm, "asd", &script, "", "");
|
|
|
|
let result = checked_call_vm!(local_vm, "asd", script, "", result.data);
|
|
|
|
let actual_trace = trace_from_result(&result);
|
|
|
|
|
|
|
|
assert_eq!(&actual_trace[6], &executed_state::scalar_number(checkable_value));
|
|
|
|
}
|
|
|
|
|
2021-12-15 12:46:09 +03:00
|
|
|
#[test]
|
|
|
|
fn lambda_with_scalar_join() {
|
|
|
|
let set_variable_peer_id = "set_variable";
|
|
|
|
let variables = maplit::hashmap! {
|
|
|
|
"string_accessor".to_string() => json!("some_field_name"),
|
|
|
|
"value".to_string() => json!({"other_name_1": 0, "some_field_name": 1, "other_name_2": 0})
|
|
|
|
};
|
|
|
|
let mut set_variable_vm = create_avm(
|
|
|
|
set_variables_call_service(variables, VariableOptionSource::FunctionName),
|
|
|
|
set_variable_peer_id,
|
|
|
|
);
|
|
|
|
|
|
|
|
let local_peer_id = "local_peer_id";
|
|
|
|
let mut local_vm = create_avm(echo_call_service(), local_peer_id);
|
|
|
|
|
|
|
|
let script = f!(r#"
|
|
|
|
(seq
|
|
|
|
(par
|
|
|
|
(call "non_exist_peer_id" ("" "string_accessor") [] string_accessor)
|
|
|
|
(call "{set_variable_peer_id}" ("" "value") [] value)
|
|
|
|
)
|
|
|
|
(call "{local_peer_id}" ("" "") [value.$.[string_accessor]])
|
|
|
|
)
|
|
|
|
"#);
|
|
|
|
|
|
|
|
let result = checked_call_vm!(set_variable_vm, "asd", &script, "", "");
|
|
|
|
let result = checked_call_vm!(local_vm, "asd", script, "", result.data);
|
|
|
|
let trace = trace_from_result(&result);
|
|
|
|
|
|
|
|
assert_eq!(&trace[3], &executed_state::request_sent_by("set_variable"));
|
|
|
|
}
|
2022-01-26 17:16:59 +03:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn lambda_with_stream_join() {
|
|
|
|
let set_variable_peer_id = "set_variable";
|
|
|
|
let variables = maplit::hashmap! {
|
|
|
|
"number_accessor".to_string() => json!(1),
|
|
|
|
"iterable".to_string() => json!([1,2,3]),
|
|
|
|
};
|
|
|
|
let mut set_variable_vm = create_avm(
|
|
|
|
set_variables_call_service(variables, VariableOptionSource::FunctionName),
|
|
|
|
set_variable_peer_id,
|
|
|
|
);
|
|
|
|
|
|
|
|
let local_peer_id = "local_peer_id";
|
|
|
|
let mut local_vm = create_avm(echo_call_service(), local_peer_id);
|
|
|
|
|
|
|
|
let script = f!(r#"
|
|
|
|
(seq
|
|
|
|
(par
|
|
|
|
(call "non_exist_peer_id" ("" "number_accessor") [] number_accessor)
|
|
|
|
(seq
|
|
|
|
(call "{set_variable_peer_id}" ("" "iterable") [] iterable)
|
|
|
|
(fold iterable iterator
|
|
|
|
(seq
|
|
|
|
(ap "value" $stream)
|
|
|
|
(next iterator)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
(call "{local_peer_id}" ("" "") [$stream.$.[number_accessor]])
|
|
|
|
)
|
|
|
|
"#);
|
|
|
|
|
|
|
|
let result = checked_call_vm!(set_variable_vm, "asd", &script, "", "");
|
|
|
|
let result = checked_call_vm!(local_vm, "asd", script, "", result.data);
|
|
|
|
let actual_trace = trace_from_result(&result);
|
|
|
|
|
|
|
|
assert_eq!(&actual_trace[6], &executed_state::request_sent_by("set_variable"));
|
|
|
|
}
|