namespace ScadaLink.InboundAPI.Tests; /// /// InboundAPI-005 / InboundAPI-015: tests for the script-trust-model checker. /// /// InboundAPI-015 hardens the textual walker against reflection reached through /// permitted-type members that never spell a forbidden namespace, e.g. /// typeof(string).Assembly.GetType("System.IO.File"). /// public class ForbiddenApiCheckerTests { private static bool IsRejected(string script) => ForbiddenApiChecker.FindViolations(script).Count > 0; // --- Baseline: legitimate scripts must still pass --- [Theory] [InlineData("return 1 + 1;")] [InlineData("var list = new List { 1, 2, 3 }; return list.Sum();")] [InlineData("return Parameters.Get(\"x\") * 2;")] [InlineData("await Task.Delay(1); return null;")] [InlineData("var r = await Route.To(\"inst\").Call(\"s\"); return r;")] [InlineData("Action a = () => {}; a.Invoke(); return null;")] public void PermittedScript_NotRejected(string script) { Assert.False(IsRejected(script), script); } // --- Baseline: forbidden namespaces (textual) must still be rejected --- [Theory] [InlineData("System.IO.File.Delete(\"/tmp/x\"); return null;")] [InlineData("System.Diagnostics.Process.Start(\"/bin/sh\"); return null;")] [InlineData("using System.Reflection; return null;")] [InlineData("var s = new System.Net.Sockets.Socket(default, default, default); return null;")] public void ForbiddenNamespace_Rejected(string script) { Assert.True(IsRejected(script), script); } // --- InboundAPI-015: reflection reachable without a forbidden namespace token --- [Fact] public void Reflection_AssemblyPropertyAccess_Rejected() { // typeof(string).Assembly — .Assembly is a reflection gateway off a permitted type. Assert.True(IsRejected("var a = typeof(string).Assembly; return null;")); } [Fact] public void Reflection_AssemblyGetType_Rejected() { // The classic bypass: obtain System.IO.File as a Type via a string literal. Assert.True(IsRejected( "var t = typeof(string).Assembly.GetType(\"System.IO.File\"); return null;")); } [Fact] public void Reflection_ObjectGetType_Rejected() { // x.GetType() returns a System.Type — a reflection gateway. Assert.True(IsRejected("var t = \"\".GetType(); return null;")); } [Fact] public void Reflection_TypeGetTypeStatic_Rejected() { Assert.True(IsRejected("var t = Type.GetType(\"System.IO.File\"); return null;")); } [Fact] public void Reflection_ActivatorCreateInstance_Rejected() { Assert.True(IsRejected( "var o = Activator.CreateInstance(Type.GetType(\"System.IO.File\")); return null;")); } [Fact] public void Reflection_InvokeMember_Rejected() { Assert.True(IsRejected( "typeof(object).InvokeMember(\"x\", default, null, null, null); return null;")); } [Fact] public void Reflection_GetMethodInvoke_Rejected() { Assert.True(IsRejected( "var m = typeof(object).GetMethod(\"ToString\"); return null;")); } [Fact] public void Reflection_GetTypeInfo_Rejected() { Assert.True(IsRejected("var ti = \"\".GetType().GetTypeInfo(); return null;")); } [Fact] public void DynamicKeyword_Rejected() { // dynamic widens late-bound member access the static walker cannot see through. Assert.True(IsRejected("dynamic d = Parameters; return null;")); } }