#include "components/sync/engine/cycle/nudge_tracker.h"
#include <stdint.h>
#include <string>
#include <utility>
#include <vector>
#include "base/run_loop.h"
#include "components/sync/protocol/data_type_progress_marker.pb.h"
#include "components/sync/test/mock_invalidation.h"
#include "components/sync/test/model_type_test_util.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace syncer {
namespace {
testing::AssertionResult ModelTypeSetEquals(ModelTypeSet a, ModelTypeSet b) {
if (a == b) {
return testing::AssertionSuccess();
} else {
return testing::AssertionFailure()
<< "Left side " << ModelTypeSetToDebugString(a)
<< ", does not match rigth side: " << ModelTypeSetToDebugString(b);
}
}
}
class NudgeTrackerTest : public ::testing::Test {
public:
NudgeTrackerTest() {
SetInvalidationsInSync();
}
bool InvalidationsOutOfSync() const {
sync_pb::GetUpdateTriggers gu_trigger;
nudge_tracker_.FillProtoMessage(BOOKMARKS, &gu_trigger);
return gu_trigger.invalidations_out_of_sync();
}
int ProtoLocallyModifiedCount(ModelType type) const {
sync_pb::GetUpdateTriggers gu_trigger;
nudge_tracker_.FillProtoMessage(type, &gu_trigger);
return gu_trigger.local_modification_nudges();
}
int ProtoRefreshRequestedCount(ModelType type) const {
sync_pb::GetUpdateTriggers gu_trigger;
nudge_tracker_.FillProtoMessage(type, &gu_trigger);
return gu_trigger.datatype_refresh_nudges();
}
void SetInvalidationsInSync() {
nudge_tracker_.OnInvalidationsEnabled();
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked({});
}
std::unique_ptr<SyncInvalidation> BuildInvalidation(
int64_t version,
const std::string& payload) {
return MockInvalidation::Build(version, payload);
}
static std::unique_ptr<SyncInvalidation> BuildUnknownVersionInvalidation() {
return MockInvalidation::BuildUnknownVersion();
}
bool IsTypeThrottled(ModelType type) {
return nudge_tracker_.GetTypeBlockingMode(type) ==
WaitInterval::BlockingMode::kThrottled;
}
bool IsTypeBackedOff(ModelType type) {
return nudge_tracker_.GetTypeBlockingMode(type) ==
WaitInterval::BlockingMode::kExponentialBackoff;
}
protected:
NudgeTracker nudge_tracker_;
};
TEST_F(NudgeTrackerTest, EmptyNudgeTracker) {
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
EXPECT_EQ(sync_pb::SyncEnums::UNKNOWN_ORIGIN, nudge_tracker_.GetOrigin());
sync_pb::GetUpdateTriggers gu_trigger;
nudge_tracker_.FillProtoMessage(BOOKMARKS, &gu_trigger);
EXPECT_EQ(sync_pb::SyncEnums::UNKNOWN_ORIGIN, nudge_tracker_.GetOrigin());
}
TEST_F(NudgeTrackerTest, OriginPriorities) {
const base::TimeTicks t0 = base::TimeTicks() + base::Microseconds(1234);
const base::TimeTicks t1 = t0 + base::Seconds(10);
nudge_tracker_.SetNextRetryTime(t0);
nudge_tracker_.SetSyncCycleStartTime(t1);
EXPECT_EQ(sync_pb::SyncEnums::RETRY, nudge_tracker_.GetOrigin());
nudge_tracker_.RecordLocalChange(BOOKMARKS);
EXPECT_EQ(sync_pb::SyncEnums::GU_TRIGGER, nudge_tracker_.GetOrigin());
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(TYPED_URLS));
EXPECT_EQ(sync_pb::SyncEnums::GU_TRIGGER, nudge_tracker_.GetOrigin());
nudge_tracker_.RecordLocalChange(BOOKMARKS);
EXPECT_EQ(sync_pb::SyncEnums::GU_TRIGGER, nudge_tracker_.GetOrigin());
nudge_tracker_.SetHasPendingInvalidations(PREFERENCES, true);
EXPECT_EQ(sync_pb::SyncEnums::GU_TRIGGER, nudge_tracker_.GetOrigin());
nudge_tracker_.RecordLocalChange(BOOKMARKS);
EXPECT_EQ(sync_pb::SyncEnums::GU_TRIGGER, nudge_tracker_.GetOrigin());
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(TYPED_URLS));
EXPECT_EQ(sync_pb::SyncEnums::GU_TRIGGER, nudge_tracker_.GetOrigin());
}
TEST_F(NudgeTrackerTest, EnableDisableInvalidations) {
nudge_tracker_.OnInvalidationsDisabled();
EXPECT_TRUE(InvalidationsOutOfSync());
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.OnInvalidationsEnabled();
EXPECT_TRUE(InvalidationsOutOfSync());
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(InvalidationsOutOfSync());
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.OnInvalidationsDisabled();
EXPECT_TRUE(InvalidationsOutOfSync());
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_TRUE(InvalidationsOutOfSync());
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_TRUE(InvalidationsOutOfSync());
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
}
TEST_F(NudgeTrackerTest, WriteLocallyModifiedTypesToProto) {
EXPECT_EQ(0, ProtoLocallyModifiedCount(PREFERENCES));
nudge_tracker_.RecordLocalChange(PREFERENCES);
EXPECT_EQ(1, ProtoLocallyModifiedCount(PREFERENCES));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_EQ(0, ProtoLocallyModifiedCount(PREFERENCES));
}
TEST_F(NudgeTrackerTest, WriteRefreshRequestedTypesToProto) {
EXPECT_EQ(0, ProtoRefreshRequestedCount(SESSIONS));
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(SESSIONS));
EXPECT_EQ(1, ProtoRefreshRequestedCount(SESSIONS));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_EQ(0, ProtoRefreshRequestedCount(SESSIONS));
}
TEST_F(NudgeTrackerTest, IsSyncRequired) {
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordInitialSyncRequired(BOOKMARKS);
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordInitialSyncDone({EXTENSIONS});
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordInitialSyncDone({BOOKMARKS});
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordCommitConflict(BOOKMARKS);
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
EXPECT_TRUE(nudge_tracker_.IsSyncRequired({BOOKMARKS}));
EXPECT_FALSE(nudge_tracker_.IsSyncRequired({SESSIONS}));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked({SESSIONS});
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalChange(SESSIONS);
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(SESSIONS));
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.SetHasPendingInvalidations(PREFERENCES, true);
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
nudge_tracker_.SetHasPendingInvalidations(PREFERENCES, false);
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
}
TEST_F(NudgeTrackerTest, IsGetUpdatesRequired) {
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordInitialSyncRequired(BOOKMARKS);
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordInitialSyncDone(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalChange(SESSIONS);
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(SESSIONS));
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.SetHasPendingInvalidations(PREFERENCES, true);
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
nudge_tracker_.SetHasPendingInvalidations(PREFERENCES, false);
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
}
TEST_F(NudgeTrackerTest, IsSyncRequired_Throttling_Backoff) {
const base::TimeTicks now = base::TimeTicks::Now();
const base::TimeDelta throttle_length = base::Minutes(0);
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalChange(SESSIONS);
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.SetTypesThrottledUntil(ModelTypeSet(SESSIONS), throttle_length,
now);
EXPECT_TRUE(IsTypeThrottled(SESSIONS));
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(BOOKMARKS));
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.SetTypeBackedOff(BOOKMARKS, throttle_length, now);
EXPECT_TRUE(IsTypeThrottled(SESSIONS));
EXPECT_TRUE(IsTypeBackedOff(BOOKMARKS));
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(PREFERENCES));
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
nudge_tracker_.UpdateTypeThrottlingAndBackoffState();
EXPECT_FALSE(nudge_tracker_.IsTypeBlocked(SESSIONS));
EXPECT_FALSE(nudge_tracker_.IsTypeBlocked(BOOKMARKS));
EXPECT_TRUE(nudge_tracker_.IsSyncRequired(ModelTypeSet::All()));
}
TEST_F(NudgeTrackerTest, IsGetUpdatesRequired_Throttling_Backoff) {
const base::TimeTicks now = base::TimeTicks::Now();
const base::TimeDelta throttle_length = base::Minutes(0);
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(SESSIONS));
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.SetTypesThrottledUntil(ModelTypeSet(SESSIONS), throttle_length,
now);
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(BOOKMARKS));
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.SetTypeBackedOff(BOOKMARKS, throttle_length, now);
EXPECT_TRUE(IsTypeThrottled(SESSIONS));
EXPECT_TRUE(IsTypeBackedOff(BOOKMARKS));
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordLocalRefreshRequest(ModelTypeSet(PREFERENCES));
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.UpdateTypeThrottlingAndBackoffState();
EXPECT_FALSE(nudge_tracker_.IsTypeBlocked(SESSIONS));
EXPECT_FALSE(nudge_tracker_.IsTypeBlocked(BOOKMARKS));
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
}
TEST_F(NudgeTrackerTest, NoTypesBlocked) {
EXPECT_FALSE(nudge_tracker_.IsAnyTypeBlocked());
EXPECT_FALSE(nudge_tracker_.IsTypeBlocked(SESSIONS));
EXPECT_TRUE(nudge_tracker_.GetBlockedTypes().Empty());
}
TEST_F(NudgeTrackerTest, ThrottleAndUnthrottle) {
const base::TimeTicks now = base::TimeTicks::Now();
const base::TimeDelta throttle_length = base::Minutes(0);
nudge_tracker_.SetTypesThrottledUntil(ModelTypeSet(SESSIONS, PREFERENCES),
throttle_length, now);
EXPECT_TRUE(nudge_tracker_.IsAnyTypeBlocked());
EXPECT_TRUE(IsTypeThrottled(SESSIONS));
EXPECT_TRUE(IsTypeThrottled(PREFERENCES));
EXPECT_FALSE(nudge_tracker_.GetBlockedTypes().Empty());
EXPECT_EQ(throttle_length, nudge_tracker_.GetTimeUntilNextUnblock());
nudge_tracker_.UpdateTypeThrottlingAndBackoffState();
EXPECT_FALSE(nudge_tracker_.IsAnyTypeBlocked());
EXPECT_FALSE(nudge_tracker_.IsTypeBlocked(SESSIONS));
EXPECT_TRUE(nudge_tracker_.GetBlockedTypes().Empty());
}
TEST_F(NudgeTrackerTest, BackoffAndUnbackoff) {
const base::TimeTicks now = base::TimeTicks::Now();
const base::TimeDelta backoff_length = base::Minutes(0);
nudge_tracker_.SetTypeBackedOff(SESSIONS, backoff_length, now);
nudge_tracker_.SetTypeBackedOff(PREFERENCES, backoff_length, now);
EXPECT_TRUE(nudge_tracker_.IsAnyTypeBlocked());
EXPECT_TRUE(IsTypeBackedOff(SESSIONS));
EXPECT_TRUE(IsTypeBackedOff(PREFERENCES));
EXPECT_FALSE(nudge_tracker_.GetBlockedTypes().Empty());
EXPECT_EQ(backoff_length, nudge_tracker_.GetTimeUntilNextUnblock());
nudge_tracker_.UpdateTypeThrottlingAndBackoffState();
EXPECT_FALSE(nudge_tracker_.IsAnyTypeBlocked());
EXPECT_FALSE(nudge_tracker_.IsTypeBlocked(SESSIONS));
EXPECT_TRUE(nudge_tracker_.GetBlockedTypes().Empty());
}
TEST_F(NudgeTrackerTest, OverlappingThrottleIntervals) {
const base::TimeTicks now = base::TimeTicks::Now();
const base::TimeDelta throttle1_length = base::Minutes(0);
const base::TimeDelta throttle2_length = base::Minutes(20);
nudge_tracker_.SetTypesThrottledUntil(ModelTypeSet(SESSIONS, PREFERENCES),
throttle2_length, now);
EXPECT_TRUE(ModelTypeSetEquals(ModelTypeSet(SESSIONS, PREFERENCES),
nudge_tracker_.GetBlockedTypes()));
EXPECT_TRUE(IsTypeThrottled(SESSIONS));
EXPECT_TRUE(IsTypeThrottled(PREFERENCES));
EXPECT_GE(throttle2_length, nudge_tracker_.GetTimeUntilNextUnblock());
nudge_tracker_.SetTypesThrottledUntil(ModelTypeSet(SESSIONS, BOOKMARKS),
throttle1_length, now);
EXPECT_TRUE(ModelTypeSetEquals(ModelTypeSet(SESSIONS, PREFERENCES, BOOKMARKS),
nudge_tracker_.GetBlockedTypes()));
EXPECT_TRUE(IsTypeThrottled(SESSIONS));
EXPECT_TRUE(IsTypeThrottled(PREFERENCES));
EXPECT_TRUE(IsTypeThrottled(BOOKMARKS));
EXPECT_GE(throttle1_length, nudge_tracker_.GetTimeUntilNextUnblock());
nudge_tracker_.UpdateTypeThrottlingAndBackoffState();
EXPECT_TRUE(ModelTypeSetEquals(ModelTypeSet(SESSIONS, PREFERENCES),
nudge_tracker_.GetBlockedTypes()));
EXPECT_TRUE(IsTypeThrottled(SESSIONS));
EXPECT_TRUE(IsTypeThrottled(PREFERENCES));
EXPECT_FALSE(IsTypeThrottled(BOOKMARKS));
EXPECT_GE(throttle2_length - throttle1_length,
nudge_tracker_.GetTimeUntilNextUnblock());
}
TEST_F(NudgeTrackerTest, OverlappingBackoffIntervals) {
const base::TimeTicks now = base::TimeTicks::Now();
const base::TimeDelta backoff1_length = base::Minutes(0);
const base::TimeDelta backoff2_length = base::Minutes(20);
nudge_tracker_.SetTypeBackedOff(SESSIONS, backoff2_length, now);
nudge_tracker_.SetTypeBackedOff(PREFERENCES, backoff2_length, now);
EXPECT_TRUE(ModelTypeSetEquals(ModelTypeSet(SESSIONS, PREFERENCES),
nudge_tracker_.GetBlockedTypes()));
EXPECT_TRUE(IsTypeBackedOff(SESSIONS));
EXPECT_TRUE(IsTypeBackedOff(PREFERENCES));
EXPECT_GE(backoff2_length, nudge_tracker_.GetTimeUntilNextUnblock());
nudge_tracker_.SetTypeBackedOff(SESSIONS, backoff1_length, now);
nudge_tracker_.SetTypeBackedOff(BOOKMARKS, backoff1_length, now);
EXPECT_TRUE(ModelTypeSetEquals(ModelTypeSet(SESSIONS, PREFERENCES, BOOKMARKS),
nudge_tracker_.GetBlockedTypes()));
EXPECT_TRUE(IsTypeBackedOff(SESSIONS));
EXPECT_TRUE(IsTypeBackedOff(PREFERENCES));
EXPECT_TRUE(IsTypeBackedOff(BOOKMARKS));
EXPECT_GE(backoff1_length, nudge_tracker_.GetTimeUntilNextUnblock());
nudge_tracker_.UpdateTypeThrottlingAndBackoffState();
EXPECT_TRUE(ModelTypeSetEquals(ModelTypeSet(SESSIONS, PREFERENCES),
nudge_tracker_.GetBlockedTypes()));
EXPECT_TRUE(IsTypeBackedOff(SESSIONS));
EXPECT_TRUE(IsTypeBackedOff(PREFERENCES));
EXPECT_FALSE(IsTypeBackedOff(BOOKMARKS));
EXPECT_GE(backoff2_length - backoff1_length,
nudge_tracker_.GetTimeUntilNextUnblock());
}
TEST_F(NudgeTrackerTest, Retry) {
const base::TimeTicks t0 = base::TimeTicks::FromInternalValue(12345);
const base::TimeTicks t3 = t0 + base::Seconds(3);
const base::TimeTicks t4 = t0 + base::Seconds(4);
nudge_tracker_.SetNextRetryTime(t3);
nudge_tracker_.SetSyncCycleStartTime(t0);
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
EXPECT_FALSE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.SetSyncCycleStartTime(t4);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
EXPECT_TRUE(nudge_tracker_.IsGetUpdatesRequired(ModelTypeSet::All()));
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetSyncCycleStartTime(t4);
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
}
TEST_F(NudgeTrackerTest, IsRetryRequired_MidCycleUpdate1) {
const base::TimeTicks t0 = base::TimeTicks::FromInternalValue(12345);
const base::TimeTicks t1 = t0 + base::Seconds(1);
const base::TimeTicks t2 = t0 + base::Seconds(2);
const base::TimeTicks t5 = t0 + base::Seconds(5);
const base::TimeTicks t6 = t0 + base::Seconds(6);
nudge_tracker_.SetNextRetryTime(t0);
nudge_tracker_.SetSyncCycleStartTime(t1);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetNextRetryTime(t5);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetSyncCycleStartTime(t2);
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetSyncCycleStartTime(t6);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
}
TEST_F(NudgeTrackerTest, IsRetryRequired_MidCycleUpdate2) {
const base::TimeTicks t0 = base::TimeTicks::FromInternalValue(12345);
const base::TimeTicks t1 = t0 + base::Seconds(1);
const base::TimeTicks t3 = t0 + base::Seconds(3);
const base::TimeTicks t5 = t0 + base::Seconds(5);
const base::TimeTicks t6 = t0 + base::Seconds(6);
nudge_tracker_.RecordLocalChange(BOOKMARKS);
nudge_tracker_.SetNextRetryTime(t1);
nudge_tracker_.SetSyncCycleStartTime(t0);
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetNextRetryTime(t5);
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
nudge_tracker_.SetSyncCycleStartTime(t3);
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetSyncCycleStartTime(t6);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
}
TEST_F(NudgeTrackerTest, IsRetryRequired_FailedCycle) {
const base::TimeTicks t0 = base::TimeTicks::FromInternalValue(12345);
const base::TimeTicks t1 = t0 + base::Seconds(1);
const base::TimeTicks t2 = t0 + base::Seconds(2);
nudge_tracker_.SetNextRetryTime(t0);
nudge_tracker_.SetSyncCycleStartTime(t1);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetSyncCycleStartTime(t2);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
}
TEST_F(NudgeTrackerTest, IsRetryRequired_FailedCycleIncludesUpdate) {
const base::TimeTicks t0 = base::TimeTicks::FromInternalValue(12345);
const base::TimeTicks t1 = t0 + base::Seconds(1);
const base::TimeTicks t3 = t0 + base::Seconds(3);
const base::TimeTicks t4 = t0 + base::Seconds(4);
const base::TimeTicks t5 = t0 + base::Seconds(5);
const base::TimeTicks t6 = t0 + base::Seconds(6);
nudge_tracker_.SetNextRetryTime(t0);
nudge_tracker_.SetSyncCycleStartTime(t1);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetNextRetryTime(t5);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.SetSyncCycleStartTime(t3);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
nudge_tracker_.SetSyncCycleStartTime(t4);
EXPECT_FALSE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
nudge_tracker_.SetSyncCycleStartTime(t6);
EXPECT_TRUE(nudge_tracker_.IsRetryRequired());
nudge_tracker_.RecordSuccessfulSyncCycleIfNotBlocked(ModelTypeSet::All());
}
TEST_F(NudgeTrackerTest, NudgeDelayTest) {
EXPECT_EQ(nudge_tracker_.RecordLocalChange(TYPED_URLS),
nudge_tracker_.RecordLocalChange(PASSWORDS));
EXPECT_EQ(nudge_tracker_.RecordLocalChange(TYPED_URLS),
nudge_tracker_.RecordLocalChange(EXTENSIONS));
EXPECT_GT(nudge_tracker_.RecordLocalChange(BOOKMARKS),
nudge_tracker_.RecordLocalChange(TYPED_URLS));
EXPECT_EQ(nudge_tracker_.RecordLocalChange(BOOKMARKS),
nudge_tracker_.RecordLocalChange(PREFERENCES));
EXPECT_GT(nudge_tracker_.RecordLocalChange(SESSIONS),
nudge_tracker_.RecordLocalChange(BOOKMARKS));
EXPECT_GT(nudge_tracker_.RecordLocalChange(AUTOFILL),
nudge_tracker_.RecordLocalChange(SESSIONS));
EXPECT_GT(nudge_tracker_.RecordLocalChange(AUTOFILL), base::Hours(1));
EXPECT_EQ(nudge_tracker_.RecordLocalChange(AUTOFILL),
nudge_tracker_.RecordLocalChange(USER_EVENTS));
}
TEST_F(NudgeTrackerTest, CustomDelayTest) {
nudge_tracker_.SetLocalChangeDelayIgnoringMinForTest(BOOKMARKS,
base::Seconds(10));
nudge_tracker_.SetLocalChangeDelayIgnoringMinForTest(SESSIONS,
base::Seconds(2));
EXPECT_NE(nudge_tracker_.RecordLocalChange(BOOKMARKS),
nudge_tracker_.RecordLocalChange(PREFERENCES));
EXPECT_EQ(base::Seconds(10), nudge_tracker_.RecordLocalChange(BOOKMARKS));
EXPECT_EQ(base::Seconds(2), nudge_tracker_.RecordLocalChange(SESSIONS));
}
TEST_F(NudgeTrackerTest, DoNotUpdateDelayIfTooSmall) {
base::TimeDelta initial_delay = nudge_tracker_.RecordLocalChange(BOOKMARKS);
nudge_tracker_.UpdateLocalChangeDelay(BOOKMARKS, base::Microseconds(100));
EXPECT_EQ(initial_delay, nudge_tracker_.RecordLocalChange(BOOKMARKS));
}
}