From 11a2dcb819c848f7def94cdb268e8cc11bd5aa8a Mon Sep 17 00:00:00 2001 From: F K <54195004+fredjk-gh@users.noreply.github.com> Date: Mon, 29 Aug 2022 08:15:51 -0400 Subject: [PATCH] Adds ring/belt helper functions, improved criteria debuggability (#96) Adds new functions to the Lua environment available to custom criteria for handling rings and belts: - the existing `rings(scan.Rings)` function behavior is unchanged - NEW: `ringsOnly(scan.Rings)` and `beltsOnly(scan.Rings)` functions return ONLY rings or belts, respectively, and shortcuts the `scan.Rings` check - NEW: `hasRings(scan.Rings)` and `hasBelts(scan.Rings)` functions return true if there are one or more proper rings or belts, respectively, in scan.Rings and shortcuts the check, too Furthermore, this adds a debugging aid that identifies what custom criteria caused an error/crash. For simple criteria, ie: `::Hot Landable::` ... the description "Hot Landable" is used. For complex criteria, you can now annotate the `::Criteria::` line with a description used for debugging like so: `::Criteria=Hot Landable::`. Criteria errors are logged like this: ``` Error encountered in Elite Observatory from plugin Explorer while processing custom criteria 'Criteria14=I just crash' on scan: ``` Duplicates are avoided by prepending the user-provided description with a generated name (ie. `Criteria14`, in the example above). Finally, the Detail expression for simple criteria can now safely reference values checked as part of the criteria condition. In particular, the Detail expression is now conditionally evaluated if the result is true. So the following will now work (previously it failed for non-star bodies when StarType was nil): ``` ::Ringed Star:: scan.StarType and hasRings(scan.Rings) ::Detail:: 'Star type: ' .. scan.StarType ``` --- ObservatoryExplorer/CustomCriteriaManager.cs | 92 ++++++++++++++++---- 1 file changed, 76 insertions(+), 16 deletions(-) diff --git a/ObservatoryExplorer/CustomCriteriaManager.cs b/ObservatoryExplorer/CustomCriteriaManager.cs index df52544..82c8842 100644 --- a/ObservatoryExplorer/CustomCriteriaManager.cs +++ b/ObservatoryExplorer/CustomCriteriaManager.cs @@ -11,7 +11,7 @@ namespace Observatory.Explorer internal class CustomCriteriaManager { private Lua LuaState; - private List CriteriaFunctions; + private Dictionary CriteriaFunctions; Action ErrorLogger; private uint ScanCount; @@ -50,17 +50,21 @@ namespace Observatory.Explorer end end"); - //Rings + //Rings - internal filterable iterator LuaState.DoString(@" - function rings (ring_list) + function _ringsFiltered (ring_list, filter_by) if ring_list then local i = 0 local count = ring_list.Count return function () i = i + 1 - if i <= count then + while i <= count do local ring = ring_list[i - 1] - return { name = ring.Name, ringclass = ring.RingClass, massmt = ring.MassMT, innerrad = ring.InnerRad, outerrad = ring.OuterRad } + if (filter_by == nil or string.find(ring.Name, filter_by)) then + return { name = ring.Name, ringclass = ring.RingClass, massmt = ring.MassMT, innerrad = ring.InnerRad, outerrad = ring.OuterRad } + else + i = i + 1 + end end end else @@ -68,6 +72,52 @@ namespace Observatory.Explorer end end"); + //Rings - internal filterable hasX check + LuaState.DoString(@" + function _hasRingsFiltered (ring_list, filter_by) + if ring_list then + local i = 0 + local count = ring_list.Count + while i < count do + if string.find(ring_list[i].Name, filter_by) then + return true + end + i = i + 1 + end + end + return false + end"); + + //Rings - iterate all - nil filter + LuaState.DoString(@" + function rings (ring_list) + return _ringsFiltered(ring_list, nil) + end"); + + //Rings - iterate proper rings only + LuaState.DoString(@" + function ringsOnly (ring_list) + return _ringsFiltered(ring_list, 'Ring') + end"); + + //Rings - has > 0 proper rings + LuaState.DoString(@" + function hasRings (ring_list) + return _hasRingsFiltered(ring_list, 'Ring') + end"); + + //Rings - iterate belts only + LuaState.DoString(@" + function beltsOnly (ring_list) + return _ringsFiltered(ring_list, 'Belt') + end"); + + //Rings - has > 0 belts + LuaState.DoString(@" + function hasBelts (ring_list) + return _hasRingsFiltered(ring_list, 'Belt') + end"); + //Bodies in system LuaState.DoString(@" function bodies (system_list) @@ -116,7 +166,7 @@ namespace Observatory.Explorer if (criteria[i].Trim().StartsWith("::")) { string scriptDescription = criteria[i].Trim().Replace("::", string.Empty); - if (scriptDescription.ToLower() == "criteria") + if (scriptDescription.ToLower() == "criteria" || scriptDescription.ToLower().StartsWith("criteria=")) { string functionName = $"Criteria{i}"; script.AppendLine($"function {functionName} (scan, parents, system, biosignals, geosignals)"); @@ -132,7 +182,7 @@ namespace Observatory.Explorer script.AppendLine("end"); LuaState.DoString(script.ToString()); - CriteriaFunctions.Add(LuaState[functionName] as LuaFunction); + CriteriaFunctions.Add(GetUniqueDescription(functionName, scriptDescription), LuaState[functionName] as LuaFunction); script.Clear(); } else if (scriptDescription.ToLower() == "global") @@ -154,22 +204,22 @@ namespace Observatory.Explorer script.AppendLine($"function {functionName} (scan, parents, system, biosignals, geosignals)"); script.AppendLine($" local result = {criteria[i]}"); + script.AppendLine(" local detail = ''"); if (criteria.Length > i + 1 && criteria[i + 1].Trim().ToLower() == "::detail::") { i++; i++; - script.AppendLine($" local detail = {criteria[i]}"); - } - else - { - script.AppendLine(" local detail = ''"); + // Gate detail evaluation on result to allow safe use of criteria-checked values in detail string. + script.AppendLine(" if result then"); + script.AppendLine($" detail = {criteria[i]}"); + script.AppendLine(" end"); } script.AppendLine($" return result, '{scriptDescription}', detail"); script.AppendLine("end"); LuaState.DoString(script.ToString()); - CriteriaFunctions.Add(LuaState[functionName] as LuaFunction); + CriteriaFunctions.Add(GetUniqueDescription(functionName, scriptDescription), LuaState[functionName] as LuaFunction); script.Clear(); } } @@ -240,7 +290,7 @@ namespace Observatory.Explorer try { - var result = criteriaFunction.Call(scan, parents, scanList, bioSignals, geoSignals); + var result = criteriaFunction.Value.Call(scan, parents, scanList, bioSignals, geoSignals); if (result.Length > 0 && ((bool?)result[0]).GetValueOrDefault(false)) { results.Add((result[1].ToString(), result[2].ToString(), false)); @@ -252,9 +302,9 @@ namespace Observatory.Explorer results.Add((e.Message, scan.Json, false)); StringBuilder errorDetail = new(); - errorDetail.AppendLine("while processing a custom criteria on scan:") + errorDetail.AppendLine($"while processing custom criteria '{criteriaFunction.Key}' on scan:") .AppendLine(scan.Json) - .AppendLine("NOTE: Custom criteria process has been disabled to prevent further errors."); + .AppendLine("NOTE: Custom criteria processing has been disabled to prevent further errors."); ErrorLogger(e, errorDetail.ToString()); break; } @@ -269,6 +319,16 @@ namespace Observatory.Explorer return results; } + private string GetUniqueDescription(string functionName, string scriptDescription) + { + string uniqueDescription = functionName; + if (scriptDescription.ToLower().StartsWith("criteria=")) + { + uniqueDescription += scriptDescription.Replace("criteria=", "=", StringComparison.CurrentCultureIgnoreCase); + } + return uniqueDescription; + } + private void LuaGC() { LuaState?.DoString("collectgarbage()");