2023-08-06 08:25:04 -04:00
|
|
|
// SPDX-FileCopyrightText: 2023 Joshua Goins <josh@redstrate.com>
|
|
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
2023-08-02 16:26:56 -04:00
|
|
|
use std::collections::HashMap;
|
|
|
|
use std::io::{BufRead, BufReader, BufWriter, Cursor, Write};
|
2023-08-06 08:25:04 -04:00
|
|
|
|
|
|
|
use crate::gamedata::MemoryBuffer;
|
2023-08-02 16:26:56 -04:00
|
|
|
|
2023-09-22 18:05:29 -04:00
|
|
|
/// Represents a collection of keys, mapped to their values.
|
2023-08-02 16:26:56 -04:00
|
|
|
#[derive(Debug)]
|
2023-09-22 18:05:29 -04:00
|
|
|
pub struct ConfigMap {
|
|
|
|
/// A map of setting name to value.
|
2023-10-11 13:00:50 -04:00
|
|
|
pub keys: Vec<(String, String)>,
|
2023-08-02 16:26:56 -04:00
|
|
|
}
|
|
|
|
|
2023-09-22 18:05:29 -04:00
|
|
|
/// Represents a config file, which is made up of categories and settings. Categories may have zero to one settings.
|
2023-08-02 16:26:56 -04:00
|
|
|
#[derive(Debug)]
|
2023-09-22 18:05:29 -04:00
|
|
|
pub struct ConfigFile {
|
|
|
|
/// The categories present in this config file.
|
2023-08-02 16:26:56 -04:00
|
|
|
pub categories: Vec<String>,
|
2023-09-22 18:05:29 -04:00
|
|
|
/// A mapping of category to keys.
|
|
|
|
pub settings: HashMap<String, ConfigMap>,
|
2023-08-02 16:26:56 -04:00
|
|
|
}
|
|
|
|
|
2023-09-22 18:05:29 -04:00
|
|
|
impl ConfigFile {
|
|
|
|
/// Parses an existing config file.
|
|
|
|
pub fn from_existing(buffer: &MemoryBuffer) -> Option<ConfigFile> {
|
|
|
|
let mut cfg = ConfigFile {
|
2023-08-02 16:26:56 -04:00
|
|
|
categories: Vec::new(),
|
|
|
|
settings: HashMap::new()
|
|
|
|
};
|
|
|
|
|
|
|
|
let cursor = Cursor::new(buffer);
|
|
|
|
let reader = BufReader::new(cursor);
|
|
|
|
|
|
|
|
let mut current_category: Option<String> = None;
|
|
|
|
|
|
|
|
for (_, line) in reader.lines().enumerate() {
|
|
|
|
let unwrap = line.unwrap();
|
2023-10-11 13:00:50 -04:00
|
|
|
if !unwrap.is_empty() && unwrap != "\0" {
|
2023-08-02 16:26:56 -04:00
|
|
|
if unwrap.contains('<') || unwrap.contains('>') {
|
|
|
|
let name = &unwrap[1..unwrap.len() - 1];
|
|
|
|
current_category = Some(String::from(name));
|
|
|
|
cfg.categories.push(String::from(name));
|
|
|
|
} else {
|
|
|
|
let parts = unwrap.split_once('\t').unwrap();
|
2023-10-11 13:00:50 -04:00
|
|
|
cfg.settings.entry(current_category.clone().unwrap()).or_insert_with(|| ConfigMap{ keys: Vec::new() });
|
2023-08-02 16:26:56 -04:00
|
|
|
|
2023-10-11 13:00:50 -04:00
|
|
|
cfg.settings.get_mut(¤t_category.clone().unwrap()).unwrap().keys.push((parts.0.to_string(), parts.1.to_string()));
|
2023-08-02 16:26:56 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(cfg)
|
|
|
|
}
|
|
|
|
|
2023-09-22 18:05:29 -04:00
|
|
|
/// Writes an existing config file to a buffer.
|
2023-08-02 16:26:56 -04:00
|
|
|
pub fn write_to_buffer(&self) -> Option<MemoryBuffer> {
|
|
|
|
let mut buffer = MemoryBuffer::new();
|
|
|
|
|
|
|
|
{
|
|
|
|
let cursor = Cursor::new(&mut buffer);
|
|
|
|
let mut writer = BufWriter::new(cursor);
|
|
|
|
|
|
|
|
for category in &self.categories {
|
2023-10-11 13:00:50 -04:00
|
|
|
writer.write_all(format!("\r\n<{}>\r\n", category).as_ref()).ok()?;
|
2023-08-02 16:26:56 -04:00
|
|
|
|
|
|
|
if self.settings.contains_key(category) {
|
|
|
|
for key in &self.settings[category].keys {
|
2023-10-11 13:00:50 -04:00
|
|
|
writer.write_all(format!("{}\t{}\r\n", key.0, key.1).as_ref()).ok()?;
|
2023-08-02 16:26:56 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-11 13:00:50 -04:00
|
|
|
|
|
|
|
writer.write_all(b"\0").ok()?;
|
2023-08-02 16:26:56 -04:00
|
|
|
}
|
|
|
|
|
2023-10-11 13:00:50 -04:00
|
|
|
|
2023-08-02 16:26:56 -04:00
|
|
|
Some(buffer)
|
|
|
|
}
|
2023-10-11 13:00:50 -04:00
|
|
|
|
|
|
|
pub fn has_key(&self, select_key: &str) -> bool {
|
|
|
|
for (category, keys) in &self.settings {
|
|
|
|
for (key, value) in &keys.keys {
|
|
|
|
if select_key == key {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn has_category(&self, select_category: &str) -> bool {
|
|
|
|
for (category, keys) in &self.settings {
|
|
|
|
if select_category == category {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_value(&mut self, select_key: &str, new_value: &str) {
|
|
|
|
for (category, keys) in &mut self.settings {
|
|
|
|
for (key, value) in &mut keys.keys {
|
|
|
|
if select_key == key {
|
|
|
|
*value = new_value.to_string();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use std::fs::{read, write};
|
|
|
|
use std::path::PathBuf;
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
fn common_setup() -> ConfigFile {
|
|
|
|
let mut d = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
|
|
|
d.push("resources/tests");
|
|
|
|
d.push("FFXIV.cfg");
|
|
|
|
|
|
|
|
ConfigFile::from_existing(&read(d).unwrap()).unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn common_setup_modified() -> MemoryBuffer {
|
|
|
|
let mut d = PathBuf::from(env!("CARGO_MANIFEST_DIR"));
|
|
|
|
d.push("resources/tests");
|
|
|
|
d.push("FFXIV.modified.cfg");
|
|
|
|
|
|
|
|
read(d).unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn basic_parsing() {
|
|
|
|
let cfg = common_setup();
|
|
|
|
|
|
|
|
assert!(cfg.has_key("TextureFilterQuality"));
|
|
|
|
assert!(cfg.has_category("Cutscene Settings"));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn basic_writing() {
|
|
|
|
let mut cfg = common_setup();
|
|
|
|
let modified_cfg = common_setup_modified();
|
|
|
|
|
|
|
|
cfg.set_value("CutsceneMovieOpening", "1");
|
|
|
|
|
|
|
|
let cfg_buffer = cfg.write_to_buffer().unwrap();
|
|
|
|
|
|
|
|
assert_eq!(modified_cfg, cfg_buffer);
|
|
|
|
}
|
2023-08-02 16:26:56 -04:00
|
|
|
}
|