mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
Add entity config migration tool
This commit is contained in:
parent
6f743b22ae
commit
765311add0
@ -11,6 +11,7 @@ simd = ["vek/platform_intrinsics"]
|
||||
bin_csv = ["ron", "csv", "structopt"]
|
||||
bin_graphviz = ["petgraph"]
|
||||
bin_cmd_doc_gen = []
|
||||
bin_entity_migrate = ["ron"]
|
||||
rrt_pathfinding = ["kiddo"]
|
||||
calendar_events = []
|
||||
|
||||
@ -106,6 +107,10 @@ required-features = ["bin_csv"]
|
||||
name = "csv_import"
|
||||
required-features = ["bin_csv"]
|
||||
|
||||
[[bin]]
|
||||
name = "entity_migrate"
|
||||
required-features = ["bin_entity_migrate"]
|
||||
|
||||
[[bin]]
|
||||
name = "recipe_graphviz"
|
||||
required-features = ["bin_graphviz"]
|
||||
|
331
common/src/bin/entity_migrate.rs
Normal file
331
common/src/bin/entity_migrate.rs
Normal file
@ -0,0 +1,331 @@
|
||||
use serde::{de::DeserializeOwned, Deserialize, Serialize};
|
||||
use veloren_common::{
|
||||
comp::{inventory::loadout_builder::ItemSpec, Alignment, Body},
|
||||
lottery::LootSpec,
|
||||
};
|
||||
|
||||
use std::{
|
||||
fs, io,
|
||||
io::Write,
|
||||
path::{Path, PathBuf},
|
||||
};
|
||||
|
||||
/// First "stable" version.
|
||||
mod v1 {
|
||||
pub(super) use super::*;
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum NameKind {
|
||||
Name(String),
|
||||
Automatic,
|
||||
Uninit,
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum BodyBuilder {
|
||||
RandomWith(String),
|
||||
Exact(Body),
|
||||
Uninit,
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum AlignmentMark {
|
||||
Alignment(Alignment),
|
||||
Uninit,
|
||||
}
|
||||
|
||||
impl Default for AlignmentMark {
|
||||
fn default() -> Self { Self::Alignment(Alignment::Wild) }
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum Hands {
|
||||
TwoHanded(ItemSpec),
|
||||
Paired(ItemSpec),
|
||||
Mix {
|
||||
mainhand: ItemSpec,
|
||||
offhand: ItemSpec,
|
||||
},
|
||||
Uninit,
|
||||
}
|
||||
|
||||
impl Default for Hands {
|
||||
fn default() -> Self { Self::Uninit }
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum Meta {
|
||||
LoadoutAsset(String),
|
||||
SkillSetAsset(String),
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub struct EntityConfig {
|
||||
pub name: NameKind,
|
||||
pub body: BodyBuilder,
|
||||
pub alignment: AlignmentMark,
|
||||
pub loot: LootSpec<String>,
|
||||
pub hands: Hands,
|
||||
#[serde(default)]
|
||||
pub meta: Vec<Meta>,
|
||||
}
|
||||
}
|
||||
|
||||
/// Loadout update.
|
||||
/// 1) Added ability to randomize loadout for entity.
|
||||
/// 2) Simplified logic by squashing hands, loadout and inventory into one pack.
|
||||
mod v2 {
|
||||
pub(super) use super::*;
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum LoadoutAsset {
|
||||
Loadout(String),
|
||||
Choice(Vec<(f32, String)>),
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum Hands {
|
||||
TwoHanded(ItemSpec),
|
||||
Paired(ItemSpec),
|
||||
Mix {
|
||||
mainhand: ItemSpec,
|
||||
offhand: ItemSpec,
|
||||
},
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum LoadoutKind {
|
||||
FromBody,
|
||||
Asset(LoadoutAsset),
|
||||
Hands(Hands),
|
||||
Extended {
|
||||
hands: Hands,
|
||||
base_asset: LoadoutAsset,
|
||||
inventory: Vec<(u32, String)>,
|
||||
},
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub enum Meta {
|
||||
SkillSetAsset(String),
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize, Deserialize, Clone)]
|
||||
pub struct EntityConfig {
|
||||
pub name: super::v1::NameKind,
|
||||
pub body: super::v1::BodyBuilder,
|
||||
pub alignment: super::v1::AlignmentMark,
|
||||
pub loadout: LoadoutKind,
|
||||
pub loot: super::v1::LootSpec<String>,
|
||||
#[serde(default)]
|
||||
pub meta: Vec<Meta>,
|
||||
}
|
||||
|
||||
impl From<super::v1::EntityConfig> for EntityConfig {
|
||||
fn from(old_config: super::v1::EntityConfig) -> Self {
|
||||
let mut loadout_asset = None;
|
||||
let mut meta = Vec::new();
|
||||
|
||||
for item in old_config.meta {
|
||||
match item {
|
||||
super::v1::Meta::SkillSetAsset(asset) => {
|
||||
meta.push(Meta::SkillSetAsset(asset));
|
||||
},
|
||||
super::v1::Meta::LoadoutAsset(asset) => {
|
||||
if loadout_asset == None {
|
||||
loadout_asset = Some(asset);
|
||||
} else {
|
||||
tracing::error!("multiple loadout assets in meta[], bad");
|
||||
}
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
let loadout_kind = match loadout_asset {
|
||||
Some(asset) => match old_config.hands {
|
||||
super::v1::Hands::TwoHanded(spec) => LoadoutKind::Extended {
|
||||
hands: Hands::TwoHanded(spec),
|
||||
base_asset: LoadoutAsset::Loadout(asset),
|
||||
inventory: vec![],
|
||||
},
|
||||
super::v1::Hands::Paired(spec) => LoadoutKind::Extended {
|
||||
hands: Hands::Paired(spec),
|
||||
base_asset: LoadoutAsset::Loadout(asset),
|
||||
inventory: vec![],
|
||||
},
|
||||
super::v1::Hands::Mix { mainhand, offhand } => LoadoutKind::Extended {
|
||||
hands: Hands::Mix { mainhand, offhand },
|
||||
base_asset: LoadoutAsset::Loadout(asset),
|
||||
inventory: vec![],
|
||||
},
|
||||
super::v1::Hands::Uninit => LoadoutKind::Asset(LoadoutAsset::Loadout(asset)),
|
||||
},
|
||||
None => match old_config.hands {
|
||||
super::v1::Hands::TwoHanded(spec) => LoadoutKind::Hands(Hands::TwoHanded(spec)),
|
||||
super::v1::Hands::Paired(spec) => LoadoutKind::Hands(Hands::Paired(spec)),
|
||||
super::v1::Hands::Mix { mainhand, offhand } => {
|
||||
LoadoutKind::Hands(Hands::Mix { mainhand, offhand })
|
||||
},
|
||||
super::v1::Hands::Uninit => LoadoutKind::FromBody,
|
||||
},
|
||||
};
|
||||
|
||||
Self {
|
||||
name: old_config.name,
|
||||
body: old_config.body,
|
||||
alignment: old_config.alignment,
|
||||
loadout: loadout_kind,
|
||||
loot: old_config.loot,
|
||||
meta,
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn input_string(prompt: &str) -> String { input_validated_string(prompt, &|_| true) }
|
||||
|
||||
fn input_validated_string(prompt: &str, check: &dyn Fn(&str) -> bool) -> String {
|
||||
println!("{}", prompt);
|
||||
|
||||
print!("> ");
|
||||
io::stdout().flush().unwrap();
|
||||
|
||||
let mut buff = String::new();
|
||||
io::stdin().read_line(&mut buff).unwrap();
|
||||
|
||||
while !check(buff.trim()) {
|
||||
buff.clear();
|
||||
print!("> ");
|
||||
io::stdout().flush().unwrap();
|
||||
io::stdin().read_line(&mut buff).unwrap();
|
||||
}
|
||||
|
||||
buff.trim().to_owned()
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
enum Walk {
|
||||
File(PathBuf),
|
||||
Dir { path: PathBuf, content: Vec<Walk> },
|
||||
}
|
||||
|
||||
fn walk_tree(dir: &Path, root: &Path) -> io::Result<Vec<Walk>> {
|
||||
let mut buff = Vec::new();
|
||||
for entry in fs::read_dir(dir)? {
|
||||
let entry = entry?;
|
||||
let path = entry.path();
|
||||
if path.is_dir() {
|
||||
buff.push(Walk::Dir {
|
||||
path: path
|
||||
.strip_prefix(root)
|
||||
.expect("strip can't fail, this path is created from root")
|
||||
.to_owned(),
|
||||
content: walk_tree(&path, root)?,
|
||||
});
|
||||
} else {
|
||||
let filename = path
|
||||
.strip_prefix(root)
|
||||
.expect("strip can't fail, this file is created from root")
|
||||
.to_owned();
|
||||
buff.push(Walk::File(filename));
|
||||
}
|
||||
}
|
||||
|
||||
Ok(buff)
|
||||
}
|
||||
|
||||
fn walk_with_migrate<OldV, NewV>(tree: Walk, from: &Path, to: &Path) -> std::io::Result<()>
|
||||
where
|
||||
NewV: From<OldV>,
|
||||
OldV: DeserializeOwned,
|
||||
NewV: Serialize,
|
||||
{
|
||||
match tree {
|
||||
Walk::Dir { path, content } => {
|
||||
let target_dir = to.join(path);
|
||||
fs::create_dir_all(target_dir)?;
|
||||
for entry in content {
|
||||
walk_with_migrate::<OldV, NewV>(entry, from, to)?;
|
||||
}
|
||||
},
|
||||
Walk::File(path) => {
|
||||
let source = fs::File::open(from.join(&path))?;
|
||||
let old: OldV = ron::de::from_reader(source).unwrap();
|
||||
let new: NewV = old.into();
|
||||
let target = fs::File::create(to.join(&path))?;
|
||||
let pretty_config = ron::ser::PrettyConfig::new();
|
||||
ron::ser::to_writer_pretty(target, &new, pretty_config).unwrap();
|
||||
println!("{path:?} done");
|
||||
},
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn convert_loop(from: &str, to: &str, old_ver: &str, new_ver: &str) {
|
||||
#[rustfmt::skip]
|
||||
println!(
|
||||
"\nRequest info:\n\
|
||||
{old_ver} -> {new_ver}.\n\
|
||||
Get data from {from} and store in {to}."
|
||||
);
|
||||
|
||||
let root = Path::new(from);
|
||||
let files = Walk::Dir {
|
||||
path: Path::new("").to_owned(),
|
||||
content: walk_tree(&root, &root).unwrap(),
|
||||
};
|
||||
if old_ver == "v1" && new_ver == "v2" {
|
||||
walk_with_migrate::<v1::EntityConfig, v2::EntityConfig>(
|
||||
files,
|
||||
Path::new(from),
|
||||
Path::new(to),
|
||||
)
|
||||
.unwrap();
|
||||
} else {
|
||||
eprintln!("Unexpected versions")
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {
|
||||
println!(
|
||||
r#"
|
||||
Hello, this tool can convert all your entity configs to newer version.
|
||||
Currently it supports converting from "v1" to "v2".
|
||||
"#
|
||||
);
|
||||
|
||||
let old_dir = input_validated_string(
|
||||
"Please input directory path with old entity configs:",
|
||||
&|path| {
|
||||
if !Path::new(path).exists() {
|
||||
eprintln!("Source directory '{path}' does not exists.");
|
||||
false
|
||||
} else {
|
||||
true
|
||||
}
|
||||
},
|
||||
);
|
||||
let new_dir = input_string("Please input directory path to place new entity configs:");
|
||||
|
||||
let old_version =
|
||||
input_validated_string("Please input old version to migrate from:", &|version| {
|
||||
let olds = ["v1"];
|
||||
if !olds.contains(&version) {
|
||||
eprintln!("Unexpected version {version}. Available: {olds:?}");
|
||||
false
|
||||
} else {
|
||||
true
|
||||
}
|
||||
});
|
||||
let new_version = input_validated_string("Please input new version:", &|version| {
|
||||
let news = ["v2"];
|
||||
if !news.contains(&version) {
|
||||
eprintln!("Unexpected version {version}. Available: {news:?}");
|
||||
false
|
||||
} else {
|
||||
true
|
||||
}
|
||||
});
|
||||
|
||||
convert_loop(&old_dir, &new_dir, &old_version, &new_version)
|
||||
}
|
@ -50,7 +50,7 @@ impl assets::Asset for LoadoutSpec {
|
||||
const EXTENSION: &'static str = "ron";
|
||||
}
|
||||
|
||||
#[derive(Debug, Deserialize, Clone)]
|
||||
#[derive(Debug, Deserialize, Serialize, Clone)]
|
||||
pub enum ItemSpec {
|
||||
/// One specific item.
|
||||
/// Example:
|
||||
|
Loading…
Reference in New Issue
Block a user