8
8
using System ;
9
9
using System . Collections ;
10
10
using System . Collections . Generic ;
11
+ using System . Globalization ;
12
+ using System . Threading ;
11
13
using UnityEngine ;
12
14
13
15
[ assembly: System . Runtime . CompilerServices . InternalsVisibleTo ( "Backtrace.Unity.Tests.Runtime" ) ]
@@ -20,14 +22,21 @@ public class BacktraceClient : MonoBehaviour, IBacktraceClient
20
22
{
21
23
public BacktraceConfiguration Configuration ;
22
24
23
- public const string VERSION = "3.3.3 " ;
25
+ public const string VERSION = "3.4.0 " ;
24
26
public bool Enabled { get ; private set ; }
25
27
26
28
/// <summary>
27
29
/// Client attributes
28
30
/// </summary>
29
31
private readonly Dictionary < string , string > _clientAttributes = new Dictionary < string , string > ( ) ;
30
32
33
+ internal readonly Stack < BacktraceReport > BackgroundExceptions = new Stack < BacktraceReport > ( ) ;
34
+
35
+ /// <summary>
36
+ /// Client report attachments
37
+ /// </summary>
38
+ private List < string > _clientReportAttachments ;
39
+
31
40
/// <summary>
32
41
/// Attribute object accessor
33
42
/// </summary>
@@ -47,6 +56,26 @@ public string this[string index]
47
56
}
48
57
}
49
58
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
+
50
79
/// <summary>
51
80
/// Set client attributes that will be included in every report
52
81
/// </summary>
@@ -268,6 +297,7 @@ internal ReportLimitWatcher ReportLimitWatcher
268
297
/// </summary>
269
298
/// <param name="configuration">Backtrace configuration scriptable object</param>
270
299
/// <param name="attributes">Client side attributes</param>
300
+ /// param name="attachments">List of attachments </param>
271
301
/// <param name="gameObjectName">game object name</param>
272
302
/// <returns>Backtrace client</returns>
273
303
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
310
340
/// <param name="gameObjectName">game object name</param>
311
341
/// <returns>Backtrace client</returns>
312
342
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" )
313
357
{
314
358
var configuration = ScriptableObject . CreateInstance < BacktraceConfiguration > ( ) ;
315
359
configuration . ServerUrl = url ;
360
+ configuration . AttachmentPaths = attachments ;
316
361
configuration . Enabled = true ;
317
362
configuration . DatabasePath = databasePath ;
318
363
configuration . CreateDatabase = true ;
@@ -327,9 +372,23 @@ public static BacktraceClient Initialize(string url, string databasePath, Dictio
327
372
/// <param name="gameObjectName">game object name</param>
328
373
/// <returns>Backtrace client</returns>
329
374
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" )
330
388
{
331
389
var configuration = ScriptableObject . CreateInstance < BacktraceConfiguration > ( ) ;
332
390
configuration . ServerUrl = url ;
391
+ configuration . AttachmentPaths = attachments ;
333
392
configuration . Enabled = false ;
334
393
return Initialize ( configuration , attributes , gameObjectName ) ;
335
394
}
@@ -352,10 +411,10 @@ public void Refresh()
352
411
}
353
412
354
413
Enabled = true ;
355
-
414
+ _current = Thread . CurrentThread ;
356
415
CaptureUnityMessages ( ) ;
357
416
_reportLimitWatcher = new ReportLimitWatcher ( Convert . ToUInt32 ( Configuration . ReportPerMin ) ) ;
358
-
417
+ _clientReportAttachments = Configuration . GetAttachmentPaths ( ) ;
359
418
360
419
BacktraceApi = new BacktraceApi (
361
420
credentials : new BacktraceCredentials ( Configuration . GetValidServerUrl ( ) ) ,
@@ -413,20 +472,32 @@ private void Awake()
413
472
/// <summary>
414
473
/// Update native client internal ANR timer.
415
474
/// </summary>
416
- private void Update ( )
475
+ private void LateUpdate ( )
417
476
{
418
477
_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
+ }
419
490
}
420
491
421
492
private void OnDestroy ( )
422
493
{
423
494
Enabled = false ;
424
495
Application . logMessageReceived -= HandleUnityMessage ;
496
+ Application . logMessageReceivedThreaded -= HandleUnityBackgroundException ;
425
497
#if UNITY_ANDROID || UNITY_IOS
426
498
Application . lowMemory -= HandleLowMemory ;
427
499
_nativeClient ? . Disable ( ) ;
428
500
#endif
429
-
430
501
}
431
502
432
503
/// <summary>
@@ -606,7 +677,7 @@ record = Database.Add(data);
606
677
607
678
if ( data . Deduplication != 0 )
608
679
{
609
- queryAttributes [ "_mod_duplicate" ] = data . Deduplication . ToString ( ) ;
680
+ queryAttributes [ "_mod_duplicate" ] = data . Deduplication . ToString ( CultureInfo . InvariantCulture ) ;
610
681
}
611
682
612
683
StartCoroutine ( BacktraceApi . Send ( json , data . Attachments , queryAttributes , ( BacktraceResult result ) =>
@@ -653,6 +724,7 @@ private BacktraceData SetupBacktraceData(BacktraceReport report)
653
724
: _backtraceLogManager . ToSourceCode ( ) ;
654
725
655
726
report . AssignSourceCodeToReport ( sourceCode ) ;
727
+ report . AttachmentPaths . AddRange ( _clientReportAttachments ) ;
656
728
657
729
// pass copy of dictionary to prevent overriding client attributes
658
730
var result = report . ToBacktraceData ( null , GameObjectDepth ) ;
@@ -691,6 +763,8 @@ internal void OnAnrDetected(string stackTrace)
691
763
}
692
764
#endif
693
765
766
+ private Thread _current ;
767
+
694
768
/// <summary>
695
769
/// Handle Unity unhandled exceptions
696
770
/// </summary>
@@ -700,12 +774,29 @@ private void CaptureUnityMessages()
700
774
if ( Configuration . HandleUnhandledExceptions || Configuration . NumberOfLogs != 0 )
701
775
{
702
776
Application . logMessageReceived += HandleUnityMessage ;
777
+ Application . logMessageReceivedThreaded += HandleUnityBackgroundException ;
703
778
#if UNITY_ANDROID || UNITY_IOS
704
779
Application . lowMemory += HandleLowMemory ;
705
780
#endif
706
781
}
707
782
}
708
783
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
+
709
800
#if UNITY_ANDROID || UNITY_IOS
710
801
internal void HandleLowMemory ( )
711
802
{
@@ -824,10 +915,22 @@ private bool ShouldSendReport(Exception exception, List<string> attachmentPaths,
824
915
{
825
916
return false ;
826
917
}
918
+
827
919
//check rate limiting
828
- bool shouldProcess = _reportLimitWatcher . WatchReport ( new DateTime ( ) . Timestamp ( ) ) ;
920
+ bool shouldProcess = _reportLimitWatcher . WatchReport ( DateTimeHelper . Timestamp ( ) ) ;
829
921
if ( shouldProcess )
830
922
{
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
+ }
831
934
return true ;
832
935
}
833
936
if ( OnClientReportLimitReached != null )
@@ -849,9 +952,20 @@ private bool ShouldSendReport(string message, List<string> attachmentPaths, Dict
849
952
}
850
953
851
954
//check rate limiting
852
- bool shouldProcess = _reportLimitWatcher . WatchReport ( new DateTime ( ) . Timestamp ( ) ) ;
955
+ bool shouldProcess = _reportLimitWatcher . WatchReport ( DateTimeHelper . Timestamp ( ) ) ;
853
956
if ( shouldProcess )
854
957
{
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
+ }
855
969
return true ;
856
970
}
857
971
if ( OnClientReportLimitReached != null )
@@ -877,9 +991,19 @@ private bool ShouldSendReport(BacktraceReport report)
877
991
return false ;
878
992
}
879
993
//check rate limiting
880
- bool shouldProcess = _reportLimitWatcher . WatchReport ( new DateTime ( ) . Timestamp ( ) ) ;
994
+ bool shouldProcess = _reportLimitWatcher . WatchReport ( DateTimeHelper . Timestamp ( ) ) ;
881
995
if ( shouldProcess )
882
996
{
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
+ }
883
1007
return true ;
884
1008
}
885
1009
if ( OnClientReportLimitReached != null )
0 commit comments