codegame 0.7.0

CodeGame framework
Documentation
use super::*;

struct Stream {
    reader: Box<dyn BufRead + Send>,
    writer: Box<dyn Write + Send>,
}

pub struct StreamPlayer<G: Game> {
    stream: Option<Stream>,
    phantom_data: PhantomData<G>,
}

impl<G: Game> StreamPlayer<G> {
    pub fn new(reader: Box<dyn BufRead + Send>, writer: Box<dyn Write + Send>) -> Self {
        Self {
            stream: Some(Stream { reader, writer }),
            phantom_data: PhantomData,
        }
    }
}

impl<G: Game> Drop for StreamPlayer<G> {
    fn drop(&mut self) {
        if let Some(stream) = &mut self.stream {
            let mut try_write_finish = || {
                ServerMessage::<G>::Finish {}.write_to(&mut stream.writer)?;
                stream.writer.flush()
            };
            if let Err(e) = try_write_finish() {
                warn!("{}", e);
            }
        }
    }
}

impl<G: Game> Player<G> for StreamPlayer<G> {
    fn get_action(
        &mut self,
        player_view: &G::PlayerView,
        debug_interface: Option<&PlayerDebugInterface<G>>,
    ) -> Result<G::Action, PlayerError> {
        let stream = self.stream.as_mut().expect("Called get_action after error");
        let mut get_action = move || {
            ServerMessage::<G>::GetAction {
                player_view: player_view.clone(), // TODO: dont clone
                debug_available: debug_interface.is_some(),
            }
            .write_to(&mut stream.writer)?;
            stream.writer.flush()?;
            loop {
                match ClientMessage::<G>::read_from(&mut stream.reader)? {
                    ClientMessage::ActionMessage { action } => return Ok(action),
                    ClientMessage::RequestDebugState {} => {
                        if let Some(debug_interface) = debug_interface {
                            debug_interface.state().write_to(&mut stream.writer)?;
                            stream.writer.flush()?;
                        } else {
                            return Err(PlayerError::IOError(std::io::Error::new(
                                std::io::ErrorKind::Other,
                                "Requested debug state with no debug interface available",
                            )));
                        }
                    }
                    ClientMessage::DebugMessage { command } => {
                        if let Some(debug_interface) = debug_interface {
                            debug_interface.send(command);
                        }
                    }
                    ClientMessage::DebugUpdateDone {} => {
                        return Err(PlayerError::IOError(std::io::Error::new(
                            std::io::ErrorKind::Other,
                            "Unexpected debug update done message in get_action",
                        )));
                    }
                }
            }
        };
        let result = get_action();
        if result.is_err() {
            self.stream = None;
        }
        result
    }
    fn debug_update(
        &mut self,
        player_view: &G::PlayerView,
        debug_interface: &PlayerDebugInterface<G>,
    ) -> Result<(), PlayerError> {
        let stream = self.stream.as_mut().expect("Called get_action after error");
        let mut debug_update = move || {
            ServerMessage::<G>::DebugUpdate {
                player_view: player_view.clone(), // TODO: do not clone
            }
            .write_to(&mut stream.writer)?;
            stream.writer.flush()?;
            loop {
                match ClientMessage::<G>::read_from(&mut stream.reader)? {
                    ClientMessage::ActionMessage { .. } => {
                        return Err(PlayerError::IOError(std::io::Error::new(
                            std::io::ErrorKind::Other,
                            "Unexpected action message in debug update",
                        )));
                    }
                    ClientMessage::RequestDebugState {} => {
                        debug_interface.state().write_to(&mut stream.writer)?;
                        stream.writer.flush()?;
                    }
                    ClientMessage::DebugMessage { command } => {
                        debug_interface.send(command);
                    }
                    ClientMessage::DebugUpdateDone {} => return Ok(()),
                }
            }
        };
        let result = debug_update();
        if result.is_err() {
            self.stream = None;
        }
        result
    }
}