Files
RobustToolbox/Robust.Shared/Configuration/ConfigurationManager.cs
Acruid 2183cd7ca1 Massive Namespace Cleanup (#1544)
* Removed the Interfaces folder.
* All objects inside the GameObjects subfolders are now in the GameObjects namespace.
* Added a Resharper DotSettings file to mark the GameObjects subfolders as not providing namespaces.
* Simplified Robust.client.Graphics namespace.
* Automated remove redundant using statements.
2021-02-10 23:27:19 -08:00

506 lines
18 KiB
C#

using Nett;
using Robust.Shared.Log;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Reflection;
using Robust.Shared.Utility;
namespace Robust.Shared.Configuration
{
/// <summary>
/// Stores and manages global configuration variables.
/// </summary>
internal class ConfigurationManager : IConfigurationManagerInternal
{
private const char TABLE_DELIMITER = '.';
protected readonly Dictionary<string, ConfigVar> _configVars = new();
private string? _configFile;
protected bool _isServer;
/// <summary>
/// Constructs a new ConfigurationManager.
/// </summary>
public ConfigurationManager()
{
}
public void Initialize(bool isServer)
{
_isServer = isServer;
}
/// <inheritdoc />
public void LoadFromFile(string configFile)
{
try
{
var tblRoot = Toml.ReadFile(configFile);
ProcessTomlObject(tblRoot);
_configFile = configFile;
Logger.InfoS("cfg", $"Configuration Loaded from '{Path.GetFullPath(configFile)}'");
}
catch (Exception e)
{
Logger.WarningS("cfg", "Unable to load configuration file:\n{0}", e);
}
}
public void SetSaveFile(string configFile)
{
_configFile = configFile;
}
/// <summary>
/// A recursive function that walks over the config tree, transforming all key nodes into CVars.
/// </summary>
/// <param name="obj">The root table of the TOML document.</param>
/// <param name="tablePath">For internal use only, the current path to the node.</param>
private void ProcessTomlObject(TomlObject obj, string tablePath = "")
{
if (obj is TomlTable table) // this is a table
{
foreach (var kvTml in table)
{
string newPath;
if ((kvTml.Value is TomlTable))
newPath = tablePath + kvTml.Key + TABLE_DELIMITER;
else
newPath = tablePath + kvTml.Key;
ProcessTomlObject(kvTml.Value, newPath);
}
}
else // this is a key, add CVar
{
// if the CVar has already been registered
var tomlValue = TypeConvert(obj);
if (_configVars.TryGetValue(tablePath, out var cfgVar))
{
// overwrite the value with the saved one
cfgVar.Value = tomlValue;
cfgVar.ValueChanged?.Invoke(cfgVar.Value);
}
else
{
//or add another unregistered CVar
//Note: the defaultValue is arbitrarily 0, it will get overwritten when the cvar is registered.
cfgVar = new ConfigVar(tablePath, 0, CVar.NONE) { Value = tomlValue };
_configVars.Add(tablePath, cfgVar);
}
cfgVar.ConfigModified = true;
}
}
/// <inheritdoc />
public void SaveToFile()
{
if (_configFile == null)
{
Logger.WarningS("cfg", "Cannot save the config file, because one was never loaded.");
return;
}
try
{
var tblRoot = Toml.Create();
foreach (var (name, cVar) in _configVars)
{
var value = cVar.Value;
if (value == null && cVar.Registered)
{
value = cVar.DefaultValue;
}
if (value == null)
{
Logger.ErrorS("cfg", $"CVar {name} has no value or default value, was the default value registered as null?");
continue;
}
// Don't write if Archive flag is not set.
// Don't write if the cVar is the default value.
if (!cVar.ConfigModified &&
(cVar.Flags & CVar.ARCHIVE) == 0 || value.Equals(cVar.DefaultValue))
{
continue;
}
var keyIndex = name.LastIndexOf(TABLE_DELIMITER);
var tblPath = name.Substring(0, keyIndex).Split(TABLE_DELIMITER);
var keyName = name.Substring(keyIndex + 1);
// locate the Table in the config tree
var table = tblRoot;
foreach (var curTblName in tblPath)
{
if (!table.TryGetValue(curTblName, out TomlObject tblObject))
{
tblObject = table.Add(curTblName, new Dictionary<string, TomlObject>()).Added;
}
table = tblObject as TomlTable ?? throw new InvalidConfigurationException($"[CFG] Object {curTblName} is being used like a table, but it is a {tblObject}. Are your CVar names formed properly?");
}
//runtime unboxing, either this or generic hell... ¯\_(ツ)_/¯
switch (value)
{
case Enum val:
table.Add(keyName, (int)(object)val); // asserts Enum value != (ulong || long)
break;
case int val:
table.Add(keyName, val);
break;
case long val:
table.Add(keyName, val);
break;
case bool val:
table.Add(keyName, val);
break;
case string val:
table.Add(keyName, val);
break;
case float val:
table.Add(keyName, val);
break;
case double val:
table.Add(keyName, val);
break;
default:
Logger.WarningS("cfg", $"Cannot serialize '{name}', unsupported type.");
break;
}
}
Toml.WriteFile(tblRoot, _configFile);
Logger.InfoS("cfg", $"config saved to '{_configFile}'.");
}
catch (Exception e)
{
Logger.WarningS("cfg", $"Cannot save the config file '{_configFile}'.\n {e}");
}
}
public void RegisterCVar<T>(string name, T defaultValue, CVar flags = CVar.NONE, Action<T>? onValueChanged = null)
where T : notnull
{
Action<object>? valueChangedDelegate = null;
if (onValueChanged != null)
{
valueChangedDelegate = v => onValueChanged((T) v);
}
RegisterCVar(name, typeof(T), defaultValue, flags, valueChangedDelegate);
}
private void RegisterCVar(string name, Type type, object defaultValue, CVar flags, Action<object>? onValueChanged)
{
DebugTools.Assert(!type.IsEnum || type.GetEnumUnderlyingType() == typeof(int),
$"{name}: Enum cvars must have int as underlying type.");
var only = _isServer ? CVar.CLIENTONLY : CVar.SERVERONLY;
if ((flags & only) != 0)
{
// Ignored on this side.
return;
}
if (_configVars.TryGetValue(name, out var cVar))
{
if (cVar.Registered)
Logger.ErrorS("cfg", $"The variable '{name}' has already been registered.");
cVar.DefaultValue = defaultValue;
cVar.Flags = flags;
cVar.Registered = true;
cVar.ValueChanged = onValueChanged;
if (cVar.OverrideValue != null)
{
cVar.OverrideValueParsed = ParseOverrideValue(cVar.OverrideValue, type);
}
return;
}
_configVars.Add(name, new ConfigVar(name, defaultValue, flags)
{
Registered = true,
Value = defaultValue,
ValueChanged = onValueChanged
});
}
public void OnValueChanged<T>(CVarDef<T> cVar, Action<T> onValueChanged, bool invokeImmediately = false)
where T : notnull
{
OnValueChanged(cVar.Name, onValueChanged, invokeImmediately);
}
public void OnValueChanged<T>(string name, Action<T> onValueChanged, bool invokeImmediately = false)
where T : notnull
{
var reg = _configVars[name];
reg.ValueChanged += o => onValueChanged((T) o);
if (invokeImmediately)
{
onValueChanged((T) (reg.Value ?? reg.DefaultValue)!);
}
}
public void LoadCVarsFromAssembly(Assembly assembly)
{
foreach (var defField in assembly
.GetTypes()
.Where(p => Attribute.IsDefined(p, typeof(CVarDefsAttribute)))
.SelectMany(p => p.GetFields(BindingFlags.Public | BindingFlags.Static)))
{
var fieldType = defField.FieldType;
if (!fieldType.IsGenericType || fieldType.GetGenericTypeDefinition() != typeof(CVarDef<>))
{
continue;
}
var type = fieldType.GetGenericArguments()[0];
if (!defField.IsInitOnly)
{
throw new InvalidOperationException($"Found CVarDef '{defField.Name}' on '{defField.DeclaringType?.FullName}' that is not readonly. Please mark it as readonly.");
}
var def = (CVarDef?) defField.GetValue(null);
if (def == null)
{
throw new InvalidOperationException($"CVarDef '{defField.Name}' on '{defField.DeclaringType?.FullName}' is null.");
}
RegisterCVar(def.Name, type, def.DefaultValue, def.Flags, null);
}
}
/// <inheritdoc />
public bool IsCVarRegistered(string name)
{
return _configVars.TryGetValue(name, out var cVar) && cVar.Registered;
}
/// <inheritdoc />
public IEnumerable<string> GetRegisteredCVars()
{
return _configVars.Select(p => p.Key);
}
/// <inheritdoc />
public virtual void SetCVar(string name, object value)
{
SetCVarInternal(name, value);
}
private void SetCVarInternal(string name, object value)
{
//TODO: Make flags work, required non-derpy net system.
if (_configVars.TryGetValue(name, out var cVar) && cVar.Registered)
{
if (!Equals(cVar.OverrideValueParsed ?? cVar.Value, value))
{
// Setting an overriden var just turns off the override, basically.
cVar.OverrideValue = null;
cVar.OverrideValueParsed = null;
cVar.Value = value;
cVar.ValueChanged?.Invoke(value);
}
}
else
throw new InvalidConfigurationException($"Trying to set unregistered variable '{name}'");
}
public void SetCVar<T>(CVarDef<T> def, T value) where T : notnull
{
SetCVar(def.Name, value);
}
/// <inheritdoc />
public T GetCVar<T>(string name)
{
if (_configVars.TryGetValue(name, out var cVar) && cVar.Registered)
//TODO: Make flags work, required non-derpy net system.
return (T)(cVar.OverrideValueParsed ?? cVar.Value ?? cVar.DefaultValue)!;
throw new InvalidConfigurationException($"Trying to get unregistered variable '{name}'");
}
public T GetCVar<T>(CVarDef<T> def) where T : notnull
{
return GetCVar<T>(def.Name);
}
public Type GetCVarType(string name)
{
if (!_configVars.TryGetValue(name, out var cVar) || !cVar.Registered)
{
throw new InvalidConfigurationException($"Trying to get type of unregistered variable '{name}'");
}
// If it's null it's a string, since the rest is primitives which aren't null.
return cVar.Value?.GetType() ?? typeof(string);
}
public void OverrideConVars(IEnumerable<(string key, string value)> cVars)
{
foreach (var (key, value) in cVars)
{
if (_configVars.TryGetValue(key, out var cfgVar))
{
cfgVar.OverrideValue = value;
cfgVar.OverrideValueParsed = ParseOverrideValue(value, cfgVar.DefaultValue?.GetType());
cfgVar.ValueChanged?.Invoke(cfgVar.OverrideValueParsed);
}
else
{
//or add another unregistered CVar
//Note: the defaultValue is arbitrarily 0, it will get overwritten when the cvar is registered.
var cVar = new ConfigVar(key, 0, CVar.NONE) { OverrideValue = value };
_configVars.Add(key, cVar);
}
}
}
private static object ParseOverrideValue(string value, Type? type)
{
if (type == typeof(int))
{
return int.Parse(value);
}
if (type == typeof(bool))
{
return bool.Parse(value);
}
if (type == typeof(float))
{
return float.Parse(value);
}
// Must be a string.
return value;
}
/// <summary>
/// Converts a TomlObject into its native type.
/// </summary>
/// <param name="obj">The object to convert.</param>
/// <returns>The boxed native type of the TomlObject.</returns>
private static object TypeConvert(TomlObject obj)
{
var tmlType = obj.TomlType;
switch (tmlType)
{
case TomlObjectType.Bool:
return obj.Get<bool>();
case TomlObjectType.Float:
return obj.Get<float>();
case TomlObjectType.Int:
return obj.Get<int>();
case TomlObjectType.String:
return obj.Get<string>();
default:
throw new InvalidConfigurationException($"Cannot convert {tmlType}.");
}
}
/// <summary>
/// Holds the data for a single configuration variable.
/// </summary>
protected class ConfigVar
{
/// <summary>
/// Constructs a CVar.
/// </summary>
/// <param name="name">The name of the CVar. This needs to contain only printable characters.
/// Underscores '_' are reserved. Everything before the last underscore is a table identifier,
/// everything after is the CVar name in the TOML document.</param>
/// <param name="defaultValue">The default value of this CVar.</param>
/// <param name="flags">Optional flags to modify the behavior of this CVar.</param>
public ConfigVar(string name, object defaultValue, CVar flags)
{
Name = name;
DefaultValue = defaultValue;
Flags = flags;
}
/// <summary>
/// The name of the CVar.
/// </summary>
public string Name { get; }
/// <summary>
/// The default value of this CVar.
/// </summary>
public object DefaultValue { get; set; }
/// <summary>
/// Optional flags to modify the behavior of this CVar.
/// </summary>
public CVar Flags { get; set; }
/// <summary>
/// The current value of this CVar.
/// </summary>
public object? Value { get; set; }
/// <summary>
/// Has this CVar been registered in code?
/// </summary>
public bool Registered { get; set; }
/// <summary>
/// Was the CVar present in the config file?
/// If so we need to always re-save it even if it's not ARCHIVE.
/// </summary>
public bool ConfigModified;
/// <summary>
/// Invoked when the value of this CVar is changed.
/// </summary>
public Action<object>? ValueChanged { get; set; }
// We don't know what the type of the var is until it's registered.
// So we can't actually parse them until then.
// So we keep the raw string around.
public string? OverrideValue { get; set; }
public object? OverrideValueParsed { get; set; }
}
}
[Serializable]
public class InvalidConfigurationException : Exception
{
public InvalidConfigurationException()
{
}
public InvalidConfigurationException(string message) : base(message)
{
}
public InvalidConfigurationException(string message, Exception inner) : base(message, inner)
{
}
protected InvalidConfigurationException(
System.Runtime.Serialization.SerializationInfo info,
System.Runtime.Serialization.StreamingContext context) : base(info, context) { }
}
}