From 27c17d5bb8c8c1489ea287cfabe22359fd574c3c Mon Sep 17 00:00:00 2001 From: Kuniwak Date: Thu, 9 Nov 2023 06:40:00 +0900 Subject: [PATCH] Make IReporter injectable --- Editor/Localization/ja.po | 22 +- Editor/UI/Reporters.meta | 3 + .../UI/Reporters/CompositeReporterEditor.cs | 61 +++++ .../Reporters/CompositeReporterEditor.cs.meta | 3 + Editor/UI/Reporters/SlackReporterEditor.cs | 81 ++++++ .../UI/Reporters/SlackReporterEditor.cs.meta | 3 + Editor/UI/Settings/AutopilotSettingsEditor.cs | 15 +- Runtime/Autopilot.cs | 15 +- .../{IReporter.cs => AbstractReporter.cs} | 17 +- ...orter.cs.meta => AbstractReporter.cs.meta} | 0 Runtime/Reporters/CompositeReporter.cs | 44 ++++ Runtime/Reporters/CompositeReporter.cs.meta | 3 + Runtime/Reporters/SlackMessageSender.cs | 154 +++++++++++ Runtime/Reporters/SlackMessageSender.cs.meta | 3 + Runtime/Reporters/SlackReporter.cs | 137 ++++------ Runtime/Settings/AutopilotSettings.cs | 14 + Runtime/Utilities/LogMessageHandler.cs | 10 +- Tests/Editor/DeNA.Anjin.Editor.Tests.asmdef | 3 +- Tests/Runtime/Reporters.meta | 3 + .../Reporters/SlackMessageSenderTest.cs | 182 +++++++++++++ .../Reporters/SlackMessageSenderTest.cs.meta | 3 + Tests/Runtime/TestDoubles/SpyReporter.cs | 42 +++ Tests/Runtime/TestDoubles/SpyReporter.cs.meta | 3 + ...owingErrorFromNotMainThreadOnClick.cs.meta | 2 +- .../Utilities/LogMessageHandlerTest.cs | 249 +++--------------- ...lotSettingsForImmediatlyFacingErrors.asset | 32 +++ ...ttingsForImmediatlyFacingErrors.asset.meta | 8 + Tests/TestAssets/DoNothingAgentForTests.asset | 2 +- 28 files changed, 786 insertions(+), 328 deletions(-) create mode 100644 Editor/UI/Reporters.meta create mode 100644 Editor/UI/Reporters/CompositeReporterEditor.cs create mode 100644 Editor/UI/Reporters/CompositeReporterEditor.cs.meta create mode 100644 Editor/UI/Reporters/SlackReporterEditor.cs create mode 100644 Editor/UI/Reporters/SlackReporterEditor.cs.meta rename Runtime/Reporters/{IReporter.cs => AbstractReporter.cs} (52%) rename Runtime/Reporters/{IReporter.cs.meta => AbstractReporter.cs.meta} (100%) create mode 100644 Runtime/Reporters/CompositeReporter.cs create mode 100644 Runtime/Reporters/CompositeReporter.cs.meta create mode 100644 Runtime/Reporters/SlackMessageSender.cs create mode 100644 Runtime/Reporters/SlackMessageSender.cs.meta create mode 100644 Tests/Runtime/Reporters.meta create mode 100644 Tests/Runtime/Reporters/SlackMessageSenderTest.cs create mode 100644 Tests/Runtime/Reporters/SlackMessageSenderTest.cs.meta create mode 100644 Tests/Runtime/TestDoubles/SpyReporter.cs create mode 100644 Tests/Runtime/TestDoubles/SpyReporter.cs.meta create mode 100644 Tests/TestAssets/AutopilotSettingsForImmediatlyFacingErrors.asset create mode 100644 Tests/TestAssets/AutopilotSettingsForImmediatlyFacingErrors.asset.meta diff --git a/Editor/Localization/ja.po b/Editor/Localization/ja.po index 75233f7..6daa19a 100644 --- a/Editor/Localization/ja.po +++ b/Editor/Localization/ja.po @@ -87,6 +87,18 @@ msgstr "JUnitレポート出力パス" msgid "JUnit report output path" msgstr "JUnit形式のレポートファイル出力パス(省略時は出力されない)" +# reporter +msgid "Reporter" +msgstr "レポータ" + +# reporter tooltip +msgid "Reporter that called when some errors occurred in target application" +msgstr "対象のアプリケーションで発生したエラーを通知するレポータ" + +# obsolete slack settings +msgid "Slack settings will be moved to SlackReporter" +msgstr "Slack 設定は SlackReporter へ移動しました" + # slackToken msgid "Slack Token" msgstr "Slackトークン" @@ -108,11 +120,11 @@ msgid "Slack Mention Settings" msgstr "Slackメンション設定" # mentionSubTeamIDs -msgid "Mention Sub Team IDs" +msgid "Sub Team IDs to Mention" msgstr "メンション宛先" # mentionSubTeamIDs tooltip -msgid "Mention to sub team ID (comma separates)" +msgid "Sub team IDs to mention (comma separates)" msgstr "Slack通知メッセージでメンションするチームのIDをカンマ区切りで指定します" # addHereInSlackMessage @@ -120,7 +132,7 @@ msgid "Add @here Into Slack Message" msgstr "@hereをメッセージにつける" # addHereInSlackMessage tooltip -msgid "Add @here into Slack message" +msgid "Whether adding @here into Slack messages or not" msgstr "Slack通知メッセージに@hereを付けます" # Header: Error Handling Settings @@ -320,3 +332,7 @@ msgstr "操作を記録したJSONファイル" # recordedJson tooltip msgid "JSON file recorded by AutomatedQA package" msgstr "Automated QAパッケージのRecorded Playbackウィンドウで記録したjsonファイルを設定します" + +# composite reporters +msgid "Reporters" +msgstr "レポータ" diff --git a/Editor/UI/Reporters.meta b/Editor/UI/Reporters.meta new file mode 100644 index 0000000..82538c7 --- /dev/null +++ b/Editor/UI/Reporters.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: 3c549ea2f5314b23a625a59b05a24220 +timeCreated: 1699484044 \ No newline at end of file diff --git a/Editor/UI/Reporters/CompositeReporterEditor.cs b/Editor/UI/Reporters/CompositeReporterEditor.cs new file mode 100644 index 0000000..0fe7f5f --- /dev/null +++ b/Editor/UI/Reporters/CompositeReporterEditor.cs @@ -0,0 +1,61 @@ +// Copyright (c) 2023 DeNA Co., Ltd. +// This software is released under the MIT License. + +using DeNA.Anjin.Reporters; +using UnityEditor; +using UnityEditorInternal; +using UnityEngine; + +namespace DeNA.Anjin.Editor.UI.Reporters +{ + /// + /// Editor for + /// + [CustomEditor(typeof(CompositeReporter))] + public class CompositeReporterEditor : UnityEditor.Editor + { + private SerializedProperty _reportersProp; + private ReorderableList _reorderableList; + private static readonly string s_reporters = L10n.Tr("Reporters"); + private GUIContent _reportersGUIContent; + private static readonly string s_reporter = L10n.Tr("Reporter"); + private GUIContent _reporterGUIContent; + + + private void OnEnable() + { + Initialize(); + } + + + private void Initialize() + { + _reportersProp = serializedObject.FindProperty(nameof(CompositeReporter.reporters)); + _reportersGUIContent = new GUIContent(s_reporters); + _reporterGUIContent = new GUIContent(s_reporter); + _reorderableList = new ReorderableList(serializedObject, _reportersProp) + { + drawHeaderCallback = rect => EditorGUI.LabelField(rect, _reportersGUIContent), + elementHeightCallback = _ => EditorGUIUtility.singleLineHeight, + // XXX: Dont use discarded parameter to treat Unity 2019.x + // ReSharper disable UnusedParameter.Local + drawElementCallback = (rect, index, active, focused) => + // ReSharper restore UnusedParameter.Local + { + var elemProp = _reportersProp.GetArrayElementAtIndex(index); + EditorGUI.PropertyField(rect, elemProp, _reporterGUIContent); + } + }; + } + + + public override void OnInspectorGUI() + { + serializedObject.Update(); + + _reorderableList.DoLayoutList(); + + serializedObject.ApplyModifiedProperties(); + } + } +} diff --git a/Editor/UI/Reporters/CompositeReporterEditor.cs.meta b/Editor/UI/Reporters/CompositeReporterEditor.cs.meta new file mode 100644 index 0000000..e6783ba --- /dev/null +++ b/Editor/UI/Reporters/CompositeReporterEditor.cs.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: ccf304a0e83345eba515322942248039 +timeCreated: 1699484060 \ No newline at end of file diff --git a/Editor/UI/Reporters/SlackReporterEditor.cs b/Editor/UI/Reporters/SlackReporterEditor.cs new file mode 100644 index 0000000..292b382 --- /dev/null +++ b/Editor/UI/Reporters/SlackReporterEditor.cs @@ -0,0 +1,81 @@ +// Copyright (c) 2023 DeNA Co., Ltd. +// This software is released under the MIT License. + +using DeNA.Anjin.Reporters; +using UnityEditor; +using UnityEngine; + +namespace DeNA.Anjin.Editor.UI.Reporters +{ + /// + /// Editor for + /// + [CustomEditor(typeof(SlackReporter))] + public class SlackReporterEditor : UnityEditor.Editor + { + private const float SpacerPixels = 10f; + private const float SpacerPixelsUnderHeader = 4f; + + private static readonly string s_slackToken = L10n.Tr("Slack Token"); + private static readonly string s_slackTokenTooltip = L10n.Tr("Slack API token"); + private SerializedProperty _slackTokenProp; + private GUIContent _slackTokenGUIContent; + + private static readonly string s_slackChannels = L10n.Tr("Slack Channels"); + private static readonly string s_slackChannelsTooltip = L10n.Tr("Slack channels to send notification"); + private SerializedProperty _slackChannelsProp; + private GUIContent _slackChannelsGUIContent; + + private static readonly string s_slackMentionSettingsHeader = L10n.Tr("Slack Mention Settings"); + + private static readonly string s_mentionSubTeamIDs = L10n.Tr("Sub Team IDs to Mention"); + private static readonly string s_mentionSubTeamIDsTooltip = L10n.Tr("Sub team IDs to mention (comma separates)"); + private SerializedProperty _mentionSubTeamIDsProp; + private GUIContent _mentionSubTeamIDsGUIContent; + + private static readonly string s_addHereInSlackMessage = L10n.Tr("Add @here Into Slack Message"); + private static readonly string s_addHereInSlackMessageTooltip = L10n.Tr("Whether adding @here into Slack messages or not"); + private SerializedProperty _addHereInSlackMessageProp; + private GUIContent _addHereInSlackMessageGUIContent; + + + private void OnEnable() + { + Initialize(); + } + + + private void Initialize() + { + _slackTokenProp = serializedObject.FindProperty(nameof(SlackReporter.slackToken)); + _slackTokenGUIContent = new GUIContent(s_slackToken, s_slackTokenTooltip); + + _slackChannelsProp = serializedObject.FindProperty(nameof(SlackReporter.slackChannels)); + _slackChannelsGUIContent = new GUIContent(s_slackChannels, s_slackChannelsTooltip); + + _mentionSubTeamIDsProp = serializedObject.FindProperty(nameof(SlackReporter.mentionSubTeamIDs)); + _mentionSubTeamIDsGUIContent = new GUIContent(s_mentionSubTeamIDs, s_mentionSubTeamIDsTooltip); + + _addHereInSlackMessageProp = serializedObject.FindProperty(nameof(SlackReporter.addHereInSlackMessage)); + _addHereInSlackMessageGUIContent = new GUIContent(s_addHereInSlackMessage, s_addHereInSlackMessageTooltip); + } + + + public override void OnInspectorGUI() + { + serializedObject.Update(); + + EditorGUILayout.PropertyField(_slackTokenProp, _slackTokenGUIContent); + EditorGUILayout.PropertyField(_slackChannelsProp, _slackChannelsGUIContent); + + GUILayout.Space(SpacerPixels); + GUILayout.Label(s_slackMentionSettingsHeader); + GUILayout.Space(SpacerPixelsUnderHeader); + + EditorGUILayout.PropertyField(_mentionSubTeamIDsProp, _mentionSubTeamIDsGUIContent); + EditorGUILayout.PropertyField(_addHereInSlackMessageProp, _addHereInSlackMessageGUIContent); + + serializedObject.ApplyModifiedProperties(); + } + } +} diff --git a/Editor/UI/Reporters/SlackReporterEditor.cs.meta b/Editor/UI/Reporters/SlackReporterEditor.cs.meta new file mode 100644 index 0000000..d061868 --- /dev/null +++ b/Editor/UI/Reporters/SlackReporterEditor.cs.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: 99b990767b4d4abe95b829b1d62b7698 +timeCreated: 1699484320 \ No newline at end of file diff --git a/Editor/UI/Settings/AutopilotSettingsEditor.cs b/Editor/UI/Settings/AutopilotSettingsEditor.cs index 86a127a..abd5f44 100644 --- a/Editor/UI/Settings/AutopilotSettingsEditor.cs +++ b/Editor/UI/Settings/AutopilotSettingsEditor.cs @@ -42,16 +42,19 @@ public class AutopilotSettingsEditor : UnityEditor.Editor private static readonly string s_junitReportPath = L10n.Tr("JUnit Report Path"); private static readonly string s_junitReportPathTooltip = L10n.Tr("JUnit report output path"); + private static readonly string s_reporter = L10n.Tr("Reporter"); + private static readonly string s_reporterTooltip = L10n.Tr("Reporter that called when some errors occurred in target application"); + private static readonly string s_slackToken = L10n.Tr("Slack Token"); private static readonly string s_slackTokenTooltip = L10n.Tr("Slack API token"); private static readonly string s_slackChannels = L10n.Tr("Slack Channels"); private static readonly string s_slackChannelsTooltip = L10n.Tr("Slack channels to send notification"); private static readonly string s_slackMentionSettingsHeader = L10n.Tr("Slack Mention Settings"); - private static readonly string s_mentionSubTeamIDs = L10n.Tr("Mention Sub Team IDs"); - private static readonly string s_mentionSubTeamIDsTooltip = L10n.Tr("Mention to sub team ID (comma separates)"); + private static readonly string s_mentionSubTeamIDs = L10n.Tr("Sub Team IDs to Mention"); + private static readonly string s_mentionSubTeamIDsTooltip = L10n.Tr("Sub team IDs to mention (comma separates)"); private static readonly string s_addHereInSlackMessage = L10n.Tr("Add @here Into Slack Message"); - private static readonly string s_addHereInSlackMessageTooltip = L10n.Tr("Add @here into Slack message"); + private static readonly string s_addHereInSlackMessageTooltip = L10n.Tr("Whether adding @here into Slack messages or not"); private static readonly string s_errorHandlingSettingsHeader = L10n.Tr("Error Handling Settings"); private static readonly string s_handleException = L10n.Tr("Handle Exception"); @@ -64,6 +67,9 @@ public class AutopilotSettingsEditor : UnityEditor.Editor private static readonly string s_handleWarningTooltip = L10n.Tr("Notify when Warning detected in log"); private static readonly string s_ignoreMessages = L10n.Tr("Ignore Messages"); + private static readonly string s_obsoletedSlackParamsHelpBox = + L10n.Tr("Slack settings will be moved to SlackReporter"); + private static readonly string s_ignoreMessagesTooltip = L10n.Tr("Do not send notifications when log messages contain this string"); @@ -105,6 +111,9 @@ public override void OnInspectorGUI() new GUIContent(s_timeScale, s_timeScaleTooltip)); EditorGUILayout.PropertyField(serializedObject.FindProperty(nameof(AutopilotSettings.junitReportPath)), new GUIContent(s_junitReportPath, s_junitReportPathTooltip)); + EditorGUILayout.PropertyField(serializedObject.FindProperty(nameof(AutopilotSettings.reporter)), + new GUIContent(s_reporter, s_reporterTooltip)); + EditorGUILayout.HelpBox(s_obsoletedSlackParamsHelpBox, MessageType.Warning); EditorGUILayout.PropertyField(serializedObject.FindProperty(nameof(AutopilotSettings.slackToken)), new GUIContent(s_slackToken, s_slackTokenTooltip)); EditorGUILayout.PropertyField(serializedObject.FindProperty(nameof(AutopilotSettings.slackChannels)), diff --git a/Runtime/Autopilot.cs b/Runtime/Autopilot.cs index 84a99c8..f6d931e 100644 --- a/Runtime/Autopilot.cs +++ b/Runtime/Autopilot.cs @@ -43,7 +43,6 @@ public enum ExitCode private ILogger _logger; private RandomFactory _randomFactory; private IAgentDispatcher _dispatcher; - private IReporter _reporter; private LogMessageHandler _logMessageHandler; private AutopilotState _state; private AutopilotSettings _settings; @@ -55,7 +54,7 @@ private void Start() _settings = _state.settings; Assert.IsNotNull(_settings); - _logger = new ConsoleLogger(Debug.unityLogger.logHandler); + _logger = CreateLogger(); if (!int.TryParse(_settings.randomSeed, out var seed)) { @@ -68,8 +67,7 @@ private void Start() // NOTE: Registering logMessageReceived must be placed before DispatchByScene. // Because some agent can throw an error immediately, so reporter can miss the error if // registering logMessageReceived is placed after DispatchByScene. - _reporter = new SlackReporter(_settings, new SlackAPI()); - _logMessageHandler = new LogMessageHandler(_settings, _reporter); + _logMessageHandler = new LogMessageHandler(_settings, _settings.reporter); Application.logMessageReceivedThreaded += _logMessageHandler.HandleLog; _dispatcher = new AgentDispatcher(_settings, _logger, _randomFactory); @@ -89,6 +87,15 @@ private void Start() _startTime = Time.realtimeSinceStartup; } + /// + /// Returns an agent dispatcher that autopilot uses. You can change a logger by overriding this method + /// + /// A new logger + protected virtual ILogger CreateLogger() + { + return new ConsoleLogger(Debug.unityLogger.logHandler); + } + /// /// Terminate when ran specified time. /// diff --git a/Runtime/Reporters/IReporter.cs b/Runtime/Reporters/AbstractReporter.cs similarity index 52% rename from Runtime/Reporters/IReporter.cs rename to Runtime/Reporters/AbstractReporter.cs index 7f85a8a..c40db21 100644 --- a/Runtime/Reporters/IReporter.cs +++ b/Runtime/Reporters/AbstractReporter.cs @@ -1,24 +1,35 @@ // Copyright (c) 2023 DeNA Co., Ltd. // This software is released under the MIT License. +using System.Threading; using Cysharp.Threading.Tasks; +using DeNA.Anjin.Settings; using UnityEngine; namespace DeNA.Anjin.Reporters { /// - /// Reporter interface + /// Reporter base class /// - public interface IReporter + public abstract class AbstractReporter : ScriptableObject { /// /// Post report log message, stacktrace and screenshot /// + /// Autopilot settings /// Log message /// Stack trace /// Log message type /// With screenshot + /// Cancellation token /// - UniTask PostReportAsync(string logString, string stackTrace, LogType type, bool withScreenshot); + public abstract UniTask PostReportAsync( + AutopilotSettings settings, + string logString, + string stackTrace, + LogType type, + bool withScreenshot, + CancellationToken cancellationToken = default + ); } } diff --git a/Runtime/Reporters/IReporter.cs.meta b/Runtime/Reporters/AbstractReporter.cs.meta similarity index 100% rename from Runtime/Reporters/IReporter.cs.meta rename to Runtime/Reporters/AbstractReporter.cs.meta diff --git a/Runtime/Reporters/CompositeReporter.cs b/Runtime/Reporters/CompositeReporter.cs new file mode 100644 index 0000000..e9fea76 --- /dev/null +++ b/Runtime/Reporters/CompositeReporter.cs @@ -0,0 +1,44 @@ +// Copyright (c) 2023 DeNA Co., Ltd. +// This software is released under the MIT License. + +using System.Collections.Generic; +using System.Linq; +using System.Threading; +using Cysharp.Threading.Tasks; +using DeNA.Anjin.Settings; +using UnityEngine; + +namespace DeNA.Anjin.Reporters +{ + /// + /// A class for reporters that delegate to multiple reporters + /// + [CreateAssetMenu(fileName = "New CompositeReporter", menuName = "Anjin/Composite Reporter", order = 51)] + public class CompositeReporter : AbstractReporter + { + /// + /// Reporters to delegate + /// + public List reporters = new List(); + + + /// + public override async UniTask PostReportAsync( + AutopilotSettings settings, + string logString, + string stackTrace, + LogType type, + bool withScreenshot, + CancellationToken cancellationToken = default + ) + { + await UniTask.WhenAll( + reporters + .Where(r => r != this && r != null) + .Select( + r => r.PostReportAsync(settings, logString, stackTrace, type, withScreenshot, cancellationToken) + ) + ); + } + } +} diff --git a/Runtime/Reporters/CompositeReporter.cs.meta b/Runtime/Reporters/CompositeReporter.cs.meta new file mode 100644 index 0000000..7aeb2c1 --- /dev/null +++ b/Runtime/Reporters/CompositeReporter.cs.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: 107581b5731b4e6d824ae77242f55dd3 +timeCreated: 1699480101 \ No newline at end of file diff --git a/Runtime/Reporters/SlackMessageSender.cs b/Runtime/Reporters/SlackMessageSender.cs new file mode 100644 index 0000000..c97eb7f --- /dev/null +++ b/Runtime/Reporters/SlackMessageSender.cs @@ -0,0 +1,154 @@ +// Copyright (c) 2023 DeNA Co., Ltd. +// This software is released under the MIT License. + +using System.Collections.Generic; +using System.Text; +using System.Threading; +using Cysharp.Threading.Tasks; +using UnityEngine; + +namespace DeNA.Anjin.Reporters +{ + /// + /// An interface for Slack message senders. The derived class of this interface must define message format, and + /// delegate touching Slack API to . + /// Purpose of introducing this interface is making be a humble object. + /// + public interface ISlackMessageSender + { + /// + /// Post report log message, stacktrace and screenshot + /// + /// Slack API token + /// Slack Channel to send notification + /// Sub team IDs to mention + /// Whether adding @here or not + /// Log message + /// Stack trace + /// With screenshot + /// Cancellation token + /// + UniTask Send( + string slackToken, + string slackChannel, + IEnumerable mentionSubTeamIDs, + bool addHereInSlackMessage, + string logString, + string stackTrace, + bool withScreenshot, + CancellationToken cancellationToken = default + ); + } + + /// + /// A class for Slack message senders. This class defines a message format and delegates touching Slack API to + /// . + /// + public class SlackMessageSender : ISlackMessageSender + { + private readonly SlackAPI _slackAPI; + private readonly string _slackToken; + + + /// + /// Creates a new instance for . + /// + /// Slack API client + public SlackMessageSender(SlackAPI api) + { + _slackAPI = api; + } + + + /// + public async UniTask Send( + string slackToken, + string slackChannel, + IEnumerable mentionSubTeamIDs, + bool addHereInSlackMessage, + string logString, + string stackTrace, + bool withScreenshot, + CancellationToken cancellationToken = default + ) + { + if (string.IsNullOrEmpty(slackToken) || string.IsNullOrEmpty(slackChannel)) + { + return; + } + + var title = Title(logString, mentionSubTeamIDs, addHereInSlackMessage); + + await UniTask.SwitchToMainThread(); + + var postTitleTask = await _slackAPI.Post( + slackToken, + slackChannel, + title, + cancellationToken: cancellationToken + ); + if (!postTitleTask.Success) + { + return; + } + + if (withScreenshot && !Application.isBatchMode) + { + var coroutineRunner = new GameObject().AddComponent(); + await UniTask.WaitForEndOfFrame(coroutineRunner); + Object.Destroy(coroutineRunner); + + var screenshot = ScreenCapture.CaptureScreenshotAsTexture(); + var withoutAlpha = new Texture2D(screenshot.width, screenshot.height, TextureFormat.RGB24, false); + withoutAlpha.SetPixels(screenshot.GetPixels()); + withoutAlpha.Apply(); + + var postScreenshotTask = await _slackAPI.Post( + slackToken, + slackChannel, + withoutAlpha.EncodeToPNG(), + postTitleTask.Ts + ); + if (!postScreenshotTask.Success) + { + return; + } + } + + var body = Body(logString, stackTrace); + await _slackAPI.Post(slackToken, slackChannel, body, postTitleTask.Ts); + } + + + private static string Title(string logString, IEnumerable mentionSubTeamIDs, bool withHere) + { + var sb = new StringBuilder(); + foreach (var s in mentionSubTeamIDs) + { + if (!string.IsNullOrEmpty(s)) + { + // ReSharper disable once StringLiteralTypo + sb.Append($" "); + } + } + + if (withHere) + { + sb.Append(" "); + } + + sb.Append(logString); + return sb.ToString(); + } + + + private static string Body(string logString, string stackTrace) + { + return $"{logString}\n\n```{stackTrace}```"; + } + + private class CoroutineRunner : MonoBehaviour + { + } + } +} diff --git a/Runtime/Reporters/SlackMessageSender.cs.meta b/Runtime/Reporters/SlackMessageSender.cs.meta new file mode 100644 index 0000000..87e16a2 --- /dev/null +++ b/Runtime/Reporters/SlackMessageSender.cs.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: 4bd66e2697dd4869b5a7fc506dc5a027 +timeCreated: 1699480963 \ No newline at end of file diff --git a/Runtime/Reporters/SlackReporter.cs b/Runtime/Reporters/SlackReporter.cs index b7014ba..0574388 100644 --- a/Runtime/Reporters/SlackReporter.cs +++ b/Runtime/Reporters/SlackReporter.cs @@ -1,7 +1,7 @@ // Copyright (c) 2023 DeNA Co., Ltd. // This software is released under the MIT License. -using System.Text; +using System.Threading; using Cysharp.Threading.Tasks; using DeNA.Anjin.Settings; using UnityEngine; @@ -11,111 +11,62 @@ namespace DeNA.Anjin.Reporters /// /// Post report to Slack /// - public class SlackReporter : IReporter + [CreateAssetMenu(fileName = "New SlackReporter", menuName = "Anjin/Slack Reporter", order = 50)] + public class SlackReporter : AbstractReporter { - private readonly AutopilotSettings _settings; - private readonly SlackAPI _slackAPI; - /// - /// Constructor + /// Slack API token /// - /// - /// - public SlackReporter(AutopilotSettings settings, SlackAPI slackAPI) - { - _settings = settings; - _slackAPI = slackAPI; - } + public string slackToken; /// - /// Post report log message, stacktrace and screenshot to Slack + /// Slack channels to send notification (comma separated) /// - /// Log message string - /// Stack trace - /// Log message type - /// With screenshot - /// - public async UniTask PostReportAsync(string logString, string stackTrace, LogType type, bool withScreenshot) - { - if (string.IsNullOrEmpty(_settings.slackToken) || string.IsNullOrEmpty(_settings.slackChannels)) - { - return; - } - - var title = Title(logString, _settings.mentionSubTeamIDs, _settings.addHereInSlackMessage); - var body = Body(logString, stackTrace); - - await UniTask.SwitchToMainThread(); - - foreach (var channel in _settings.slackChannels.Split(',')) - { - if (string.IsNullOrEmpty(channel)) - { - continue; - } - - var postTitleTask = await _slackAPI.Post(_settings.slackToken, channel, title); - if (!postTitleTask.Success) - { - return; - } - - if (withScreenshot && !Application.isBatchMode) - { - var coroutineRunner = new GameObject().AddComponent(); - await UniTask.WaitForEndOfFrame(coroutineRunner); - Object.Destroy(coroutineRunner); - - var screenshot = ScreenCapture.CaptureScreenshotAsTexture(); - var withoutAlpha = new Texture2D(screenshot.width, screenshot.height, TextureFormat.RGB24, false); - withoutAlpha.SetPixels(screenshot.GetPixels()); - withoutAlpha.Apply(); - - var postScreenshotTask = await _slackAPI.Post(_settings.slackToken, channel, - withoutAlpha.EncodeToPNG(), postTitleTask.Ts); - if (!postScreenshotTask.Success) - { - return; - } - } + public string slackChannels; - var postBodyTask = await _slackAPI.Post(_settings.slackToken, channel, body, - postTitleTask.Ts); - if (!postBodyTask.Success) - { - return; - } - } - } + /// + /// Sub team IDs to mention (comma separated) + /// + public string mentionSubTeamIDs; - private static string Title(string logString, string mentionSubTeamIDs, bool withHere) + /// + /// Whether adding @here or not + /// + public bool addHereInSlackMessage; + + private readonly ISlackMessageSender _sender = new SlackMessageSender(new SlackAPI()); + + /// + public override async UniTask PostReportAsync( + AutopilotSettings settings, + string logString, + string stackTrace, + LogType type, + bool withScreenshot, + CancellationToken cancellationToken = default + ) { - var title = new StringBuilder(); - - foreach (var s in mentionSubTeamIDs.Split(',')) + // NOTE: In _sender.send, switch the execution thread to the main thread, so UniTask.WhenAll is meaningless. + foreach (var slackChannel in (string.IsNullOrEmpty(settings.slackChannels) + ? slackChannels + : settings.slackChannels).Split(",")) { - if (!string.IsNullOrEmpty(s)) + if (cancellationToken.IsCancellationRequested) { - // ReSharper disable once StringLiteralTypo - title.Append($" "); + return; } + + await _sender.Send( + string.IsNullOrEmpty(settings.slackToken) ? slackToken : settings.slackToken, + slackChannel, + mentionSubTeamIDs.Split(","), + addHereInSlackMessage, + logString, + stackTrace, + withScreenshot, + cancellationToken + ); } - - if (withHere) - { - title.Append(" "); - } - - return title.Append(logString).ToString(); - } - - private static string Body(string logString, string stackTrace) - { - return $"{logString}\n\n```{stackTrace}```"; - } - - private class CoroutineRunner : MonoBehaviour - { } } } diff --git a/Runtime/Settings/AutopilotSettings.cs b/Runtime/Settings/AutopilotSettings.cs index 1a4a79b..78279a0 100644 --- a/Runtime/Settings/AutopilotSettings.cs +++ b/Runtime/Settings/AutopilotSettings.cs @@ -4,6 +4,7 @@ using System; using System.Collections.Generic; using DeNA.Anjin.Agents; +using DeNA.Anjin.Reporters; using UnityEngine; namespace DeNA.Anjin.Settings @@ -78,21 +79,25 @@ public class AutopilotSettings : ScriptableObject /// /// Slack API token /// + [Obsolete("this option moved to SlackReporter")] public string slackToken; /// /// Slack channels to send notification /// + [Obsolete("this option moved to SlackReporter")] public string slackChannels; /// /// Mention to sub team ID (comma separates) /// + [Obsolete("this option moved to SlackReporter")] public string mentionSubTeamIDs; /// /// Add @here /// + [Obsolete("this option moved to SlackReporter")] public bool addHereInSlackMessage; /// @@ -120,6 +125,11 @@ public class AutopilotSettings : ScriptableObject /// public string[] ignoreMessages; + /// + /// Reporter that called when some errors occurred + /// + public AbstractReporter reporter; + /// /// Overwrites specified values in the command line arguments /// @@ -147,12 +157,16 @@ public void OverrideByCommandLineArguments(Arguments args) if (args.SlackToken.IsCaptured()) { +#pragma warning disable CS0618 // Type or member is obsolete slackToken = args.SlackToken.Value(); +#pragma warning restore CS0618 // Type or member is obsolete } if (args.SlackChannels.IsCaptured()) { +#pragma warning disable CS0618 // Type or member is obsolete slackChannels = args.SlackChannels.Value(); +#pragma warning restore CS0618 // Type or member is obsolete } } } diff --git a/Runtime/Utilities/LogMessageHandler.cs b/Runtime/Utilities/LogMessageHandler.cs index 1a369e6..6563470 100644 --- a/Runtime/Utilities/LogMessageHandler.cs +++ b/Runtime/Utilities/LogMessageHandler.cs @@ -14,14 +14,14 @@ namespace DeNA.Anjin.Utilities public class LogMessageHandler { private readonly AutopilotSettings _settings; - private readonly IReporter _reporter; + private readonly AbstractReporter _reporter; /// /// Constructor /// /// Autopilot settings /// Reporter implementation - public LogMessageHandler(AutopilotSettings settings, IReporter reporter) + public LogMessageHandler(AutopilotSettings settings, AbstractReporter reporter) { _settings = settings; _reporter = reporter; @@ -42,7 +42,11 @@ public async void HandleLog(string logString, string stackTrace, LogType type) // NOTE: HandleLog may called by non-main thread because it subscribe Application.logMessageReceivedThreaded await UniTask.SwitchToMainThread(); - await _reporter.PostReportAsync(logString, stackTrace, type, true); + + if (_reporter != null) + { + await _reporter.PostReportAsync(_settings, logString, stackTrace, type, true); + } var autopilot = Object.FindObjectOfType(); if (autopilot != null) diff --git a/Tests/Editor/DeNA.Anjin.Editor.Tests.asmdef b/Tests/Editor/DeNA.Anjin.Editor.Tests.asmdef index 6c7f934..c4492e6 100644 --- a/Tests/Editor/DeNA.Anjin.Editor.Tests.asmdef +++ b/Tests/Editor/DeNA.Anjin.Editor.Tests.asmdef @@ -7,7 +7,8 @@ "DeNA.Anjin", "DeNA.Anjin.Annotations", "DeNA.Anjin.Editor", - "UniTask" + "UniTask", + "DeNA.Anjin.Tests" ], "includePlatforms": [ "Editor" diff --git a/Tests/Runtime/Reporters.meta b/Tests/Runtime/Reporters.meta new file mode 100644 index 0000000..3b23278 --- /dev/null +++ b/Tests/Runtime/Reporters.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: 84a88262c36d4c7e8b5e92add0851edf +timeCreated: 1699486146 \ No newline at end of file diff --git a/Tests/Runtime/Reporters/SlackMessageSenderTest.cs b/Tests/Runtime/Reporters/SlackMessageSenderTest.cs new file mode 100644 index 0000000..359ac18 --- /dev/null +++ b/Tests/Runtime/Reporters/SlackMessageSenderTest.cs @@ -0,0 +1,182 @@ +// Copyright (c) 2023 DeNA Co., Ltd. +// This software is released under the MIT License. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using DeNA.Anjin.TestDoubles; +using NUnit.Framework; + +namespace DeNA.Anjin.Reporters +{ + [TestFixture] + public class SlackMessageSenderTest + { + [Test] + public async Task NoMentionsAndNoScreenshotsAndNoAtHere() + { + var spySlackAPI = new SpySlackAPI(); + var sender = new SlackMessageSender(spySlackAPI); + + await sender.Send( + "TOKEN", + "CHANNEL", + Array.Empty(), + false, + "MESSAGE", + "STACKTRACE", + false + ); + + var actual = spySlackAPI.Arguments; + var expected = new List> + { + new Dictionary + { + { "token", "TOKEN" }, { "channel", "CHANNEL" }, { "message", "MESSAGE" }, { "ts", null } + }, + new Dictionary + { + { "token", "TOKEN" }, + { "channel", "CHANNEL" }, + { "message", "MESSAGE\n\n```STACKTRACE```" }, + { "ts", "1" } + }, + }; + Assert.That(actual, Is.EqualTo(expected), Format(actual)); + } + + [Test] + public async Task WithMention() + { + var spySlackAPI = new SpySlackAPI(); + var sender = new SlackMessageSender(spySlackAPI); + + await sender.Send( + "TOKEN", + "CHANNEL", + new []{"MENTION1", "MENTION2"}, + false, + "MESSAGE", + "STACKTRACE", + false + ); + + var actual = spySlackAPI.Arguments; + var expected = new List> + { + new Dictionary + { + { "token", "TOKEN" }, { "channel", "CHANNEL" }, { "message", " MESSAGE" }, { "ts", null } + }, + new Dictionary + { + { "token", "TOKEN" }, + { "channel", "CHANNEL" }, + { "message", "MESSAGE\n\n```STACKTRACE```" }, + { "ts", "1" } + }, + }; + Assert.That(actual, Is.EqualTo(expected), Format(actual)); + } + + [Test] + public async Task WithScreenshot() + { + var spySlackAPI = new SpySlackAPI(); + var sender = new SlackMessageSender(spySlackAPI); + + await sender.Send( + "TOKEN", + "CHANNEL", + new []{"MENTION1", "MENTION2"}, + false, + "MESSAGE", + "STACKTRACE", + true + ); + + var actual = spySlackAPI.Arguments; + var expected = new List> + { + new Dictionary + { + { "token", "TOKEN" }, { "channel", "CHANNEL" }, { "message", " MESSAGE" }, { "ts", null } + }, + new Dictionary + { + { "token", "TOKEN" }, { "channel", "CHANNEL" }, { "message", "IMAGE" }, { "ts", "1" } + }, + new Dictionary + { + { "token", "TOKEN" }, + { "channel", "CHANNEL" }, + { "message", "MESSAGE\n\n```STACKTRACE```" }, + { "ts", "1" } + }, + }; + Assert.That(actual, Is.EqualTo(expected)); + } + + [Test] + public async Task WithAtHere() + { + var spySlackAPI = new SpySlackAPI(); + var sender = new SlackMessageSender(spySlackAPI); + + await sender.Send( + "TOKEN", + "CHANNEL", + Array.Empty(), + true, + "MESSAGE", + "STACKTRACE", + false + ); + + var actual = spySlackAPI.Arguments; + var expected = new List> + { + new Dictionary + { + { "token", "TOKEN" }, { "channel", "CHANNEL" }, { "message", " MESSAGE" }, { "ts", null } + }, + new Dictionary + { + { "token", "TOKEN" }, + { "channel", "CHANNEL" }, + { "message", "MESSAGE\n\n```STACKTRACE```" }, + { "ts", "1" } + }, + }; + Assert.That(actual, Is.EqualTo(expected), Format(actual)); + } + + + private static string Format(List> dicts) + { + var sb = new StringBuilder(); + sb.AppendLine("["); + foreach (var dict in dicts) + { + sb.AppendLine("\t{"); + var keys = dict.Keys.ToArray(); + Array.Sort(keys); + foreach (var key in keys) + { + var value = dict[key]; + sb.Append("\t\t"); + sb.Append(key); + sb.Append(": "); + sb.Append(value); + sb.AppendLine(","); + } + sb.AppendLine("\t},"); + } + sb.AppendLine("]"); + return sb.ToString(); + } + } +} diff --git a/Tests/Runtime/Reporters/SlackMessageSenderTest.cs.meta b/Tests/Runtime/Reporters/SlackMessageSenderTest.cs.meta new file mode 100644 index 0000000..b8d2786 --- /dev/null +++ b/Tests/Runtime/Reporters/SlackMessageSenderTest.cs.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: 69a1be0e7f9c4e0aa62cb7d0884e5459 +timeCreated: 1699486165 \ No newline at end of file diff --git a/Tests/Runtime/TestDoubles/SpyReporter.cs b/Tests/Runtime/TestDoubles/SpyReporter.cs new file mode 100644 index 0000000..a1aca42 --- /dev/null +++ b/Tests/Runtime/TestDoubles/SpyReporter.cs @@ -0,0 +1,42 @@ +// Copyright (c) 2023 DeNA Co., Ltd. +// This software is released under the MIT License. + +using System.Collections.Generic; +using System.Threading; +using Cysharp.Threading.Tasks; +using DeNA.Anjin.Reporters; +using DeNA.Anjin.Settings; +using UnityEngine; + +namespace DeNA.Anjin.TestDoubles +{ + /// + /// A spy for + /// + // [CreateAssetMenu(fileName = "New SpyReporter", menuName = "Anjin/Spy Reporter", order = 34)] + public class SpyReporter : AbstractReporter + { + public List> Arguments { get; } = new List>(); + + public override async UniTask PostReportAsync( + AutopilotSettings settings, + string logString, + string stackTrace, + LogType type, + bool withScreenshot, + CancellationToken cancellationToken = default + ) + { + Debug.Log("Reporter called"); + Arguments.Add(new Dictionary + { + {"settings", settings.ToString()}, + {"logString", logString}, + {"stackTrace", stackTrace}, + {"type", type.ToString()}, + {"withScreenshot", withScreenshot.ToString()} + }); + await UniTask.NextFrame(cancellationToken); + } + } +} diff --git a/Tests/Runtime/TestDoubles/SpyReporter.cs.meta b/Tests/Runtime/TestDoubles/SpyReporter.cs.meta new file mode 100644 index 0000000..fcd1990 --- /dev/null +++ b/Tests/Runtime/TestDoubles/SpyReporter.cs.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: fbb88a2914494167b3e9f91d4dc51fdf +timeCreated: 1699485033 \ No newline at end of file diff --git a/Tests/Runtime/TestDoubles/StubThrowingErrorFromNotMainThreadOnClick.cs.meta b/Tests/Runtime/TestDoubles/StubThrowingErrorFromNotMainThreadOnClick.cs.meta index 37b8bb9..b7e3fd4 100644 --- a/Tests/Runtime/TestDoubles/StubThrowingErrorFromNotMainThreadOnClick.cs.meta +++ b/Tests/Runtime/TestDoubles/StubThrowingErrorFromNotMainThreadOnClick.cs.meta @@ -1,3 +1,3 @@ fileFormatVersion: 2 guid: c733c26bf53b475ea76992de2fcc418a -timeCreated: 1699513479 \ No newline at end of file +timeCreated: 1699513479 diff --git a/Tests/Runtime/Utilities/LogMessageHandlerTest.cs b/Tests/Runtime/Utilities/LogMessageHandlerTest.cs index d59cfdd..af45411 100644 --- a/Tests/Runtime/Utilities/LogMessageHandlerTest.cs +++ b/Tests/Runtime/Utilities/LogMessageHandlerTest.cs @@ -1,10 +1,8 @@ // Copyright (c) 2023 DeNA Co., Ltd. // This software is released under the MIT License. -using System.Collections.Generic; using System.Threading.Tasks; using Cysharp.Threading.Tasks; -using DeNA.Anjin.Reporters; using DeNA.Anjin.Settings; using DeNA.Anjin.TestDoubles; using NUnit.Framework; @@ -19,339 +17,158 @@ public class LogMessageHandlerTest public async Task HandleLog_LogTypeIsLog_notReported() { var settings = ScriptableObject.CreateInstance(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); sut.HandleLog(string.Empty, string.Empty, LogType.Log); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Empty); + Assert.That(spyReporter.Arguments, Is.Empty); } [Test] public async Task HandleLog_LogTypeExceptionHandle_reported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleException = true; sut.HandleLog(string.Empty, string.Empty, LogType.Exception); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Not.Empty); + Assert.That(spyReporter.Arguments, Is.Not.Empty); } [Test] public async Task HandleLog_LogTypeExceptionNotHandle_notReported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleException = false; sut.HandleLog(string.Empty, string.Empty, LogType.Exception); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Empty); + Assert.That(spyReporter.Arguments, Is.Empty); } [Test] public async Task HandleLog_LogTypeAssertHandle_reported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleAssert = true; sut.HandleLog(string.Empty, string.Empty, LogType.Assert); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Not.Empty); + Assert.That(spyReporter.Arguments, Is.Not.Empty); } [Test] public async Task HandleLog_LogTypeAssertNotHandle_notReported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleAssert = false; sut.HandleLog(string.Empty, string.Empty, LogType.Assert); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Empty); + Assert.That(spyReporter.Arguments, Is.Empty); } [Test] public async Task HandleLog_LogTypeErrorHandle_reported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleError = true; sut.HandleLog(string.Empty, string.Empty, LogType.Error); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Not.Empty); + Assert.That(spyReporter.Arguments, Is.Not.Empty); } [Test] public async Task HandleLog_LogTypeErrorNotHandle_notReported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleError = false; sut.HandleLog(string.Empty, string.Empty, LogType.Error); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Empty); + Assert.That(spyReporter.Arguments, Is.Empty); } [Test] public async Task HandleLog_LogTypeWarningHandle_reported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleWarning = true; sut.HandleLog(string.Empty, string.Empty, LogType.Warning); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Not.Empty); + Assert.That(spyReporter.Arguments, Is.Not.Empty); } [Test] public async Task HandleLog_LogTypeWarningNotHandle_notReported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleWarning = false; sut.HandleLog(string.Empty, string.Empty, LogType.Warning); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Empty); + Assert.That(spyReporter.Arguments, Is.Empty); } [Test] public async Task HandleLog_ContainsIgnoreMessage_notReported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.ignoreMessages = new[] { "ignore" }; settings.handleException = true; sut.HandleLog("xxx_ignore_xxx", string.Empty, LogType.Exception); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Empty); + Assert.That(spyReporter.Arguments, Is.Empty); } [Test] public async Task HandleLog_StacktraceByLogMessageHandler_notReported() { var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); + var spyReporter = ScriptableObject.CreateInstance(); + var sut = new LogMessageHandler(settings, spyReporter); settings.handleException = true; sut.HandleLog(string.Empty, "at DeNA.Anjin.Utilities.LogMessageHandler", LogType.Exception); await UniTask.NextFrame(); - Assert.That(spySlackAPI.Arguments, Is.Empty); - } - - [Test] - public async Task HandleLog_StacktraceBySlackAPI_notReported() - { - var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); - - settings.handleException = true; - sut.HandleLog(string.Empty, "at DeNA.Anjin.Reporters.SlackAPI", LogType.Exception); - await UniTask.NextFrame(); - - Assert.That(spySlackAPI.Arguments, Is.Empty); - } - - [Test] - [Category("IgnoreCI")] // Reason: screen capture is not work run on batchmode - public async Task HandleLog_SingleChannel_PostToSlack() - { - var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); - - settings.slackToken = "TOKEN"; - settings.slackChannels = "CHANNEL"; - settings.mentionSubTeamIDs = string.Empty; - settings.addHereInSlackMessage = false; - settings.handleException = true; - sut.HandleLog("MESSAGE", "STACKTRACE", LogType.Exception); - await UniTask.DelayFrame(5); - - var expected1 = new Dictionary - { - { "token", "TOKEN" }, { "channel", "CHANNEL" }, { "message", "MESSAGE" }, { "ts", null } - }; - var expected2 = new Dictionary - { - { "token", "TOKEN" }, { "channel", "CHANNEL" }, { "message", "IMAGE" }, { "ts", "1" } - }; - var expected3 = new Dictionary - { - { "token", "TOKEN" }, - { "channel", "CHANNEL" }, - { "message", "MESSAGE\n\n```STACKTRACE```" }, - { "ts", "1" } - }; - var expected = new List> { expected1, expected2, expected3 }; - var actual = spySlackAPI.Arguments; - - Assert.That(actual, Is.EqualTo(expected)); - } - - [Test] - public async Task HandleLog_SingleChannelWithMention_PostToSlack() - { - var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); - - settings.slackToken = "TOKEN"; - settings.slackChannels = "CHANNEL"; - settings.mentionSubTeamIDs = "MENTION1,MENTION2"; - settings.addHereInSlackMessage = false; - settings.handleException = true; - sut.HandleLog("MESSAGE", "STACKTRACE", LogType.Exception); - await UniTask.DelayFrame(5); - - var expected1 = new Dictionary - { - { "token", "TOKEN" }, - { "channel", "CHANNEL" }, - { "message", " MESSAGE" }, - { "ts", null } - }; - var actual = spySlackAPI.Arguments; - - Assert.That(actual[0], Is.EqualTo(expected1)); - } - - [Test] - public async Task HandleLog_SingleChannelWithHere_PostToSlack() - { - var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); - - settings.slackToken = "TOKEN"; - settings.slackChannels = "CHANNEL"; - settings.mentionSubTeamIDs = string.Empty; - settings.addHereInSlackMessage = true; // add `@here` - settings.handleException = true; - sut.HandleLog("MESSAGE", "STACKTRACE", LogType.Exception); - await UniTask.DelayFrame(5); - - var expected1 = new Dictionary - { - { "token", "TOKEN" }, { "channel", "CHANNEL" }, { "message", " MESSAGE" }, { "ts", null } - }; - var actual = spySlackAPI.Arguments; - - Assert.That(actual[0], Is.EqualTo(expected1)); - } - - [Test] - [Category("IgnoreCI")] // Reason: screen capture is not work run on batchmode - public async Task HandleLog_MultiChannel_PostToSlack() - { - var settings = CreateEmptyAutopilotSettings(); - var spySlackAPI = new SpySlackAPI(); - var reporter = new SlackReporter(settings, spySlackAPI); - var sut = new LogMessageHandler(settings, reporter); - - settings.slackToken = "TOKEN"; - settings.slackChannels = "CHANNEL1,CHANNEL2"; - settings.mentionSubTeamIDs = string.Empty; - settings.addHereInSlackMessage = false; - settings.handleException = true; - sut.HandleLog("MESSAGE", "STACKTRACE", LogType.Exception); - await UniTask.DelayFrame(8); - - var expected1 = new Dictionary - { - { "token", "TOKEN" }, { "channel", "CHANNEL1" }, { "message", "MESSAGE" }, { "ts", null } - }; - var expected2 = new Dictionary - { - { "token", "TOKEN" }, { "channel", "CHANNEL1" }, { "message", "IMAGE" }, { "ts", "1" } - }; - var expected3 = new Dictionary - { - { "token", "TOKEN" }, - { "channel", "CHANNEL1" }, - { "message", "MESSAGE\n\n```STACKTRACE```" }, - { "ts", "1" } - }; - var expected4 = new Dictionary - { - { "token", "TOKEN" }, { "channel", "CHANNEL2" }, { "message", "MESSAGE" }, { "ts", null } - }; - var expected5 = new Dictionary - { - { "token", "TOKEN" }, { "channel", "CHANNEL2" }, { "message", "IMAGE" }, { "ts", "1" } - }; - var expected6 = new Dictionary - { - { "token", "TOKEN" }, - { "channel", "CHANNEL2" }, - { "message", "MESSAGE\n\n```STACKTRACE```" }, - { "ts", "1" } - }; - var expected = new List> - { - expected1, - expected2, - expected3, - expected4, - expected5, - expected6 - }; - var actual = spySlackAPI.Arguments; - - Assert.That(actual, Is.EqualTo(expected)); + Assert.That(spyReporter.Arguments, Is.Empty); } private static AutopilotSettings CreateEmptyAutopilotSettings() { var settings = ScriptableObject.CreateInstance(); - settings.slackToken = "dummy token"; - settings.slackChannels = "dummy channel"; - settings.mentionSubTeamIDs = string.Empty; settings.ignoreMessages = new string[] { }; return settings; } diff --git a/Tests/TestAssets/AutopilotSettingsForImmediatlyFacingErrors.asset b/Tests/TestAssets/AutopilotSettingsForImmediatlyFacingErrors.asset new file mode 100644 index 0000000..51727c3 --- /dev/null +++ b/Tests/TestAssets/AutopilotSettingsForImmediatlyFacingErrors.asset @@ -0,0 +1,32 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!114 &11400000 +MonoBehaviour: + m_ObjectHideFlags: 0 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 0 + m_Script: {fileID: 11500000, guid: c7a1731caa574ed4b9334b7d52febe27, type: 3} + m_Name: AutopilotSettingsForImmediatlyFacingErrors + m_EditorClassIdentifier: + description: + sceneAgentMaps: [] + fallbackAgent: {fileID: 0} + observerAgent: {fileID: 0} + lifespanSec: 300 + randomSeed: + timeScale: 1 + junitReportPath: + slackToken: + slackChannels: + mentionSubTeamIDs: + addHereInSlackMessage: 0 + handleException: 1 + handleError: 1 + handleAssert: 1 + handleWarning: 0 + ignoreMessages: [] + reporter: {fileID: 0} diff --git a/Tests/TestAssets/AutopilotSettingsForImmediatlyFacingErrors.asset.meta b/Tests/TestAssets/AutopilotSettingsForImmediatlyFacingErrors.asset.meta new file mode 100644 index 0000000..43a69ed --- /dev/null +++ b/Tests/TestAssets/AutopilotSettingsForImmediatlyFacingErrors.asset.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: a0d887db4a0ca470f845d1c3d0236d31 +NativeFormatImporter: + externalObjects: {} + mainObjectFileID: 11400000 + userData: + assetBundleName: + assetBundleVariant: diff --git a/Tests/TestAssets/DoNothingAgentForTests.asset b/Tests/TestAssets/DoNothingAgentForTests.asset index 52bdab6..1979cf5 100644 --- a/Tests/TestAssets/DoNothingAgentForTests.asset +++ b/Tests/TestAssets/DoNothingAgentForTests.asset @@ -10,7 +10,7 @@ MonoBehaviour: m_Enabled: 1 m_EditorHideFlags: 0 m_Script: {fileID: 11500000, guid: 10ad27a6631a40909bf3a0615a1c7dc5, type: 3} - m_Name: DoNothingAgentForTests + m_Name: Observer m_EditorClassIdentifier: description: "1\u79D2\u3067\u505C\u6B62\u3057\u307E\u3059" lifespanSec: 1