// Copyright 2014 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #import "base/test/ios/wait_util.h" #import #include "base/logging.h" #include "base/mac/scoped_nsobject.h" #include "base/run_loop.h" #include "base/test/test_timeouts.h" #include "base/timer/elapsed_timer.h" namespace base { namespace test { namespace ios { const NSTimeInterval kSpinDelaySeconds = 0.01; const NSTimeInterval kWaitForJSCompletionTimeout = 4.0; const NSTimeInterval kWaitForUIElementTimeout = 4.0; const NSTimeInterval kWaitForDownloadTimeout = 10.0; const NSTimeInterval kWaitForPageLoadTimeout = 10.0; const NSTimeInterval kWaitForActionTimeout = 10.0; const NSTimeInterval kWaitForCookiesTimeout = 4.0; const NSTimeInterval kWaitForFileOperationTimeout = 2.0; bool WaitUntilConditionOrTimeout(NSTimeInterval timeout, ConditionBlock condition) { NSDate* deadline = [NSDate dateWithTimeIntervalSinceNow:timeout]; bool success = condition(); while (!success && [[NSDate date] compare:deadline] != NSOrderedDescending) { base::test::ios::SpinRunLoopWithMaxDelay( base::TimeDelta::FromSecondsD(kSpinDelaySeconds)); success = condition(); } return success; } TimeDelta TimeUntilCondition(ProceduralBlock action, ConditionBlock condition, bool run_message_loop, TimeDelta timeout) { ElapsedTimer timer; if (action) action(); if (timeout.is_zero()) timeout = TestTimeouts::action_timeout(); const TimeDelta spin_delay(TimeDelta::FromMilliseconds(10)); bool condition_evaluation_result = false; while (timer.Elapsed() < timeout && (!condition || !(condition_evaluation_result = condition()))) { SpinRunLoopWithMaxDelay(spin_delay); if (run_message_loop) RunLoop().RunUntilIdle(); } const TimeDelta elapsed = timer.Elapsed(); // If DCHECK is ever hit, check if |action| is doing something that is // taking an unreasonably long time, or if |condition| does not come // true quickly enough. Increase |timeout| only if necessary. DCHECK(!condition || condition_evaluation_result); return elapsed; } void WaitUntilCondition(ConditionBlock condition, bool run_message_loop, TimeDelta timeout) { TimeUntilCondition(nil, condition, run_message_loop, timeout); } void WaitUntilCondition(ConditionBlock condition) { WaitUntilCondition(condition, false, TimeDelta()); } void SpinRunLoopWithMaxDelay(TimeDelta max_delay) { scoped_nsobject beforeDate( [[NSDate alloc] initWithTimeIntervalSinceNow:max_delay.InSecondsF()]); [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:beforeDate]; } void SpinRunLoopWithMinDelay(TimeDelta min_delay) { ElapsedTimer timer; while (timer.Elapsed() < min_delay) { SpinRunLoopWithMaxDelay(TimeDelta::FromMilliseconds(10)); } } } // namespace ios } // namespace test } // namespace base