Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: Always start SDK on the main thread #3291

Merged
merged 47 commits into from
Oct 11, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
47 commits
Select commit Hold shift + click to select a range
b11c6c1
Update SentryCrashIntegration.m
brustolin Sep 19, 2023
23da6cc
Revert "Update SentryCrashIntegration.m"
brustolin Sep 19, 2023
95ab17a
Update SentrySDK.m
brustolin Sep 19, 2023
5a9e4ea
tests
brustolin Sep 19, 2023
f0b432f
Update CHANGELOG.md
brustolin Sep 19, 2023
4fa8e1f
Format code
getsentry-bot Sep 19, 2023
6ee2b67
Update Sources/SentryCrash/Recording/Tools/SentryCrashObjCApple.h
brustolin Sep 20, 2023
490ed4f
Format code
getsentry-bot Sep 20, 2023
3ffba00
Apply suggestions from code review
brustolin Sep 20, 2023
af5a3e5
Format code
getsentry-bot Sep 20, 2023
26f8b49
Update SentrySDK.m
brustolin Sep 20, 2023
b87e505
Format code
getsentry-bot Sep 20, 2023
673d34c
Update SentrySDKTests.swift
brustolin Sep 20, 2023
9304c1d
Merge branch 'main' into fix/access-UISession-main-thread
brustolin Sep 21, 2023
5791da4
Update SentrySDK.m
brustolin Sep 21, 2023
ddc053b
Merge branch 'fix/access-UISession-main-thread' of github.com:getsent…
brustolin Sep 21, 2023
17e6590
Merge branch 'main' into fix/access-UISession-main-thread
brustolin Sep 25, 2023
b96bc4b
Format code
getsentry-bot Sep 25, 2023
3418390
Merge branch 'main' into fix/access-UISession-main-thread
brustolin Sep 28, 2023
73db556
Update SentrySDKTests.swift
brustolin Sep 28, 2023
581b8df
Merge branch 'fix/access-UISession-main-thread' of github.com:getsent…
brustolin Sep 28, 2023
45b181b
Update SentrySDK.m
brustolin Sep 28, 2023
cd29f40
Format code
getsentry-bot Sep 28, 2023
fb5eebc
Update SentrySDKTests.swift
brustolin Sep 28, 2023
427f0a9
Merge branch 'fix/access-UISession-main-thread' of github.com:getsent…
brustolin Sep 28, 2023
e682529
Fix Reachability flackness
brustolin Sep 28, 2023
262c625
Format code
getsentry-bot Sep 28, 2023
d41ebfc
Update SentrySDK.m
brustolin Sep 28, 2023
60723f7
Merge branch 'fix/access-UISession-main-thread' of github.com:getsent…
brustolin Sep 28, 2023
3194c1d
Format code
getsentry-bot Sep 28, 2023
431978a
Merge branch 'main' into fix/access-UISession-main-thread
brustolin Oct 4, 2023
6692ff9
Update SentryReachability.m
brustolin Oct 4, 2023
aedd343
Format code
getsentry-bot Oct 4, 2023
cc4d0e3
Update SentrySDKTests.swift
brustolin Oct 5, 2023
f58236a
Update CHANGELOG.md
philipphofmann Oct 11, 2023
33e5dbe
ref: Replace deprecated function (#3327)
brustolin Oct 4, 2023
9bb1c59
ci: update unit test iOS/Xcode versions (#3283)
armcknight Oct 4, 2023
4815a2f
Fix: Remove unnecessaries build settings (#3325)
brustolin Oct 5, 2023
4ecd1bb
meta: update ruby version and gem/brew bundles (#3324)
armcknight Oct 5, 2023
86d4247
fix: remove relative component of import (#3332)
armcknight Oct 6, 2023
ff64ac7
rename replaceOptionIntegrations
philipphofmann Oct 11, 2023
b1292b1
Merge branch 'main' into fix/access-UISession-main-thread
philipphofmann Oct 11, 2023
d174abf
fix changelog
philipphofmann Oct 11, 2023
0c88c23
Make SDK init synchronous
philipphofmann Oct 11, 2023
e875917
change back to init async
philipphofmann Oct 11, 2023
0f5d42c
remove not needed file
philipphofmann Oct 11, 2023
d2a9730
simplify comment
philipphofmann Oct 11, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@

### Fixes

- Always start SDK on the main thread (#3291)
- App hang with race condition for tick counter (#3290)
- Remove "duplicate library" warning (#3312)
- Remove unnecessary build settings (#3325)
Expand Down
6 changes: 6 additions & 0 deletions Sources/Sentry/Public/SentrySDK.h
Original file line number Diff line number Diff line change
Expand Up @@ -27,12 +27,18 @@ SENTRY_NO_INIT
/**
* Inits and configures Sentry (SentryHub, SentryClient) and sets up all integrations. Make sure to
* set a valid DSN.
*
* @discussion Call this method on the main thread. When calling it from a background thread, the
* SDK starts on the main thread async.
*/
+ (void)startWithOptions:(SentryOptions *)options NS_SWIFT_NAME(start(options:));

/**
* Inits and configures Sentry (SentryHub, SentryClient) and sets up all integrations. Make sure to
* set a valid DSN.
*
* @discussion Call this method on the main thread. When calling it from a background thread, the
* SDK starts on the main thread async.
*/
+ (void)startWithConfigureOptions:(void (^)(SentryOptions *options))configureOptions
NS_SWIFT_NAME(start(configureOptions:));
Expand Down
17 changes: 14 additions & 3 deletions Sources/Sentry/SentryReachability.m
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,13 @@
}
}

void
SentryConnectivityReset(void)
{
[sentry_reachability_change_blocks removeAllObjects];
sentry_current_reachability_state = kSCNetworkReachabilityFlagsUninitialized;
}

@implementation SentryReachability

+ (void)initialize
Expand All @@ -109,9 +116,8 @@ + (void)initialize

- (void)dealloc
{
for (id<SentryReachabilityObserver> observer in sentry_reachability_change_blocks.allKeys) {
[self removeObserver:observer];
}
Comment on lines -112 to -114
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This make reachability flaky during tests.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shouldn't this change be in an extra PR?

sentry_reachability_change_blocks = [NSMutableDictionary dictionary];
[self removeReachabilityNotification];
}

- (void)addObserver:(id<SentryReachabilityObserver>)observer
Expand Down Expand Up @@ -147,6 +153,11 @@ - (void)removeObserver:(id<SentryReachabilityObserver>)observer
return;
}

[self removeReachabilityNotification];
}

- (void)removeReachabilityNotification
{
sentry_current_reachability_state = kSCNetworkReachabilityFlagsUninitialized;

if (_sentry_reachability_ref == nil) {
Expand Down
37 changes: 22 additions & 15 deletions Sources/Sentry/SentrySDK.m
Original file line number Diff line number Diff line change
Expand Up @@ -8,12 +8,14 @@
#import "SentryCrash.h"
#import "SentryCrashWrapper.h"
#import "SentryDependencyContainer.h"
#import "SentryDispatchQueueWrapper.h"
#import "SentryFileManager.h"
#import "SentryHub+Private.h"
#import "SentryLog.h"
#import "SentryMeta.h"
#import "SentryOptions+Private.h"
#import "SentryScope.h"
#import "SentryThreadWrapper.h"
#import "SentryUIDeviceWrapper.h"

@interface
Expand Down Expand Up @@ -135,27 +137,32 @@ + (void)setStartInvocations:(NSUInteger)value

+ (void)startWithOptions:(SentryOptions *)options
{
startInvocations++;
// We accept the tradeoff that the SDK might not be fully initialized directly after
// initializing it on a background thread because scheduling the init synchronously on the main
// thread could lead to deadlocks.
[SentryThreadWrapper onMainThread:^{
startInvocations++;

[SentryLog configure:options.debug diagnosticLevel:options.diagnosticLevel];
[SentryLog configure:options.debug diagnosticLevel:options.diagnosticLevel];

SentryClient *newClient = [[SentryClient alloc] initWithOptions:options];
[newClient.fileManager moveAppStateToPreviousAppState];
[newClient.fileManager moveBreadcrumbsToPreviousBreadcrumbs];
SentryClient *newClient = [[SentryClient alloc] initWithOptions:options];
[newClient.fileManager moveAppStateToPreviousAppState];
[newClient.fileManager moveBreadcrumbsToPreviousBreadcrumbs];

SentryScope *scope
= options.initialScope([[SentryScope alloc] initWithMaxBreadcrumbs:options.maxBreadcrumbs]);
// The Hub needs to be initialized with a client so that closing a session
// can happen.
[SentrySDK setCurrentHub:[[SentryHub alloc] initWithClient:newClient andScope:scope]];
SENTRY_LOG_DEBUG(@"SDK initialized! Version: %@", SentryMeta.versionString);
[SentrySDK installIntegrations];
SentryScope *scope = options.initialScope(
[[SentryScope alloc] initWithMaxBreadcrumbs:options.maxBreadcrumbs]);
// The Hub needs to be initialized with a client so that closing a session
// can happen.
[SentrySDK setCurrentHub:[[SentryHub alloc] initWithClient:newClient andScope:scope]];
SENTRY_LOG_DEBUG(@"SDK initialized! Version: %@", SentryMeta.versionString);
[SentrySDK installIntegrations];

[SentryCrashWrapper.sharedInstance startBinaryImageCache];
[SentryDependencyContainer.sharedInstance.binaryImageCache start];
[SentryCrashWrapper.sharedInstance startBinaryImageCache];
[SentryDependencyContainer.sharedInstance.binaryImageCache start];
#if TARGET_OS_IOS
[SentryDependencyContainer.sharedInstance.uiDeviceWrapper start];
[SentryDependencyContainer.sharedInstance.uiDeviceWrapper start];
#endif
}];
}

+ (void)startWithConfigureOptions:(void (^)(SentryOptions *options))configureOptions
Expand Down
3 changes: 3 additions & 0 deletions Tests/SentryTests/Networking/SentryReachabilityTests.m
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@
#import "SentryReachability.h"
#import <XCTest/XCTest.h>

void SentryConnectivityReset(void);

@interface TestSentryReachabilityObserver : NSObject <SentryReachabilityObserver>
@end
@implementation TestSentryReachabilityObserver
Expand All @@ -22,6 +24,7 @@ - (void)setUp
- (void)tearDown
{
self.reachability = nil;
SentryConnectivityReset();
}

- (void)testConnectivityRepresentations
Expand Down
34 changes: 33 additions & 1 deletion Tests/SentryTests/SentrySDKTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -653,7 +653,25 @@ class SentrySDKTests: XCTestCase {

XCTAssertEqual(flushTimeout, transport.flushInvocations.first)
}


func testStartOnTheMainThread() {

let expectation = expectation(description: "MainThreadTestIntegration install called")
MainThreadTestIntegration.expectation = expectation

DispatchQueue.global(qos: .background).async {
SentrySDK.start { options in
options.integrations = [ NSStringFromClass(MainThreadTestIntegration.self) ]
}
}

wait(for: [expectation], timeout: 1.0)

let mainThreadIntegration = SentrySDK.currentHub().installedIntegrations().first { integration in integration is MainThreadTestIntegration } as? MainThreadTestIntegration
XCTAssertEqual(mainThreadIntegration?.installedInTheMainThread, true, "SDK is not being initialized in the main thread")

}

#if SENTRY_HAS_UIKIT

func testSetAppStartMeasurementConcurrently() {
Expand Down Expand Up @@ -773,3 +791,17 @@ class SentrySDKTests: XCTestCase {
fixture.currentDate.setDate(date: SentryDependencyContainer.sharedInstance().dateProvider.date().addingTimeInterval(bySeconds))
}
}

public class MainThreadTestIntegration: NSObject, SentryIntegrationProtocol {

static var expectation: XCTestExpectation?

public var installedInTheMainThread = false

public func install(with options: Options) -> Bool {
installedInTheMainThread = Thread.isMainThread
MainThreadTestIntegration.expectation?.fulfill()
MainThreadTestIntegration.expectation = nil
return true
}
}
13 changes: 13 additions & 0 deletions develop-docs/DECISIONS.md
Original file line number Diff line number Diff line change
Expand Up @@ -127,3 +127,16 @@ This way, if these files are changed, we will be reminded to test the changes wi
Additionally, two new 'make' commands(test-alamofire, test-homekit) are being added to the project to assist in testing the Sentry SDK in third-party projects.

Related to [GH-2916](https://github.com/getsentry/sentry-cocoa/pull/2916)

## Async SDK init on main thread

Date: October 11th 2023
Contributors: @philipphofmann, @brustolin

We decided to initialize the SDK on the main thread async when being initialized from a background thread.
We accept the tradeoff that the SDK might not be fully initialized directly after initializing it on a background
thread because scheduling the init synchronously on the main thread could lead to deadlocks, such as https://github.com/getsentry/sentry-cocoa/issues/3277.

Related links:

- https://github.com/getsentry/sentry-cocoa/pull/3291