use super::{common::ReplyDetails, PayloadSizeError};
use crate::{
ids::{MessageId, ProgramId},
message::{
Dispatch, DispatchKind, GasLimit, Message, Packet, Payload, StoredDispatch, StoredMessage,
Value,
},
};
use gear_core_errors::{ErrorReplyReason, ReplyCode, SuccessReplyReason};
use scale_info::{
scale::{Decode, Encode},
TypeInfo,
};
#[derive(Clone, Default, Debug, Eq, Hash, Ord, PartialEq, PartialOrd, Decode, Encode, TypeInfo)]
pub struct ReplyMessage {
id: MessageId,
payload: Payload,
gas_limit: Option<GasLimit>,
value: Value,
code: ReplyCode,
}
impl ReplyMessage {
pub fn from_packet(id: MessageId, packet: ReplyPacket) -> Self {
Self {
id,
payload: packet.payload,
gas_limit: packet.gas_limit,
value: packet.value,
code: packet.code,
}
}
pub fn system(
origin_msg_id: MessageId,
payload: Payload,
err: impl Into<ErrorReplyReason>,
) -> Self {
let id = MessageId::generate_reply(origin_msg_id);
let packet = ReplyPacket::system(payload, err);
Self::from_packet(id, packet)
}
pub fn auto(origin_msg_id: MessageId) -> Self {
let id = MessageId::generate_reply(origin_msg_id);
let packet = ReplyPacket::auto();
Self::from_packet(id, packet)
}
pub fn into_message(
self,
program_id: ProgramId,
destination: ProgramId,
origin_msg_id: MessageId,
) -> Message {
Message::new(
self.id,
program_id,
destination,
self.payload,
self.gas_limit,
self.value,
Some(ReplyDetails::new(origin_msg_id, self.code).into()),
)
}
pub fn into_stored(
self,
program_id: ProgramId,
destination: ProgramId,
origin_msg_id: MessageId,
) -> StoredMessage {
self.into_message(program_id, destination, origin_msg_id)
.into()
}
pub fn into_dispatch(
self,
source: ProgramId,
destination: ProgramId,
origin_msg_id: MessageId,
) -> Dispatch {
Dispatch::new(
DispatchKind::Reply,
self.into_message(source, destination, origin_msg_id),
)
}
pub fn into_stored_dispatch(
self,
source: ProgramId,
destination: ProgramId,
origin_msg_id: MessageId,
) -> StoredDispatch {
self.into_dispatch(source, destination, origin_msg_id)
.into()
}
pub fn id(&self) -> MessageId {
self.id
}
pub fn payload_bytes(&self) -> &[u8] {
self.payload.inner()
}
pub fn gas_limit(&self) -> Option<GasLimit> {
self.gas_limit
}
pub fn value(&self) -> Value {
self.value
}
pub fn code(&self) -> ReplyCode {
self.code
}
}
#[derive(Clone, Default, Debug, Eq, Hash, Ord, PartialEq, PartialOrd, Decode, Encode, TypeInfo)]
pub struct ReplyPacket {
payload: Payload,
gas_limit: Option<GasLimit>,
value: Value,
code: ReplyCode,
}
impl ReplyPacket {
pub fn new(payload: Payload, value: Value) -> Self {
Self {
payload,
gas_limit: None,
value,
code: ReplyCode::Success(SuccessReplyReason::Manual),
}
}
pub fn new_with_gas(payload: Payload, gas_limit: GasLimit, value: Value) -> Self {
Self {
payload,
gas_limit: Some(gas_limit),
value,
code: ReplyCode::Success(SuccessReplyReason::Manual),
}
}
pub fn system(payload: Payload, err: impl Into<ErrorReplyReason>) -> Self {
Self {
payload,
code: ReplyCode::error(err),
..Default::default()
}
}
pub fn auto() -> Self {
Self {
gas_limit: Some(0),
code: ReplyCode::Success(SuccessReplyReason::Auto),
..Default::default()
}
}
pub(super) fn try_prepend(&mut self, data: Payload) -> Result<(), PayloadSizeError> {
self.payload.try_prepend(data)
}
pub fn code(&self) -> ReplyCode {
self.code
}
}
impl Packet for ReplyPacket {
fn payload_bytes(&self) -> &[u8] {
self.payload.inner()
}
fn gas_limit(&self) -> Option<GasLimit> {
self.gas_limit
}
fn value(&self) -> Value {
self.value
}
}