#region Copyright & License Information /* * Copyright 2007-2010 The OpenRA Developers (see AUTHORS) * This file is part of OpenRA, which is free software. It is made * available to you under the terms of the GNU General Public License * as published by the Free Software Foundation. For more information, * see LICENSE. */ #endregion using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using OpenRA; using OpenRA.FileFormats; using OpenRA.Traits; namespace RALint { static class RALint { static int errors = 0; static void EmitError(string e) { Console.WriteLine("RALint(1,1): Error: {0}", e); ++errors; } static Dictionary ValidPrereqs; static int Main(string[] args) { try { // bind some nonfatal error handling into FieldLoader, so we don't just *explode*. ObjectCreator.MissingTypeAction = s => EmitError("Missing Type: {0}".F(s)); FieldLoader.UnknownFieldAction = (s, f) => EmitError("FieldLoader: Missing field `{0}` on `{1}`".F(s, f.Name)); AppDomain.CurrentDomain.AssemblyResolve += FileSystem.ResolveAssembly; Game.modData = new ModData(args); Rules.LoadRules(Game.modData.Manifest, new Map()); // all the @something names which actually EXIST. var psuedoPrereqs = Rules.Info.Values.Select(a => a.Traits.GetOrDefault()).Where(b => b != null) .Select(b => b.AlternateName).Where(n => n != null).SelectMany(a => a).Select(a => a.ToLowerInvariant()).Distinct(); ValidPrereqs = Rules.Info.Keys.Concat(psuedoPrereqs).ToDictionary(a => a, a => 0); foreach (var actorInfo in Rules.Info) foreach (var traitInfo in actorInfo.Value.Traits.WithInterface()) CheckTrait(actorInfo.Value, traitInfo); if (errors > 0) { Console.WriteLine("Errors: {0}", errors); return 1; } return 0; } catch (Exception e) { EmitError("Failed with exception: {0}".F(e)); return 1; } } static void CheckTrait(ActorInfo actorInfo, ITraitInfo traitInfo) { var actualType = traitInfo.GetType(); foreach (var field in actualType.GetFields()) { if (field.HasAttribute()) CheckReference(actorInfo, traitInfo, field, ValidPrereqs, "actor"); if (field.HasAttribute()) CheckReference(actorInfo, traitInfo, field, Rules.Weapons, "weapon"); if (field.HasAttribute()) CheckReference(actorInfo, traitInfo, field, Rules.Voices, "voice"); } } static string[] GetFieldValues(ITraitInfo traitInfo, FieldInfo fieldInfo) { var type = fieldInfo.FieldType; if (type == typeof(string)) return new string[] { (string)fieldInfo.GetValue(traitInfo) }; if (type == typeof(string[])) return (string[])fieldInfo.GetValue(traitInfo); EmitError("Bad type for reference on {0}.{1}. Supported types: string, string[]" .F(traitInfo.GetType().Name, fieldInfo.Name)); return new string[] { }; } static void CheckReference(ActorInfo actorInfo, ITraitInfo traitInfo, FieldInfo fieldInfo, Dictionary dict, string type) { var values = GetFieldValues(traitInfo, fieldInfo); foreach (var v in values) if (v != null && !dict.ContainsKey(v.ToLowerInvariant())) EmitError("{0}.{1}.{2}: Missing {3} `{4}`." .F(actorInfo.Name, traitInfo.GetType().Name, fieldInfo.Name, type, v)); } } }