fdev 0.3.205

Freenet development tool
use std::{
    fmt::Display,
    fs::File,
    io::{BufReader, Read, Seek},
    sync::Arc,
    time::Duration,
};

use either::Either;
use freenet::dev_tool::{ClientEventsProxy, ClientId, OpenRequest};
use freenet_stdlib::{
    client_api::{
        ClientError, ClientRequest, ContractRequest, ContractResponse, ErrorKind, HostResponse,
    },
    prelude::*,
};
use futures::future::BoxFuture;
use serde::{Deserialize, Serialize, de::DeserializeOwned};

use crate::{CommandSender, util};

use super::{DeserializationFmt, ExecutorConfig, state::AppState};

const HELP: &str = "Freenet Contract Development Environment

SUBCOMMANDS:
    help        Print this message
    get         Gets the current value of the contract. It will be piped into the set output pipe (file, terminal, etc.)
    update      Attempts to update the contract and prints out the result of the operation
    put         Puts the state for the contract for the first time
    exit        Exit from the TUI";

type HostIncomingMsg = Result<OpenRequest<'static>, ClientError>;

pub(super) async fn user_fn_handler(
    config: ExecutorConfig,
    command_sender: CommandSender,
    app_state: AppState,
) -> anyhow::Result<()> {
    let mut input = StdInput::new(config, app_state)?;
    tracing::debug!("running... send a command or write \"help\" for help");
    loop {
        let command = input.recv().await.map_err(|err| anyhow::anyhow!(err));
        let command = command?;
        let dc = command.request.is_disconnect();
        command_sender
            .send(*command.request)
            .await
            .map_err(anyhow::Error::new)?;
        if dc {
            break;
        }
    }
    Ok(())
}

struct StdInput {
    config: ExecutorConfig,
    contract: ContractContainer,
    input: File,
    buf: Vec<u8>,
    app_state: AppState,
}

impl StdInput {
    fn new(config: ExecutorConfig, app_state: AppState) -> anyhow::Result<Self> {
        let paths = config.paths.clone().build(None)?;
        let params = config
            .params
            .as_ref()
            .map(|p| {
                let mut f = File::open(p)?;
                let mut buf = vec![];
                f.read_to_end(&mut buf)?;
                Ok::<_, anyhow::Error>(buf)
            })
            .transpose()?
            .unwrap_or_default();
        let (contract_code, _ver) =
            ContractCode::load_versioned_from_path(&paths.contracts_dir(config.mode))?;
        let contract = ContractContainer::Wasm(ContractWasmAPIVersion::V1(WrappedContract::new(
            Arc::new(contract_code),
            params.into(),
        )));
        Ok(StdInput {
            input: File::open(&config.input_file)?,
            config,
            contract,
            buf: vec![],
            app_state,
        })
    }

    fn read_input<T>(&mut self) -> anyhow::Result<T>
    where
        T: DeserializeOwned,
    {
        let mut buf = vec![];
        self.input.read_to_end(&mut buf).unwrap();
        util::deserialize(self.config.ser_format, &buf)
    }

    fn get_command_input<T>(&mut self, cmd: Command) -> Result<T, Box<ClientError>>
    where
        T: From<Vec<u8>>,
    {
        self.input.rewind().map_err(|e| {
            Box::new(ClientError::from(ErrorKind::Unhandled {
                cause: format!("{e}").into(),
            }))
        })?;
        match self.config.ser_format {
            Some(DeserializationFmt::Json) => {
                let state: serde_json::Value = self.read_input().map_err(|e| {
                    Box::new(ClientError::from(ErrorKind::Unhandled {
                        cause: format!("deserialization error: {e}").into(),
                    }))
                })?;
                let json_str = serde_json::to_string_pretty(&state).map_err(|e| {
                    Box::new(ClientError::from(ErrorKind::Unhandled {
                        cause: format!("{e}").into(),
                    }))
                })?;
                tracing::debug!("{cmd:?} value:\n{json_str}");
                Ok(json_str.into_bytes().into())
            }
            _ => {
                let state: Vec<u8> = self.read_input().map_err(|e| {
                    Box::new(ClientError::from(ErrorKind::Unhandled {
                        cause: format!("deserialization error: {e}").into(),
                    }))
                })?;
                Ok(state.into())
            }
        }
    }
}

#[derive(Serialize, Deserialize)]
/// Data to be read from the input file after commands are issued.
pub(super) enum CommandInput<'a> {
    Put {
        #[serde(borrow)]
        state: State<'a>,
    },
    Update {
        #[serde(borrow)]
        delta: StateDelta<'a>,
    },
}

impl<'a> CommandInput<'a> {
    fn unwrap_put(self) -> State<'a> {
        match self {
            Self::Put { state } => state,
            _ => panic!("expected put"),
        }
    }

    fn unwrap_delta(self) -> StateDelta<'a> {
        match self {
            Self::Update { delta } => delta,
            _ => panic!("expected put"),
        }
    }
}

impl Display for CommandInput<'_> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            Self::Put { .. } => {
                write!(f, "Put")
            }
            Self::Update { .. } => {
                write!(f, "Update")
            }
        }
    }
}

#[derive(Debug)]
enum Command {
    Put,
    Get,
    GetParams,
    Update,
    Help,
    Exit,
}

impl TryFrom<&[u8]> for Command {
    type Error = String;

    fn try_from(value: &[u8]) -> Result<Self, Self::Error> {
        let cmd = std::str::from_utf8(value).map_err(|e| format!("{e}"))?;
        match cmd {
            "put" => Ok(Command::Put),
            "get" => Ok(Command::Get),
            "get params" => Ok(Command::GetParams),
            "update" => Ok(Command::Update),
            "help" => Ok(Command::Help),
            "exit" => Ok(Command::Exit),
            v => Err(format!("unknown command: {v}")),
        }
    }
}

struct CommandInfo {
    cmd: Command,
    contract: ContractContainer,
    input: Option<CommandInput<'static>>,
}

impl From<CommandInfo> for OpenRequest<'static> {
    fn from(cmd: CommandInfo) -> Self {
        let key = cmd.contract.key();
        let req = match cmd.cmd {
            Command::Get => ContractRequest::Get {
                key: *key.id(),
                return_contract_code: false,
                subscribe: false,
                blocking_subscribe: false,
            }
            .into(),
            Command::Put => {
                let state = cmd.input.unwrap().unwrap_put();
                ContractRequest::Put {
                    contract: cmd.contract,
                    state: WrappedState::new(state.into_bytes()),
                    related_contracts: Default::default(),
                    subscribe: false,
                    blocking_subscribe: false,
                }
                .into()
            }
            Command::Update => {
                let data = cmd.input.unwrap().unwrap_delta().into();
                ContractRequest::Update { key, data }.into()
            }
            Command::Exit => ClientRequest::Disconnect {
                cause: Some("shutdown".to_owned().into()),
            },
            _ => unreachable!(),
        };
        OpenRequest::new(ClientId::FIRST, Box::new(req))
    }
}

impl ClientEventsProxy for StdInput {
    fn recv(&mut self) -> BoxFuture<'_, HostIncomingMsg> {
        Box::pin(async {
            loop {
                self.buf.clear();
                let f = async {
                    let stdin = BufReader::new(std::io::stdin());
                    'stdin: for b in stdin.bytes() {
                        let b = b.map_err(|_| {
                            ClientError::from(ErrorKind::TransportProtocolDisconnect)
                        })?;
                        if b == b'\n' {
                            break 'stdin;
                        }
                        self.buf.push(b);
                    }
                    if String::from_utf8_lossy(&self.buf[..]).trim().is_empty() {
                        return Ok(Either::Right(()));
                    }
                    // try parse command
                    match Command::try_from(&self.buf[..]) {
                        Ok(Command::Help) => {
                            tracing::debug!("{HELP}");
                        }
                        Ok(cmd @ Command::Put) => {
                            let state: State = match self.get_command_input(Command::Put) {
                                Ok(v) => v,
                                Err(e) => {
                                    tracing::debug!("Put event error: {}", e);
                                    return Ok(Either::Right(()));
                                }
                            };
                            return Ok(Either::Left(
                                CommandInfo {
                                    cmd,
                                    contract: self.contract.clone(),
                                    input: Some(CommandInput::Put { state }),
                                }
                                .into(),
                            ));
                        }
                        Ok(cmd @ Command::Update) => {
                            let delta: StateDelta = match self.get_command_input(Command::Update) {
                                Ok(v) => v,
                                Err(e) => {
                                    tracing::debug!("Update event error: {e}");
                                    return Ok(Either::Right(()));
                                }
                            };
                            return Ok(Either::Left(
                                CommandInfo {
                                    cmd,
                                    contract: self.contract.clone(),
                                    input: Some(CommandInput::Update { delta }),
                                }
                                .into(),
                            ));
                        }
                        Ok(Command::GetParams) => {
                            let node = &mut *self.app_state.local_node.lock().await;
                            let key = self.contract.key();
                            node.send(ClientRequest::ContractOp(ContractRequest::Get {
                                key: *key.id(),
                                return_contract_code: true,
                                subscribe: false,
                                blocking_subscribe: false,
                            }))
                            .await
                            .map_err(|e| {
                                ClientError::from(ErrorKind::Unhandled {
                                    cause: format!("{e}").into(),
                                })
                            })?;
                            let resp = node.recv().await.map_err(|e| {
                                ClientError::from(ErrorKind::Unhandled {
                                    cause: format!("{e}").into(),
                                })
                            })?;

                            if let HostResponse::ContractResponse(ContractResponse::GetResponse {
                                contract,
                                ..
                            }) = resp
                            {
                                if let Some(contract) = contract {
                                    if let Err(e) =
                                        self.app_state.printout_deser(&contract.params())
                                    {
                                        tracing::error!("error printing params: {e}");
                                    }
                                } else {
                                    return Err(ClientError::from(ErrorKind::Unhandled {
                                        cause: "missing contract container".into(),
                                    }));
                                }
                            }
                        }
                        Ok(cmd) => {
                            return Ok(Either::Left(
                                CommandInfo {
                                    cmd,
                                    contract: self.contract.clone(),
                                    input: None,
                                }
                                .into(),
                            ));
                        }
                        Err(err) => {
                            tracing::debug!("{err}");
                        }
                    }
                    tokio::time::sleep(Duration::from_millis(10)).await;
                    Ok(Either::<OpenRequest, _>::Right(()))
                };
                match f.await {
                    Ok(Either::Right(_)) => continue,
                    Ok(Either::Left(r)) => return Ok(r),
                    Err(err) => return Err(err),
                }
            }
        })
    }

    fn send(
        &mut self,
        _id: ClientId,
        _response: Result<HostResponse, ClientError>,
    ) -> BoxFuture<'_, Result<(), ClientError>> {
        unimplemented!()
    }
}

impl Clone for StdInput {
    fn clone(&self) -> Self {
        Self {
            config: self.config.clone(),
            contract: self.contract.clone(),
            buf: Vec::new(),
            input: File::open(&self.config.input_file).unwrap(),
            app_state: self.app_state.clone(),
        }
    }
}