| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649 |
- using System.Diagnostics;
- using System.Reflection;
- using System.Runtime.InteropServices;
- using UICatalog;
- using UnitTests;
- using Xunit.Abstractions;
- namespace IntegrationTests.UICatalog;
- public class ScenarioTests : TestsAllViews
- {
- public ScenarioTests (ITestOutputHelper output)
- {
- #if DEBUG_IDISPOSABLE
- View.EnableDebugIDisposableAsserts = true;
- View.Instances.Clear ();
- #endif
- _output = output;
- }
- private readonly ITestOutputHelper _output;
- /// <summary>
- /// <para>This runs through all Scenarios defined in UI Catalog, calling Init, Setup, and Run.</para>
- /// <para>Should find any Scenarios which crash on load or do not respond to <see cref="Application.RequestStop()"/>.</para>
- /// </summary>
- [Theory]
- [MemberData (nameof (AllScenarioTypes))]
- public void All_Scenarios_Quit_And_Init_Shutdown_Properly (Type scenarioType)
- {
- // Disable on Mac due to random failures related to timing issues
- if (RuntimeInformation.IsOSPlatform (OSPlatform.OSX))
- {
- _output.WriteLine ($"Skipping Scenario '{scenarioType}' on macOS due to random timeout failures.");
- return;
- }
- // Force a complete reset
- ApplicationImpl.SetInstance (null);
- CM.Disable (true);
- _output.WriteLine ($"Running Scenario '{scenarioType}'");
- Scenario? scenario = null;
- var scenarioName = string.Empty;
- // Do not use Application.AddTimer for out-of-band watchdogs as
- // they will be stopped by Shutdown/ResetState.
- Timer? watchdogTimer = null;
- var timeoutFired = false;
- // Increase timeout for macOS - it's consistently slower
- uint abortTime = 5000;
- if (RuntimeInformation.IsOSPlatform (OSPlatform.OSX))
- {
- abortTime = 10000;
- }
- var initialized = false;
- var shutdownGracefully = false;
- var iterationCount = 0;
- Key quitKey = Application.QuitKey;
- // Track if we've already unsubscribed to prevent double-removal
- var iterationHandlerRemoved = false;
- try
- {
- scenario = Activator.CreateInstance (scenarioType) as Scenario;
- scenarioName = scenario!.GetName ();
- Application.InitializedChanged += OnApplicationOnInitializedChanged;
- Application.ForceDriver = "FakeDriver";
- scenario!.Main ();
- Application.ForceDriver = string.Empty;
- }
- finally
- {
- // Ensure cleanup happens regardless of how we exit
- Application.InitializedChanged -= OnApplicationOnInitializedChanged;
- // Remove iteration handler if it wasn't removed
- if (!iterationHandlerRemoved)
- {
- Application.Iteration -= OnApplicationOnIteration;
- iterationHandlerRemoved = true;
- }
- watchdogTimer?.Dispose ();
- scenario?.Dispose ();
- scenario = null;
- ConfigurationManager.Disable (true);
- }
- Assert.True (initialized, $"Scenario '{scenarioName}' failed to initialize.");
- if (timeoutFired)
- {
- _output.WriteLine ($"WARNING: Scenario '{scenarioName}' timed out after {abortTime}ms. This may indicate a performance issue on this runner.");
- }
- Assert.True (
- shutdownGracefully,
- $"Scenario '{scenarioName}' failed to quit with {quitKey} after {abortTime}ms and {iterationCount} iterations. "
- + $"TimeoutFired={timeoutFired}");
- #if DEBUG_IDISPOSABLE
- Assert.Empty (View.Instances);
- #endif
- return;
- void OnApplicationOnInitializedChanged (object? s, EventArgs<bool> a)
- {
- if (a.Value)
- {
- Application.Iteration += OnApplicationOnIteration;
- initialized = true;
- // Use a System.Threading.Timer for the watchdog to ensure it's not affected by Application.StopAllTimers
- watchdogTimer = new Timer (_ => ForceCloseCallback (), null, (int)abortTime, System.Threading.Timeout.Infinite);
- }
- else
- {
- shutdownGracefully = true;
- }
- _output.WriteLine ($"Initialized == {a.Value}; shutdownGracefully == {shutdownGracefully}.");
- }
- // If the scenario doesn't close within abortTime ms, this will force it to quit
- void ForceCloseCallback ()
- {
- timeoutFired = true;
- _output.WriteLine ($"TIMEOUT FIRED for {scenarioName} after {abortTime}ms. Attempting graceful shutdown.");
- // Don't call ResetState here - let the finally block handle cleanup
- // Just try to stop the application gracefully
- try
- {
- if (Application.Initialized)
- {
- Application.RequestStop ();
- }
- }
- catch (Exception ex)
- {
- _output.WriteLine ($"Exception during timeout callback: {ex.Message}");
- }
- }
- void OnApplicationOnIteration (object? s, IterationEventArgs a)
- {
- iterationCount++;
- if (Application.Initialized)
- {
- // Press QuitKey
- quitKey = Application.QuitKey;
- _output.WriteLine ($"Attempting to quit with {quitKey} after {iterationCount} iterations.");
- try
- {
- Application.RaiseKeyDownEvent (quitKey);
- }
- catch (Exception ex)
- {
- _output.WriteLine ($"Exception raising quit key: {ex.Message}");
- }
- Application.Iteration -= OnApplicationOnIteration;
- iterationHandlerRemoved = true;
- }
- }
- }
- public static IEnumerable<object []> AllScenarioTypes =>
- typeof (Scenario).Assembly
- .GetTypes ()
- .Where (type => type.IsClass && !type.IsAbstract && type.IsSubclassOf (typeof (Scenario)))
- .Select (type => new object [] { type });
- [Fact]
- public void Run_All_Views_Tester_Scenario ()
- {
- // Disable any UIConfig settings
- ConfigurationManager.Disable (true);
- View? curView = null;
- // Settings
- var xVal = 0;
- var yVal = 0;
- var wVal = 0;
- var hVal = 0;
- List<string> posNames = ["Percent", "AnchorEnd", "Center", "Absolute"];
- List<string> dimNames = ["Auto", "Percent", "Fill", "Absolute"];
- Application.Init ("fake");
- var top = new Toplevel ();
- Dictionary<string, Type> viewClasses = GetAllViewClasses ().ToDictionary (t => t.Name);
- Window leftPane = new ()
- {
- Title = "Classes",
- X = 0,
- Y = 0,
- Width = 15,
- Height = Dim.Fill (1), // for status bar
- CanFocus = false,
- SchemeName = "TopLevel"
- };
- ListView classListView = new ()
- {
- X = 0,
- Y = 0,
- Width = Dim.Fill (),
- Height = Dim.Fill (),
- AllowsMarking = false,
- SchemeName = "TopLevel",
- Source = new ListWrapper<string> (new (viewClasses.Keys.ToList ()))
- };
- leftPane.Add (classListView);
- FrameView settingsPane = new ()
- {
- X = Pos.Right (leftPane),
- Y = 0, // for menu
- Width = Dim.Fill (),
- Height = 10,
- CanFocus = false,
- SchemeName = "TopLevel",
- Title = "Settings"
- };
- var radioItems = new [] { "Percent(x)", "AnchorEnd(x)", "Center", "Absolute(x)" };
- FrameView locationFrame = new ()
- {
- X = 0,
- Y = 0,
- Height = 3 + radioItems.Length,
- Width = 36,
- Title = "Location (Pos)"
- };
- settingsPane.Add (locationFrame);
- var label = new Label { X = 0, Y = 0, Text = "x:" };
- locationFrame.Add (label);
- OptionSelector xOptionSelector = new () { X = 0, Y = Pos.Bottom (label), Labels = radioItems };
- TextField xText = new () { X = Pos.Right (label) + 1, Y = 0, Width = 4, Text = $"{xVal}" };
- locationFrame.Add (xText);
- locationFrame.Add (xOptionSelector);
- radioItems = new [] { "Percent(y)", "AnchorEnd(y)", "Center", "Absolute(y)" };
- label = new () { X = Pos.Right (xOptionSelector) + 1, Y = 0, Text = "y:" };
- locationFrame.Add (label);
- TextField yText = new () { X = Pos.Right (label) + 1, Y = 0, Width = 4, Text = $"{yVal}" };
- locationFrame.Add (yText);
- OptionSelector yOptionSelector = new () { X = Pos.X (label), Y = Pos.Bottom (label), Labels = radioItems };
- locationFrame.Add (yOptionSelector);
- FrameView sizeFrame = new ()
- {
- X = Pos.Right (locationFrame),
- Y = Pos.Y (locationFrame),
- Height = 3 + radioItems.Length,
- Width = 40,
- Title = "Size (Dim)"
- };
- radioItems = new [] { "Auto()", "Percent(width)", "Fill(width)", "Absolute(width)" };
- label = new () { X = 0, Y = 0, Text = "width:" };
- sizeFrame.Add (label);
- OptionSelector wOptionSelector = new () { X = 0, Y = Pos.Bottom (label), Labels = radioItems };
- TextField wText = new () { X = Pos.Right (label) + 1, Y = 0, Width = 4, Text = $"{wVal}" };
- sizeFrame.Add (wText);
- sizeFrame.Add (wOptionSelector);
- radioItems = new [] { "Auto()", "Percent(height)", "Fill(height)", "Absolute(height)" };
- label = new () { X = Pos.Right (wOptionSelector) + 1, Y = 0, Text = "height:" };
- sizeFrame.Add (label);
- TextField hText = new () { X = Pos.Right (label) + 1, Y = 0, Width = 4, Text = $"{hVal}" };
- sizeFrame.Add (hText);
- OptionSelector hOptionSelector = new () { X = Pos.X (label), Y = Pos.Bottom (label), Labels = radioItems };
- sizeFrame.Add (hOptionSelector);
- settingsPane.Add (sizeFrame);
- FrameView hostPane = new ()
- {
- X = Pos.Right (leftPane),
- Y = Pos.Bottom (settingsPane),
- Width = Dim.Fill (),
- Height = Dim.Fill (1), // + 1 for status bar
- SchemeName = "Dialog"
- };
- classListView.OpenSelectedItem += (s, a) => { settingsPane.SetFocus (); };
- classListView.SelectedItemChanged += (s, args) =>
- {
- // Remove existing class, if any
- if (curView is { })
- {
- curView.SubViewsLaidOut -= LayoutCompleteHandler;
- hostPane.Remove (curView);
- curView.Dispose ();
- curView = null;
- hostPane.FillRect (hostPane.Viewport);
- }
- curView = CreateClass (viewClasses.Values.ToArray () [classListView.SelectedItem!.Value]);
- };
- xOptionSelector.ValueChanged += (_, _) => DimPosChanged (curView);
- xText.TextChanged += (s, args) =>
- {
- try
- {
- xVal = int.Parse (xText.Text);
- DimPosChanged (curView);
- }
- catch
- { }
- };
- yText.TextChanged += (s, e) =>
- {
- try
- {
- yVal = int.Parse (yText.Text);
- DimPosChanged (curView);
- }
- catch
- { }
- };
- yOptionSelector.ValueChanged += (_, _) => DimPosChanged (curView);
- wOptionSelector.ValueChanged += (_, _) => DimPosChanged (curView);
- wText.TextChanged += (s, args) =>
- {
- try
- {
- wVal = int.Parse (wText.Text);
- DimPosChanged (curView);
- }
- catch
- { }
- };
- hText.TextChanged += (s, args) =>
- {
- try
- {
- hVal = int.Parse (hText.Text);
- DimPosChanged (curView);
- }
- catch
- { }
- };
- hOptionSelector.ValueChanged += (_, _) => DimPosChanged (curView);
- top.Add (leftPane, settingsPane, hostPane);
- top.LayoutSubViews ();
- curView = CreateClass (viewClasses.First ().Value);
- var iterations = 0;
- Application.Iteration += OnApplicationOnIteration;
- Application.Run (top);
- Application.Iteration -= OnApplicationOnIteration;
- Assert.Equal (viewClasses.Count, iterations);
- top.Dispose ();
- Application.Shutdown ();
- ConfigurationManager.Disable (true);
- return;
- void OnApplicationOnIteration (object? s, IterationEventArgs a)
- {
- iterations++;
- if (iterations < viewClasses.Count)
- {
- classListView.MoveDown ();
- if (curView is { })
- {
- Assert.Equal (
- curView.GetType ().Name,
- viewClasses.Values.ToArray () [classListView.SelectedItem!.Value].Name);
- }
- }
- else
- {
- Application.RequestStop ();
- }
- }
- void DimPosChanged (View? view)
- {
- if (view == null)
- {
- return;
- }
- try
- {
- switch (xOptionSelector.Value)
- {
- case 0:
- view.X = Pos.Percent (xVal);
- break;
- case 1:
- view.X = Pos.AnchorEnd (xVal);
- break;
- case 2:
- view.X = Pos.Center ();
- break;
- case 3:
- view.X = Pos.Absolute (xVal);
- break;
- }
- switch (yOptionSelector.Value)
- {
- case 0:
- view.Y = Pos.Percent (yVal);
- break;
- case 1:
- view.Y = Pos.AnchorEnd (yVal);
- break;
- case 2:
- view.Y = Pos.Center ();
- break;
- case 3:
- view.Y = Pos.Absolute (yVal);
- break;
- }
- switch (wOptionSelector.Value)
- {
- case 0:
- view.Width = Dim.Percent (wVal);
- break;
- case 1:
- view.Width = Dim.Fill (wVal);
- break;
- case 2:
- view.Width = Dim.Absolute (wVal);
- break;
- }
- switch (hOptionSelector.Value)
- {
- case 0:
- view.Height = Dim.Percent (hVal);
- break;
- case 1:
- view.Height = Dim.Fill (hVal);
- break;
- case 2:
- view.Height = Dim.Absolute (hVal);
- break;
- }
- }
- catch (Exception e)
- {
- MessageBox.ErrorQuery ("Exception", e.Message, "Ok");
- }
- UpdateTitle (view);
- }
- void UpdateSettings (View view)
- {
- var x = view.X.ToString ();
- var y = view.Y.ToString ();
- try
- {
- xOptionSelector.Value = posNames.IndexOf (posNames.First (s => x.Contains (s)));
- yOptionSelector.Value = posNames.IndexOf (posNames.First (s => y.Contains (s)));
- }
- catch (InvalidOperationException e)
- {
- // This is a hack to work around the fact that the Pos enum doesn't have an "Align" value yet
- Debug.WriteLine ($"{e}");
- }
- xText.Text = $"{view.Frame.X}";
- yText.Text = $"{view.Frame.Y}";
- var w = view.Width!.ToString ();
- var h = view.Height!.ToString ();
- wOptionSelector.Value = dimNames.IndexOf (dimNames.First (s => w.Contains (s)));
- hOptionSelector.Value = dimNames.IndexOf (dimNames.First (s => h.Contains (s)));
- wText.Text = $"{view.Frame.Width}";
- hText.Text = $"{view.Frame.Height}";
- }
- void UpdateTitle (View? view) { hostPane.Title = $"{view!.GetType ().Name} - {view.X}, {view.Y}, {view.Width}, {view.Height}"; }
- View? CreateClass (Type type)
- {
- // If we are to create a generic Type
- if (type.IsGenericType)
- {
- // For each of the <T> arguments
- List<Type> typeArguments = new ();
- // use <object> or the original type if applicable
- foreach (Type arg in type.GetGenericArguments ())
- {
- if (arg.IsValueType && Nullable.GetUnderlyingType (arg) == null)
- {
- typeArguments.Add (arg);
- }
- else
- {
- typeArguments.Add (typeof (object));
- }
- }
- // Ensure the type does not contain any generic parameters
- if (type.ContainsGenericParameters)
- {
- Logging.Warning ($"Cannot create an instance of {type} because it contains generic parameters.");
- //throw new ArgumentException ($"Cannot create an instance of {type} because it contains generic parameters.");
- return null;
- }
- // And change what type we are instantiating from MyClass<T> to MyClass<object>
- type = type.MakeGenericType (typeArguments.ToArray ());
- }
- // Instantiate view
- var view = Activator.CreateInstance (type) as View;
- if (view is null)
- {
- return null;
- }
- if (view.Width is not DimAuto)
- {
- view.Width = Dim.Percent (75);
- }
- if (view.Height is not DimAuto)
- {
- view.Height = Dim.Percent (75);
- }
- // Set the colorscheme to make it stand out if is null by default
- if (!view.HasScheme)
- {
- view.SchemeName = "Base";
- }
- // If the view supports a Text property, set it so we have something to look at
- if (view.GetType ().GetProperty ("Text") != null)
- {
- try
- {
- view.GetType ().GetProperty ("Text")?.GetSetMethod ()?.Invoke (view, new [] { "Test Text" });
- }
- catch (TargetInvocationException e)
- {
- MessageBox.ErrorQuery ("Exception", e.InnerException!.Message, "Ok");
- view = null;
- }
- }
- // If the view supports a Title property, set it so we have something to look at
- if (view != null && view.GetType ().GetProperty ("Title") != null)
- {
- if (view.GetType ().GetProperty ("Title")!.PropertyType == typeof (string))
- {
- view?.GetType ().GetProperty ("Title")?.GetSetMethod ()?.Invoke (view, new [] { "Test Title" });
- }
- else
- {
- view?.GetType ().GetProperty ("Title")?.GetSetMethod ()?.Invoke (view, new [] { "Test Title" });
- }
- }
- // If the view supports a Source property, set it so we have something to look at
- if (view != null
- && view.GetType ().GetProperty ("Source") != null
- && view.GetType ().GetProperty ("Source")!.PropertyType == typeof (IListDataSource))
- {
- ListWrapper<string> source = new (["Test Text #1", "Test Text #2", "Test Text #3"]);
- view?.GetType ().GetProperty ("Source")?.GetSetMethod ()?.Invoke (view, new [] { source });
- }
- // Add
- hostPane.Add (view);
- //DimPosChanged ();
- hostPane.LayoutSubViews ();
- hostPane.ClearViewport ();
- hostPane.SetNeedsDraw ();
- UpdateSettings (view!);
- UpdateTitle (view);
- view!.SubViewsLaidOut += LayoutCompleteHandler;
- return view;
- }
- void LayoutCompleteHandler (object? sender, LayoutEventArgs args) { UpdateTitle (curView); }
- }
- }
|