Skip to content

Commit ebe06b3

Browse files
authored
Version 3.4.0 - attachment and background thread support, prevent false-positive anrs (#69)
* Background thread support (#67) * Background thread support * Fixed exception flow * Removed renamed file * Detect false-positive ANRs report and prevent ANR watchdog from sending them (#66) * Detect false-positive ANRs report and prevent ANR watchdog from sending them + anr thread improvements * Fixed typos + better documentation - added Samy's advices * Attachment support (#68) * Attachment support on Android - iOS shouldn't compile * Updated the latest version of backtrace-android libraries and also adjust iOS integration code - code should now compile on iOS * Native attachment support * Attachment improvements * Updated label * Native client updates * Android libraries * Version update * 3.4.0-rc1 * Prevents from reading empty attachments * Arabic language support (#70) * Fixed invalid date in log manager when calendar is unsupported * Use single method to generate UTC timestamps + move it from extension code to static class to avoid unnecessary allocation * Fixed line endings * Updated native library * Removed debug logs from native library * Updated library version * Added invariant culture to rest of the code * native library update * Fixed issue in the readme file * Correct attachment name (#73) * Correct attachment name * Moved constants to const variables * Prepare for final release * Fixed typo in function name * Changelog change
1 parent f126a8f commit ebe06b3

39 files changed

+1117
-577
lines changed
Binary file not shown.
8 Bytes
Binary file not shown.
4 Bytes
Binary file not shown.
Binary file not shown.
4 KB
Binary file not shown.

CHANGELOG.md

+60-29
Large diffs are not rendered by default.

Editor/BacktraceConfigurationEditor.cs

+7-1
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,6 @@ public override void OnInspectorGUI()
6464
serializedObject.FindProperty("SendUnhandledGameCrashesOnGameStartup"),
6565
new GUIContent(BacktraceConfigurationLabels.LABEL_SEND_UNHANDLED_GAME_CRASHES_ON_STARTUP));
6666
#endif
67-
6867
EditorGUILayout.PropertyField(
6968
serializedObject.FindProperty("ReportFilterType"),
7069
new GUIContent(BacktraceConfigurationLabels.LABEL_REPORT_FILTER));
@@ -93,6 +92,13 @@ public override void OnInspectorGUI()
9392
EditorGUILayout.HelpBox("Please insert value greater or equal -1", MessageType.Error);
9493
}
9594
}
95+
96+
#if UNITY_ANDROID || UNITY_IOS
97+
EditorGUILayout.PropertyField(
98+
serializedObject.FindProperty("AttachmentPaths"),
99+
new GUIContent(BacktraceConfigurationLabels.LABEL_REPORT_ATTACHMENTS));
100+
#endif
101+
96102
#if !UNITY_SWITCH
97103
SerializedProperty enabled = serializedObject.FindProperty("Enabled");
98104
EditorGUILayout.PropertyField(enabled, new GUIContent(BacktraceConfigurationLabels.LABEL_ENABLE_DATABASE));

Editor/BacktraceConfigurationLabels.cs

+1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ internal static class BacktraceConfigurationLabels
1818
"(Early access) Send Out of memory exceptions to Backtrace";
1919
#endif
2020
#endif
21+
internal static string LABEL_REPORT_ATTACHMENTS = "Report attachment paths";
2122
internal static string CAPTURE_NATIVE_CRASHES = "Capture native crashes";
2223
internal static string LABEL_REPORT_FILTER = "Filter reports";
2324
internal static string LABEL_NUMBER_OF_LOGS = "Collect last n game logs";

README.md

+63-48
Large diffs are not rendered by default.

Runtime/BacktraceClient.cs

+133-9
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
using System;
99
using System.Collections;
1010
using System.Collections.Generic;
11+
using System.Globalization;
12+
using System.Threading;
1113
using UnityEngine;
1214

1315
[assembly: System.Runtime.CompilerServices.InternalsVisibleTo("Backtrace.Unity.Tests.Runtime")]
@@ -20,14 +22,21 @@ public class BacktraceClient : MonoBehaviour, IBacktraceClient
2022
{
2123
public BacktraceConfiguration Configuration;
2224

23-
public const string VERSION = "3.3.3";
25+
public const string VERSION = "3.4.0";
2426
public bool Enabled { get; private set; }
2527

2628
/// <summary>
2729
/// Client attributes
2830
/// </summary>
2931
private readonly Dictionary<string, string> _clientAttributes = new Dictionary<string, string>();
3032

33+
internal readonly Stack<BacktraceReport> BackgroundExceptions = new Stack<BacktraceReport>();
34+
35+
/// <summary>
36+
/// Client report attachments
37+
/// </summary>
38+
private List<string> _clientReportAttachments;
39+
3140
/// <summary>
3241
/// Attribute object accessor
3342
/// </summary>
@@ -47,6 +56,26 @@ public string this[string index]
4756
}
4857
}
4958

59+
/// <summary>
60+
/// Add attachment to managed reports.
61+
/// Note: this option won't add attachment to your native reports. You can add attachments to
62+
/// native reports only on BacktraceClient initialization.
63+
/// </summary>
64+
/// <param name="pathToAttachment">Path to attachment</param>
65+
public void AddAttachment(string pathToAttachment)
66+
{
67+
_clientReportAttachments.Add(pathToAttachment);
68+
}
69+
70+
/// <summary>
71+
/// Returns list of defined path to attachments stored by Backtrace client.
72+
/// </summary>
73+
/// <returns>List of client attachments</returns>
74+
public List<string> GetAttachments()
75+
{
76+
return _clientReportAttachments;
77+
}
78+
5079
/// <summary>
5180
/// Set client attributes that will be included in every report
5281
/// </summary>
@@ -268,6 +297,7 @@ internal ReportLimitWatcher ReportLimitWatcher
268297
/// </summary>
269298
/// <param name="configuration">Backtrace configuration scriptable object</param>
270299
/// <param name="attributes">Client side attributes</param>
300+
/// param name="attachments">List of attachments </param>
271301
/// <param name="gameObjectName">game object name</param>
272302
/// <returns>Backtrace client</returns>
273303
public static BacktraceClient Initialize(BacktraceConfiguration configuration, Dictionary<string, string> attributes = null, string gameObjectName = "BacktraceClient")
@@ -310,9 +340,24 @@ public static BacktraceClient Initialize(BacktraceConfiguration configuration, D
310340
/// <param name="gameObjectName">game object name</param>
311341
/// <returns>Backtrace client</returns>
312342
public static BacktraceClient Initialize(string url, string databasePath, Dictionary<string, string> attributes = null, string gameObjectName = "BacktraceClient")
343+
{
344+
return Initialize(url, databasePath, attributes, null, gameObjectName);
345+
}
346+
347+
/// <summary>
348+
/// Initialize new Backtrace integration with database path. Note - database path will be auto created by Backtrace Unity plugin
349+
/// </summary>
350+
/// <param name="url">Server url</param>
351+
/// <param name="databasePath">Database path</param>
352+
/// <param name="attributes">Client side attributes</param>
353+
/// <param name="attachments">Paths to attachments that Backtrace client will include in managed/native reports</param>
354+
/// <param name="gameObjectName">game object name</param>
355+
/// <returns>Backtrace client</returns>
356+
public static BacktraceClient Initialize(string url, string databasePath, Dictionary<string, string> attributes = null, string[] attachments = null, string gameObjectName = "BacktraceClient")
313357
{
314358
var configuration = ScriptableObject.CreateInstance<BacktraceConfiguration>();
315359
configuration.ServerUrl = url;
360+
configuration.AttachmentPaths = attachments;
316361
configuration.Enabled = true;
317362
configuration.DatabasePath = databasePath;
318363
configuration.CreateDatabase = true;
@@ -327,9 +372,23 @@ public static BacktraceClient Initialize(string url, string databasePath, Dictio
327372
/// <param name="gameObjectName">game object name</param>
328373
/// <returns>Backtrace client</returns>
329374
public static BacktraceClient Initialize(string url, Dictionary<string, string> attributes = null, string gameObjectName = "BacktraceClient")
375+
{
376+
return Initialize(url, attributes, new string[0], gameObjectName);
377+
}
378+
379+
/// <summary>
380+
/// Initialize new Backtrace integration
381+
/// </summary>
382+
/// <param name="url">Server url</param>
383+
/// <param name="attributes">Client side attributes</param>
384+
/// <param name="attachments">Paths to attachments that Backtrace client will include in managed/native reports</param>
385+
/// <param name="gameObjectName">game object name</param>
386+
/// <returns>Backtrace client</returns>
387+
public static BacktraceClient Initialize(string url, Dictionary<string, string> attributes = null, string[] attachments = null, string gameObjectName = "BacktraceClient")
330388
{
331389
var configuration = ScriptableObject.CreateInstance<BacktraceConfiguration>();
332390
configuration.ServerUrl = url;
391+
configuration.AttachmentPaths = attachments;
333392
configuration.Enabled = false;
334393
return Initialize(configuration, attributes, gameObjectName);
335394
}
@@ -352,10 +411,10 @@ public void Refresh()
352411
}
353412

354413
Enabled = true;
355-
414+
_current = Thread.CurrentThread;
356415
CaptureUnityMessages();
357416
_reportLimitWatcher = new ReportLimitWatcher(Convert.ToUInt32(Configuration.ReportPerMin));
358-
417+
_clientReportAttachments = Configuration.GetAttachmentPaths();
359418

360419
BacktraceApi = new BacktraceApi(
361420
credentials: new BacktraceCredentials(Configuration.GetValidServerUrl()),
@@ -413,20 +472,32 @@ private void Awake()
413472
/// <summary>
414473
/// Update native client internal ANR timer.
415474
/// </summary>
416-
private void Update()
475+
private void LateUpdate()
417476
{
418477
_nativeClient?.UpdateClientTime(Time.unscaledTime);
478+
479+
if (BackgroundExceptions.Count == 0)
480+
{
481+
return;
482+
}
483+
while (BackgroundExceptions.Count > 0)
484+
{
485+
// use SendReport method isntead of Send method
486+
// because we already applied all watchdog/skipReport rules
487+
// so we don't need to apply them once again
488+
SendReport(BackgroundExceptions.Pop());
489+
}
419490
}
420491

421492
private void OnDestroy()
422493
{
423494
Enabled = false;
424495
Application.logMessageReceived -= HandleUnityMessage;
496+
Application.logMessageReceivedThreaded -= HandleUnityBackgroundException;
425497
#if UNITY_ANDROID || UNITY_IOS
426498
Application.lowMemory -= HandleLowMemory;
427499
_nativeClient?.Disable();
428500
#endif
429-
430501
}
431502

432503
/// <summary>
@@ -606,7 +677,7 @@ record = Database.Add(data);
606677

607678
if (data.Deduplication != 0)
608679
{
609-
queryAttributes["_mod_duplicate"] = data.Deduplication.ToString();
680+
queryAttributes["_mod_duplicate"] = data.Deduplication.ToString(CultureInfo.InvariantCulture);
610681
}
611682

612683
StartCoroutine(BacktraceApi.Send(json, data.Attachments, queryAttributes, (BacktraceResult result) =>
@@ -653,6 +724,7 @@ private BacktraceData SetupBacktraceData(BacktraceReport report)
653724
: _backtraceLogManager.ToSourceCode();
654725

655726
report.AssignSourceCodeToReport(sourceCode);
727+
report.AttachmentPaths.AddRange(_clientReportAttachments);
656728

657729
// pass copy of dictionary to prevent overriding client attributes
658730
var result = report.ToBacktraceData(null, GameObjectDepth);
@@ -691,6 +763,8 @@ internal void OnAnrDetected(string stackTrace)
691763
}
692764
#endif
693765

766+
private Thread _current;
767+
694768
/// <summary>
695769
/// Handle Unity unhandled exceptions
696770
/// </summary>
@@ -700,12 +774,29 @@ private void CaptureUnityMessages()
700774
if (Configuration.HandleUnhandledExceptions || Configuration.NumberOfLogs != 0)
701775
{
702776
Application.logMessageReceived += HandleUnityMessage;
777+
Application.logMessageReceivedThreaded += HandleUnityBackgroundException;
703778
#if UNITY_ANDROID || UNITY_IOS
704779
Application.lowMemory += HandleLowMemory;
705780
#endif
706781
}
707782
}
708783

784+
internal void OnApplicationPause(bool pause)
785+
{
786+
_nativeClient?.PauseAnrThread(pause);
787+
}
788+
789+
internal void HandleUnityBackgroundException(string message, string stackTrace, LogType type)
790+
{
791+
// validate if a message is from main thread
792+
// and skip messages from main thread
793+
if (Thread.CurrentThread == _current)
794+
{
795+
return;
796+
}
797+
HandleUnityMessage(message, stackTrace, type);
798+
}
799+
709800
#if UNITY_ANDROID || UNITY_IOS
710801
internal void HandleLowMemory()
711802
{
@@ -824,10 +915,22 @@ private bool ShouldSendReport(Exception exception, List<string> attachmentPaths,
824915
{
825916
return false;
826917
}
918+
827919
//check rate limiting
828-
bool shouldProcess = _reportLimitWatcher.WatchReport(new DateTime().Timestamp());
920+
bool shouldProcess = _reportLimitWatcher.WatchReport(DateTimeHelper.Timestamp());
829921
if (shouldProcess)
830922
{
923+
// This condition checks if we should send exception from current thread
924+
// if comparision result confirm that we're trying to send an exception from different
925+
// thread than main, we should add the exception object to the exception list
926+
// and let update method send data to Backtrace.
927+
if (Thread.CurrentThread.ManagedThreadId != _current.ManagedThreadId)
928+
{
929+
var report = new BacktraceReport(exception, attributes, attachmentPaths);
930+
report.Attributes["exception.thread"] = Thread.CurrentThread.ManagedThreadId.ToString(CultureInfo.InvariantCulture);
931+
BackgroundExceptions.Push(report);
932+
return false;
933+
}
831934
return true;
832935
}
833936
if (OnClientReportLimitReached != null)
@@ -849,9 +952,20 @@ private bool ShouldSendReport(string message, List<string> attachmentPaths, Dict
849952
}
850953

851954
//check rate limiting
852-
bool shouldProcess = _reportLimitWatcher.WatchReport(new DateTime().Timestamp());
955+
bool shouldProcess = _reportLimitWatcher.WatchReport(DateTimeHelper.Timestamp());
853956
if (shouldProcess)
854957
{
958+
// This condition checks if we should send exception from current thread
959+
// if comparision result confirm that we're trying to send an exception from different
960+
// thread than main, we should add the exception object to the exception list
961+
// and let update method send data to Backtrace.
962+
if (Thread.CurrentThread.ManagedThreadId != _current.ManagedThreadId)
963+
{
964+
var report = new BacktraceReport(message, attributes, attachmentPaths);
965+
report.Attributes["exception.thread"] = Thread.CurrentThread.ManagedThreadId.ToString(CultureInfo.InvariantCulture);
966+
BackgroundExceptions.Push(report);
967+
return false;
968+
}
855969
return true;
856970
}
857971
if (OnClientReportLimitReached != null)
@@ -877,9 +991,19 @@ private bool ShouldSendReport(BacktraceReport report)
877991
return false;
878992
}
879993
//check rate limiting
880-
bool shouldProcess = _reportLimitWatcher.WatchReport(new DateTime().Timestamp());
994+
bool shouldProcess = _reportLimitWatcher.WatchReport(DateTimeHelper.Timestamp());
881995
if (shouldProcess)
882996
{
997+
// This condition checks if we should send exception from current thread
998+
// if comparision result confirm that we're trying to send an exception from different
999+
// thread than main, we should add the exception object to the exception list
1000+
// and let update method send data to Backtrace.
1001+
if (Thread.CurrentThread.ManagedThreadId != _current.ManagedThreadId)
1002+
{
1003+
report.Attributes["exception.thread"] = Thread.CurrentThread.ManagedThreadId.ToString(CultureInfo.InvariantCulture);
1004+
BackgroundExceptions.Push(report);
1005+
return false;
1006+
}
8831007
return true;
8841008
}
8851009
if (OnClientReportLimitReached != null)

Runtime/BacktraceDatabase.cs

+4-3
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
using Backtrace.Unity.Types;
77
using System;
88
using System.Collections.Generic;
9+
using System.Globalization;
910
using System.IO;
1011
using UnityEngine;
1112

@@ -372,7 +373,7 @@ private void FlushRecord(BacktraceDatabaseRecord record)
372373
return;
373374
}
374375

375-
queryAttributes["_mod_duplicate"] = record.Count.ToString();
376+
queryAttributes["_mod_duplicate"] = record.Count.ToString(CultureInfo.InvariantCulture);
376377

377378
StartCoroutine(
378379
BacktraceApi.Send(backtraceData, record.Attachments, queryAttributes, (BacktraceResult result) =>
@@ -403,7 +404,7 @@ private void SendData(BacktraceDatabaseRecord record)
403404
stopWatch.Stop();
404405
queryAttributes["performance.database.send"] = stopWatch.GetMicroseconds();
405406
}
406-
queryAttributes["_mod_duplicate"] = record.Count.ToString();
407+
queryAttributes["_mod_duplicate"] = record.Count.ToString(CultureInfo.InvariantCulture);
407408

408409
StartCoroutine(
409410
BacktraceApi.Send(backtraceData, record.Attachments, queryAttributes, (BacktraceResult sendResult) =>
@@ -418,7 +419,7 @@ private void SendData(BacktraceDatabaseRecord record)
418419
BacktraceDatabaseContext.IncrementBatchRetry();
419420
return;
420421
}
421-
bool shouldProcess = _reportLimitWatcher.WatchReport(new DateTime().Timestamp());
422+
bool shouldProcess = _reportLimitWatcher.WatchReport(DateTimeHelper.Timestamp());
422423
if (!shouldProcess)
423424
{
424425
return;

0 commit comments

Comments
 (0)