use super::*;
use crate::service_protocol::messages::*;
use crate::service_protocol::CANCEL_SIGNAL_ID;
use crate::Value;
use googletest::prelude::*;
use test_log::test;
#[test]
fn call_then_get_invocation_id_then_cancel() {
let mut output = VMTestCase::new()
.input(start_message(1))
.input(input_entry_message(b"my-data"))
.input(CallInvocationIdCompletionNotificationMessage {
completion_id: 1,
invocation_id: "my-id".to_string(),
})
.input(cancel_signal_notification())
.run(|vm| {
vm.sys_input().unwrap();
let call_handle = vm
.sys_call(
Target {
service: "MySvc".to_string(),
handler: "MyHandler".to_string(),
key: None,
idempotency_key: None,
headers: Vec::new(),
},
Bytes::new(),
None,
PayloadOptions::default(),
)
.unwrap();
assert_eq!(
vm.do_progress(vec![call_handle.invocation_id_notification_handle])
.unwrap(),
DoProgressResponse::AnyCompleted
);
assert2::assert!(
let Some(Value::InvocationId(invocation_id)) = vm
.take_notification(call_handle.invocation_id_notification_handle)
.unwrap()
);
assert_eq!(invocation_id, "my-id");
assert_eq!(
vm.do_progress(vec![call_handle.call_notification_handle])
.unwrap(),
DoProgressResponse::CancelSignalReceived
);
vm.sys_end().unwrap();
});
assert_that!(
output.next_decoded::<CallCommandMessage>().unwrap(),
pat!(CallCommandMessage {
service_name: eq("MySvc"),
handler_name: eq("MyHandler"),
invocation_id_notification_idx: eq(1),
})
);
assert_eq!(
output.next_decoded::<SendSignalCommandMessage>().unwrap(),
SendSignalCommandMessage {
target_invocation_id: "my-id".to_string(),
signal_id: Some(send_signal_command_message::SignalId::Idx(CANCEL_SIGNAL_ID)),
result: Some(send_signal_command_message::Result::Void(Default::default())),
..Default::default()
}
);
assert_eq!(
output.next_decoded::<EndMessage>().unwrap(),
EndMessage::default()
);
assert_eq!(output.next(), None);
}
#[test]
fn call_then_cancel() {
let mut output = VMTestCase::new()
.input(start_message(1))
.input(input_entry_message(b"my-data"))
.input(CallInvocationIdCompletionNotificationMessage {
completion_id: 1,
invocation_id: "my-id".to_string(),
})
.input(cancel_signal_notification())
.run(|vm| {
vm.sys_input().unwrap();
let call_handle = vm
.sys_call(
Target {
service: "MySvc".to_string(),
handler: "MyHandler".to_string(),
key: None,
idempotency_key: None,
headers: Vec::new(),
},
Bytes::new(),
None,
PayloadOptions::default(),
)
.unwrap();
assert_eq!(
vm.do_progress(vec![call_handle.call_notification_handle])
.unwrap(),
DoProgressResponse::CancelSignalReceived
);
vm.sys_end().unwrap();
});
assert_that!(
output.next_decoded::<CallCommandMessage>().unwrap(),
pat!(CallCommandMessage {
service_name: eq("MySvc"),
handler_name: eq("MyHandler"),
invocation_id_notification_idx: eq(1),
})
);
assert_eq!(
output.next_decoded::<SendSignalCommandMessage>().unwrap(),
SendSignalCommandMessage {
target_invocation_id: "my-id".to_string(),
signal_id: Some(send_signal_command_message::SignalId::Idx(CANCEL_SIGNAL_ID)),
result: Some(send_signal_command_message::Result::Void(Default::default())),
..Default::default()
}
);
assert_eq!(
output.next_decoded::<EndMessage>().unwrap(),
EndMessage::default()
);
assert_eq!(output.next(), None);
}
#[test]
fn call_then_cancel_without_invocation_id() {
let mut output = VMTestCase::new()
.input(start_message(1))
.input(input_entry_message(b"my-data"))
.input(cancel_signal_notification())
.run(|vm| {
vm.sys_input().unwrap();
let call_handle = vm
.sys_call(
Target {
service: "MySvc".to_string(),
handler: "MyHandler".to_string(),
key: None,
idempotency_key: None,
headers: Vec::new(),
},
Bytes::new(),
None,
PayloadOptions::default(),
)
.unwrap();
assert_that!(
vm.do_progress(vec![call_handle.call_notification_handle]),
err(is_suspended())
);
});
assert_that!(
output.next_decoded::<CallCommandMessage>().unwrap(),
pat!(CallCommandMessage {
service_name: eq("MySvc"),
handler_name: eq("MyHandler"),
invocation_id_notification_idx: eq(1),
})
);
assert_that!(
output.next_decoded::<SuspensionMessage>().unwrap(),
pat!(SuspensionMessage {
waiting_completions: eq(vec![1]),
waiting_signals: empty()
})
);
assert_eq!(output.next(), None);
}
#[test]
fn call_then_then_cancel_disabling_children_cancellation() {
let mut output = VMTestCase::with_vm_options(VMOptions {
implicit_cancellation: ImplicitCancellationOption::Enabled {
cancel_children_calls: false,
cancel_children_one_way_calls: true,
},
..VMOptions::default()
})
.input(start_message(1))
.input(input_entry_message(b"my-data"))
.input(cancel_signal_notification())
.run(|vm| {
vm.sys_input().unwrap();
let call_handle = vm
.sys_call(
Target {
service: "MySvc".to_string(),
handler: "MyHandler".to_string(),
key: None,
idempotency_key: None,
headers: Vec::new(),
},
Bytes::new(),
None,
PayloadOptions::default(),
)
.unwrap();
assert_eq!(
vm.do_progress(vec![call_handle.call_notification_handle])
.unwrap(),
DoProgressResponse::CancelSignalReceived
);
vm.sys_end().unwrap();
});
assert_that!(
output.next_decoded::<CallCommandMessage>().unwrap(),
pat!(CallCommandMessage {
service_name: eq("MySvc"),
handler_name: eq("MyHandler"),
invocation_id_notification_idx: eq(1),
})
);
assert_eq!(
output.next_decoded::<EndMessage>().unwrap(),
EndMessage::default()
);
assert_eq!(output.next(), None);
}
#[test]
fn disabled_implicit_cancellation() {
let mut output = VMTestCase::with_vm_options(VMOptions {
implicit_cancellation: ImplicitCancellationOption::Disabled,
..VMOptions::default()
})
.input(start_message(1))
.input(input_entry_message(b"my-data"))
.input(cancel_signal_notification())
.run(|vm| {
vm.sys_input().unwrap();
let call_handle = vm
.sys_call(
Target {
service: "MySvc".to_string(),
handler: "MyHandler".to_string(),
key: None,
idempotency_key: None,
headers: Vec::new(),
},
Bytes::new(),
None,
PayloadOptions::default(),
)
.unwrap();
assert_that!(
vm.do_progress(vec![call_handle.call_notification_handle]),
err(is_suspended())
);
});
assert_that!(
output.next_decoded::<CallCommandMessage>().unwrap(),
pat!(CallCommandMessage {
service_name: eq("MySvc"),
handler_name: eq("MyHandler"),
invocation_id_notification_idx: eq(1),
result_completion_id: eq(2),
})
);
assert_that!(
output.next_decoded::<SuspensionMessage>().unwrap(),
pat!(SuspensionMessage {
waiting_completions: eq(vec![2]),
waiting_signals: empty()
})
);
assert_eq!(output.next(), None);
}
#[test]
fn replay_while_cancelling() {
let mut output = VMTestCase::new()
.input(start_message(7))
.input(input_entry_message(b"my-data"))
.input(CallCommandMessage {
service_name: "MySvc".to_string(),
handler_name: "MyHandler".to_string(),
invocation_id_notification_idx: 1,
result_completion_id: 2,
..CallCommandMessage::default()
})
.input(CallInvocationIdCompletionNotificationMessage {
completion_id: 1,
invocation_id: "my-id-1".to_string(),
})
.input(CallCommandMessage {
service_name: "MySvc".to_string(),
handler_name: "MyHandler".to_string(),
invocation_id_notification_idx: 3,
result_completion_id: 4,
..CallCommandMessage::default()
})
.input(CallInvocationIdCompletionNotificationMessage {
completion_id: 3,
invocation_id: "my-id-2".to_string(),
})
.input(cancel_signal_notification())
.input(SendSignalCommandMessage {
target_invocation_id: "my-id-1".to_string(),
signal_id: Some(send_signal_command_message::SignalId::Idx(CANCEL_SIGNAL_ID)),
result: Some(send_signal_command_message::Result::Void(Default::default())),
..Default::default()
})
.run(|vm| {
vm.sys_input().unwrap();
let call_handle_1 = vm
.sys_call(
Target {
service: "MySvc".to_string(),
handler: "MyHandler".to_string(),
key: None,
idempotency_key: None,
headers: Vec::new(),
},
Bytes::new(),
None,
PayloadOptions::default(),
)
.unwrap();
let call_handle_2 = vm
.sys_call(
Target {
service: "MySvc".to_string(),
handler: "MyHandler".to_string(),
key: None,
idempotency_key: None,
headers: Vec::new(),
},
Bytes::new(),
None,
PayloadOptions::default(),
)
.unwrap();
assert_eq!(
vm.do_progress(vec![
call_handle_1.call_notification_handle,
call_handle_2.call_notification_handle
])
.unwrap(),
DoProgressResponse::CancelSignalReceived
);
vm.sys_end().unwrap();
});
assert_eq!(
output.next_decoded::<SendSignalCommandMessage>().unwrap(),
SendSignalCommandMessage {
target_invocation_id: "my-id-2".to_string(),
signal_id: Some(send_signal_command_message::SignalId::Idx(CANCEL_SIGNAL_ID)),
result: Some(send_signal_command_message::Result::Void(Default::default())),
..Default::default()
}
);
assert_eq!(
output.next_decoded::<EndMessage>().unwrap(),
EndMessage::default()
);
assert_eq!(output.next(), None);
}