-
Notifications
You must be signed in to change notification settings - Fork 104
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
938 Fixed Device UA being empty on First Bid Request #957
938 Fixed Device UA being empty on First Bid Request #957
Conversation
PrebidMobile/PrebidMobileRendering/Utilities/PBMUserAgentService.m
Outdated
Show resolved
Hide resolved
PrebidMobile/PrebidMobileRendering/Utilities/PBMUserAgentService.m
Outdated
Show resolved
Hide resolved
Issue:I am actually quite stumped on this because of the following: The current implementation MUST include a completion handler in
We cannot use ASYNC AWAIT or a similar concurrent Swift function as it is too new (iOS 15+). This will not work as the minimum target version for GMA SDK is iOS 12.0. Proposed Solutions:
|
… nslock
This reverts commit 5f35026.
@@ -68,6 +70,13 @@ - (instancetype)init { | |||
#pragma mark - Public Methods | |||
|
|||
- (nonnull NSString *)getFullUserAgent { | |||
if (self.userAgent == nil) { | |||
self.uaSemaphore = dispatch_semaphore_create(0); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If two ad units call this method at the same time before ua is obtained, the semaphore will be recreated, with unpredictable consequences
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure about using semaphore here. It is a tool dedicated to multithreading access to resources. But we need just to lock the process for any resource consumer. So, for me, using NSLock will be much more convenient here. It will let us avoid using the while loop for waiting for the resource. But we still need to be sure of proper unlocking.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I moved the instantiation of the semaphore. The larger issue is that there are multiple threads at work here:
- The thread that asks for the userAgent (that has to be the main thread because only the main thread can reference
webView.evaluateJavaScript
) - The thread where
self.webView.evaluateJavaScript
runs which is run on a background thread - The
evaluateJavaScript completionHandler
which is executed on the main thread.
So we cannot use NSLock here because we can't lock the main thread as the completionHandler
has to execute.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
NSLock doesn't lock anything itself. But yes, the requirement that lock() and unlock() should be called on the same thread makes it hardly usable in this situation.
The current implementation looks good. However, this loop looks potentially endless. In the case when the semaphore will be actually incremented before the start of the loop, the wait will always be finished with a timeout. Does it make sense to add an extra check for userAgent inside the loop or limit the number of possible loops?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM
Fixes #938
Removes the race condition and uses UIWebView instead of WKWebView. UIWebView does javascript calls synchronously while WKWebView does javascript asynchronously which is why the race condition is occurring. While synchronous execution might take an extra second, it prevents a lot of headaches and dispatch_once makes sure this only happens once. The UserAgent is also cached in the shared PBMUserAgentService.