serde
This commit is contained in:
+32
-51
@@ -653,66 +653,46 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg_attr(
|
||||||
|
feature = "serde",
|
||||||
|
derive(serde_derive::Deserialize),
|
||||||
|
serde(bound = "
|
||||||
|
G: serde::Deserialize<'de>,
|
||||||
|
SessionConfig<G::Config>: serde::Deserialize<'de>,
|
||||||
|
Vec<G::Instruction>: serde::Deserialize<'de>,
|
||||||
|
")
|
||||||
|
)]
|
||||||
|
struct SerializedSession<G: Game> {
|
||||||
|
config: SessionConfig<G::Config>,
|
||||||
|
state: G,
|
||||||
|
history: Vec<G::Instruction>,
|
||||||
|
}
|
||||||
#[cfg(feature = "serde")]
|
#[cfg(feature = "serde")]
|
||||||
impl<'de, G: Game> serde::de::DeserializeSeed<'de> for Session<G>
|
impl<'de, G: Game<Score = i32>> serde::de::Deserialize<'de> for Session<G>
|
||||||
where
|
where
|
||||||
G: serde::Deserialize<'de> + Clone,
|
G: serde::Deserialize<'de> + Clone + Eq + core::hash::Hash,
|
||||||
G::Instruction: serde::Deserialize<'de>,
|
G::Stats: Default,
|
||||||
|
G::Instruction: serde::Deserialize<'de> + Eq + core::hash::Hash,
|
||||||
|
SessionConfig<G::Config>: serde::Deserialize<'de>,
|
||||||
|
Vec<G::Instruction>: serde::Deserialize<'de>,
|
||||||
{
|
{
|
||||||
type Value = SessionState<G>;
|
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
||||||
fn deserialize<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
|
|
||||||
where
|
where
|
||||||
D: serde::Deserializer<'de>,
|
D: serde::Deserializer<'de>,
|
||||||
{
|
{
|
||||||
struct SessionStateVisitor<G: Game> {
|
let serialized = SerializedSession::deserialize(deserializer)?;
|
||||||
state: G,
|
let mut session = Session::new(serialized.state, serialized.config);
|
||||||
stats: SessionStats<G::Stats>,
|
for instruction in serialized.history {
|
||||||
config: SessionConfig<G::Config>,
|
session.process_instruction(instruction);
|
||||||
}
|
}
|
||||||
impl<'de, G: Game> serde::de::Visitor<'de> for SessionStateVisitor<G>
|
Ok(session)
|
||||||
where
|
|
||||||
G: serde::Deserialize<'de> + Clone,
|
|
||||||
G::Instruction: serde::Deserialize<'de>,
|
|
||||||
{
|
|
||||||
type Value = SessionState<G>;
|
|
||||||
fn expecting(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
|
|
||||||
write!(f, "State History")
|
|
||||||
}
|
|
||||||
fn visit_seq<A>(mut self, mut seq: A) -> Result<Self::Value, A::Error>
|
|
||||||
where
|
|
||||||
A: serde::de::SeqAccess<'de>,
|
|
||||||
{
|
|
||||||
let mut state = self.state.clone();
|
|
||||||
let mut history = match seq.size_hint() {
|
|
||||||
Some(capacity) => Vec::with_capacity(capacity),
|
|
||||||
None => Vec::new(),
|
|
||||||
};
|
|
||||||
while let Some(instruction) = seq.next_element::<G::Instruction>()? {
|
|
||||||
history.push(StateSnapshot {
|
|
||||||
state: state.clone(),
|
|
||||||
instruction: instruction.clone(),
|
|
||||||
});
|
|
||||||
state.process_instruction(
|
|
||||||
&mut self.stats.inner,
|
|
||||||
&self.config.inner,
|
|
||||||
instruction,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
Ok(SessionState { state, history })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
let state = G::deserialize(deserializer)?;
|
|
||||||
deserializer.deserialize_seq(SessionStateVisitor {
|
|
||||||
state,
|
|
||||||
stats: self.stats,
|
|
||||||
config: self.config,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[cfg(feature = "serde")]
|
#[cfg(feature = "serde")]
|
||||||
impl<G: Game> serde::Serialize for SessionState<G>
|
impl<G: Game> serde::Serialize for Session<G>
|
||||||
where
|
where
|
||||||
G: serde::Serialize,
|
G: serde::Serialize,
|
||||||
|
G::Config: serde::Serialize,
|
||||||
G::Instruction: serde::Serialize,
|
G::Instruction: serde::Serialize,
|
||||||
{
|
{
|
||||||
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
||||||
@@ -744,13 +724,14 @@ where
|
|||||||
// serialize the initial state of the game.
|
// serialize the initial state of the game.
|
||||||
// if there is history, it is the first snapshot's state,
|
// if there is history, it is the first snapshot's state,
|
||||||
// otherwise it is the current game state since there are no moves.
|
// otherwise it is the current game state since there are no moves.
|
||||||
let state = if let Some(snapshot) = self.history.first() {
|
let state = if let Some(snapshot) = self.state.history.first() {
|
||||||
snapshot.state()
|
snapshot.state()
|
||||||
} else {
|
} else {
|
||||||
&self.state
|
&self.state.state
|
||||||
};
|
};
|
||||||
|
map.serialize_entry("config", &self.config)?;
|
||||||
map.serialize_entry("state", state)?;
|
map.serialize_entry("state", state)?;
|
||||||
map.serialize_entry("history", &History(&self.history))?;
|
map.serialize_entry("history", &History(&self.state.history))?;
|
||||||
map.end()
|
map.end()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user