1//! Versioned server description settings files.
23// NOTE: Needed to allow the second-to-last migration to call try_into().
45use super::{MIGRATION_UPGRADE_GUARANTEE, SERVER_DESCRIPTION_FILENAME as FILENAME};
6use crate::settings::editable::{EditableSetting, Version};
7use core::convert::{Infallible, TryFrom, TryInto};
8use serde::{Deserialize, Serialize};
910/// NOTE: Always replace this with the latest server description version. Then
11/// update the ServerDescriptionRaw, the TryFrom<ServerDescriptionRaw> for
12/// ServerDescription, the previously most recent module, and add a new module
13/// for the latest version! Please respect the migration upgrade guarantee
14/// found in the parent module with any upgrade.
15pub use self::v2::*;
1617/// Versioned settings files, one per version (v0 is only here as an example; we
18/// do not expect to see any actual v0 settings files).
19#[derive(Deserialize, Serialize)]
20pub enum ServerDescriptionRaw {
21 V0(v0::ServerDescription),
22 V1(v1::ServerDescription),
23 V2(ServerDescriptions),
24}
2526impl From<ServerDescriptions> for ServerDescriptionRaw {
27fn from(value: ServerDescriptions) -> Self {
28// Replace variant with that of current latest version.
29Self::V2(value)
30 }
31}
3233impl TryFrom<ServerDescriptionRaw> for (Version, ServerDescriptions) {
34type Error = <ServerDescriptions as EditableSetting>::Error;
3536fn try_from(
37 value: ServerDescriptionRaw,
38 ) -> Result<Self, <ServerDescriptions as EditableSetting>::Error> {
39use ServerDescriptionRaw::*;
40Ok(match value {
41// Old versions
42V0(value) => (Version::Old, value.try_into()?),
43 V1(value) => (Version::Old, value.try_into()?),
44// Latest version (move to old section using the pattern of other old version when it
45 // is no longer latest).
46V2(mut value) => (value.validate()?, value),
47 })
48 }
49}
5051type Final = ServerDescriptions;
5253impl EditableSetting for ServerDescriptions {
54type Error = Infallible;
55type Legacy = legacy::ServerDescription;
56type Setting = ServerDescriptionRaw;
5758const FILENAME: &'static str = FILENAME;
59}
6061mod legacy {
62use super::{Final, MIGRATION_UPGRADE_GUARANTEE, v0 as next};
63use core::convert::TryInto;
64use serde::{Deserialize, Serialize};
6566#[derive(Deserialize, Serialize)]
67 #[serde(transparent)]
68pub struct ServerDescription(pub(super) String);
6970impl From<ServerDescription> for Final {
71/// Legacy migrations can be migrated to the latest version through the
72 /// process of "chaining" migrations, starting from
73 /// `next::ServerDescription`.
74 ///
75 /// Note that legacy files are always valid, which is why we implement
76 /// From rather than TryFrom.
77fn from(value: ServerDescription) -> Self {
78 next::ServerDescription::migrate(value)
79 .try_into()
80 .expect(MIGRATION_UPGRADE_GUARANTEE)
81 }
82 }
83}
8485/// This module represents a server description version that isn't actually
86/// used. It is here and part of the migration process to provide an example
87/// for how to perform a migration for an old version; please use this as a
88/// reference when constructing new migrations.
89mod v0 {
90use super::{Final, MIGRATION_UPGRADE_GUARANTEE, legacy as prev, v1 as next};
91use crate::settings::editable::{EditableSetting, Version};
92use core::convert::{TryFrom, TryInto};
93use serde::{Deserialize, Serialize};
9495#[derive(Clone, Deserialize, Serialize)]
96 #[serde(transparent)]
97pub struct ServerDescription(pub(super) String);
9899impl ServerDescription {
100/// One-off migration from the previous version. This must be
101 /// guaranteed to produce a valid settings file as long as it is
102 /// called with a valid settings file from the previous version.
103pub(super) fn migrate(prev: prev::ServerDescription) -> Self { ServerDescription(prev.0) }
104105/// Perform any needed validation on this server description that can't
106 /// be done using parsing.
107 ///
108 /// The returned version being "Old" indicates the loaded setting has
109 /// been modified during validation (this is why validate takes
110 /// `&mut self`).
111pub(super) fn validate(&mut self) -> Result<Version, <Final as EditableSetting>::Error> {
112Ok(Version::Latest)
113 }
114 }
115116/// Pretty much every TryFrom implementation except that of the very last
117 /// version should look exactly like this.
118impl TryFrom<ServerDescription> for Final {
119type Error = <Final as EditableSetting>::Error;
120121fn try_from(mut value: ServerDescription) -> Result<Final, Self::Error> {
122 value.validate()?;
123Ok(next::ServerDescription::migrate(value)
124 .try_into()
125 .expect(MIGRATION_UPGRADE_GUARANTEE))
126 }
127 }
128}
129130mod v1 {
131use super::{Final, v0 as prev};
132use crate::settings::editable::{EditableSetting, Version};
133use core::ops::{Deref, DerefMut};
134use serde::{Deserialize, Serialize};
135136#[derive(Clone, Deserialize, Serialize)]
137 #[serde(transparent)]
138pub struct ServerDescription(pub(super) String);
139140impl Default for ServerDescription {
141fn default() -> Self { Self("This is the best Veloren server".into()) }
142 }
143144impl Deref for ServerDescription {
145type Target = String;
146147fn deref(&self) -> &Self::Target { &self.0 }
148 }
149150impl DerefMut for ServerDescription {
151fn deref_mut(&mut self) -> &mut Self::Target { &mut self.0 }
152 }
153154impl ServerDescription {
155/// One-off migration from the previous version. This must be
156 /// guaranteed to produce a valid settings file as long as it is
157 /// called with a valid settings file from the previous version.
158pub(super) fn migrate(prev: prev::ServerDescription) -> Self { ServerDescription(prev.0) }
159160/// Perform any needed validation on this server description that can't
161 /// be done using parsing.
162 ///
163 /// The returned version being "Old" indicates the loaded setting has
164 /// been modified during validation (this is why validate takes
165 /// `&mut self`).
166pub(super) fn validate(&mut self) -> Result<Version, <Final as EditableSetting>::Error> {
167Ok(Version::Latest)
168 }
169 }
170171use super::v2 as next;
172impl TryFrom<ServerDescription> for Final {
173type Error = <Final as EditableSetting>::Error;
174175fn try_from(mut value: ServerDescription) -> Result<Final, Self::Error> {
176 value.validate()?;
177Ok(next::ServerDescriptions::migrate(value))
178 }
179 }
180}
181182mod v2 {
183use std::collections::HashMap;
184185use super::{Final, v1 as prev};
186use crate::settings::editable::{EditableSetting, Version};
187use serde::{Deserialize, Serialize};
188189/// Map of all localized [`ServerDescription`]s
190#[derive(Clone, Deserialize, Serialize)]
191pub struct ServerDescriptions {
192pub default_locale: String,
193pub descriptions: HashMap<String, ServerDescription>,
194 }
195196#[derive(Clone, Deserialize, Serialize)]
197pub struct ServerDescription {
198pub motd: String,
199pub rules: Option<String>,
200 }
201202impl Default for ServerDescriptions {
203fn default() -> Self {
204Self {
205 default_locale: "en".to_string(),
206 descriptions: HashMap::from([("en".to_string(), ServerDescription::default())]),
207 }
208 }
209 }
210211impl Default for ServerDescription {
212fn default() -> Self {
213Self {
214 motd: "This is the best Veloren server".into(),
215 rules: None,
216 }
217 }
218 }
219220impl ServerDescriptions {
221fn unwrap_locale_or_default<'a, 'b: 'a>(&'b self, locale: Option<&'a str>) -> &'a str {
222 locale.map_or(&self.default_locale, |locale| {
223if self.descriptions.contains_key(locale) {
224 locale
225 } else {
226&self.default_locale
227 }
228 })
229 }
230231pub fn get(&self, locale: Option<&str>) -> Option<&ServerDescription> {
232self.descriptions.get(self.unwrap_locale_or_default(locale))
233 }
234235/// Attempts to get the rules in the specified locale, falls back to
236 /// `default_locale` if no rules were specified in this locale
237pub fn get_rules(&self, locale: Option<&str>) -> Option<&str> {
238self.descriptions
239 .get(self.unwrap_locale_or_default(locale))
240 .and_then(|d| d.rules.as_deref())
241 .or_else(|| {
242self.descriptions
243 .get(&self.default_locale)?
244.rules
245 .as_deref()
246 })
247 }
248 }
249250impl ServerDescriptions {
251/// One-off migration from the previous version. This must be
252 /// guaranteed to produce a valid settings file as long as it is
253 /// called with a valid settings file from the previous version.
254pub(super) fn migrate(prev: prev::ServerDescription) -> Self {
255Self {
256 default_locale: "en".to_string(),
257 descriptions: HashMap::from([("en".to_string(), ServerDescription {
258 motd: prev.0,
259 rules: None,
260 })]),
261 }
262 }
263264/// Perform any needed validation on this server description that can't
265 /// be done using parsing.
266 ///
267 /// The returned version being "Old" indicates the loaded setting has
268 /// been modified during validation (this is why validate takes
269 /// `&mut self`).
270pub(super) fn validate(&mut self) -> Result<Version, <Final as EditableSetting>::Error> {
271if self.descriptions.is_empty() {
272*self = Self::default();
273Ok(Version::Old)
274 } else if !self.descriptions.contains_key(&self.default_locale) {
275// default locale not present, select the a random one (as ordering in hashmaps
276 // isn't predictable)
277self.default_locale = self
278.descriptions
279 .keys()
280 .next()
281 .expect("We know descriptions isn't empty")
282 .to_string();
283Ok(Version::Old)
284 } else {
285Ok(Version::Latest)
286 }
287 }
288 }
289290// NOTE: Whenever there is a version upgrade, copy this note as well as the
291 // commented-out code below to the next version, then uncomment the code
292 // for this version.
293 /*
294 use super::{v3 as next, MIGRATION_UPGRADE_GUARANTEE};
295 impl TryFrom<ServerDescription> for Final {
296 type Error = <Final as EditableSetting>::Error;
297298 fn try_from(mut value: ServerDescription) -> Result<Final, Self::Error> {
299 value.validate()?;
300 Ok(next::ServerDescription::migrate(value).try_into().expect(MIGRATION_UPGRADE_GUARANTEE))
301 }
302 } */
303}