/*
* Copyright (C) 2018 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "src/perfetto_cmd/rate_limiter.h"
#include <stdio.h>
#include "perfetto/base/file_utils.h"
#include "perfetto/base/scoped_file.h"
#include "perfetto/base/temp_file.h"
#include "perfetto/base/utils.h"
#include "gmock/gmock.h"
#include "gtest/gtest.h"
using testing::_;
using testing::NiceMock;
using testing::StrictMock;
using testing::Invoke;
using testing::Return;
namespace perfetto {
namespace {
class MockRateLimiter : public RateLimiter {
public:
MockRateLimiter() : dir_(base::TempDir::Create()) {
ON_CALL(*this, LoadState(_))
.WillByDefault(Invoke(this, &MockRateLimiter::LoadStateConcrete));
ON_CALL(*this, SaveState(_))
.WillByDefault(Invoke(this, &MockRateLimiter::SaveStateConcrete));
}
virtual std::string GetStateFilePath() const {
return std::string(dir_.path()) + "/.guardraildata";
}
virtual ~MockRateLimiter() override {
if (StateFileExists())
remove(GetStateFilePath().c_str());
}
bool LoadStateConcrete(PerfettoCmdState* state) {
return RateLimiter::LoadState(state);
}
bool SaveStateConcrete(const PerfettoCmdState& state) {
return RateLimiter::SaveState(state);
}
MOCK_METHOD1(LoadState, bool(PerfettoCmdState*));
MOCK_METHOD1(SaveState, bool(const PerfettoCmdState&));
private:
base::TempDir dir_;
};
void WriteGarbageToFile(const std::string& path) {
base::ScopedFile fd(base::OpenFile(path, O_WRONLY | O_CREAT, 0600));
constexpr char data[] = "Some random bytes.";
if (base::WriteAll(fd.get(), data, sizeof(data)) != sizeof(data))
ADD_FAILURE() << "Could not write garbage";
}
TEST(RateLimiterTest, RoundTripState) {
NiceMock<MockRateLimiter> limiter;
PerfettoCmdState input{};
PerfettoCmdState output{};
input.set_total_bytes_uploaded(42);
ASSERT_TRUE(limiter.SaveState(input));
ASSERT_TRUE(limiter.LoadState(&output));
ASSERT_EQ(output.total_bytes_uploaded(), 42u);
}
TEST(RateLimiterTest, LoadFromEmpty) {
NiceMock<MockRateLimiter> limiter;
PerfettoCmdState input{};
input.set_total_bytes_uploaded(0);
input.set_last_trace_timestamp(0);
input.set_first_trace_timestamp(0);
PerfettoCmdState output{};
ASSERT_TRUE(limiter.SaveState(input));
ASSERT_TRUE(limiter.LoadState(&output));
ASSERT_EQ(output.total_bytes_uploaded(), 0u);
}
TEST(RateLimiterTest, LoadFromNoFileFails) {
NiceMock<MockRateLimiter> limiter;
PerfettoCmdState output{};
ASSERT_FALSE(limiter.LoadState(&output));
ASSERT_EQ(output.total_bytes_uploaded(), 0u);
}
TEST(RateLimiterTest, LoadFromGarbageFails) {
NiceMock<MockRateLimiter> limiter;
WriteGarbageToFile(limiter.GetStateFilePath().c_str());
PerfettoCmdState output{};
ASSERT_FALSE(limiter.LoadState(&output));
ASSERT_EQ(output.total_bytes_uploaded(), 0u);
}
TEST(RateLimiterTest, NotDropBox) {
StrictMock<MockRateLimiter> limiter;
ASSERT_TRUE(limiter.ShouldTrace({}));
ASSERT_TRUE(limiter.OnTraceDone({}, true, 10000));
ASSERT_FALSE(limiter.StateFileExists());
}
TEST(RateLimiterTest, NotDropBox_FailedToTrace) {
StrictMock<MockRateLimiter> limiter;
ASSERT_FALSE(limiter.OnTraceDone({}, false, 0));
ASSERT_FALSE(limiter.StateFileExists());
}
TEST(RateLimiterTest, DropBox_IgnoreGuardrails) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
args.allow_user_build_tracing = true;
args.is_dropbox = true;
args.ignore_guardrails = true;
args.current_time = base::TimeSeconds(41);
EXPECT_CALL(limiter, SaveState(_));
EXPECT_CALL(limiter, LoadState(_));
ASSERT_TRUE(limiter.ShouldTrace(args));
EXPECT_CALL(limiter, SaveState(_));
ASSERT_TRUE(limiter.OnTraceDone(args, true, 42u));
PerfettoCmdState output{};
ASSERT_TRUE(limiter.LoadStateConcrete(&output));
ASSERT_EQ(output.first_trace_timestamp(), 41u);
ASSERT_EQ(output.last_trace_timestamp(), 41u);
ASSERT_EQ(output.total_bytes_uploaded(), 42u);
}
TEST(RateLimiterTest, DropBox_EmptyState) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
args.allow_user_build_tracing = true;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(10000);
EXPECT_CALL(limiter, SaveState(_));
EXPECT_CALL(limiter, LoadState(_));
ASSERT_TRUE(limiter.ShouldTrace(args));
EXPECT_CALL(limiter, SaveState(_));
ASSERT_TRUE(limiter.OnTraceDone(args, true, 1024 * 1024));
PerfettoCmdState output{};
ASSERT_TRUE(limiter.LoadStateConcrete(&output));
EXPECT_EQ(output.total_bytes_uploaded(), 1024u * 1024u);
EXPECT_EQ(output.first_trace_timestamp(), 10000u);
EXPECT_EQ(output.last_trace_timestamp(), 10000u);
}
TEST(RateLimiterTest, DropBox_NormalUpload) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
PerfettoCmdState input{};
input.set_first_trace_timestamp(10000);
input.set_last_trace_timestamp(10000 + 60 * 10);
input.set_total_bytes_uploaded(1024 * 1024 * 2);
ASSERT_TRUE(limiter.SaveStateConcrete(input));
args.allow_user_build_tracing = true;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(input.last_trace_timestamp() + 60 * 10);
EXPECT_CALL(limiter, LoadState(_));
ASSERT_TRUE(limiter.ShouldTrace(args));
EXPECT_CALL(limiter, SaveState(_));
ASSERT_TRUE(limiter.OnTraceDone(args, true, 1024 * 1024));
PerfettoCmdState output{};
ASSERT_TRUE(limiter.LoadStateConcrete(&output));
EXPECT_EQ(output.total_bytes_uploaded(), 1024u * 1024u * 3);
EXPECT_EQ(output.first_trace_timestamp(), input.first_trace_timestamp());
EXPECT_EQ(output.last_trace_timestamp(),
static_cast<uint64_t>(args.current_time.count()));
}
TEST(RateLimiterTest, DropBox_FailedToLoadState) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
args.allow_user_build_tracing = true;
args.is_dropbox = true;
WriteGarbageToFile(limiter.GetStateFilePath().c_str());
EXPECT_CALL(limiter, LoadState(_));
EXPECT_CALL(limiter, SaveState(_));
ASSERT_FALSE(limiter.ShouldTrace(args));
PerfettoCmdState output{};
ASSERT_TRUE(limiter.LoadStateConcrete(&output));
EXPECT_EQ(output.total_bytes_uploaded(), 0u);
EXPECT_EQ(output.first_trace_timestamp(), 0u);
EXPECT_EQ(output.last_trace_timestamp(), 0u);
}
TEST(RateLimiterTest, DropBox_NoTimeTravel) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
PerfettoCmdState input{};
input.set_first_trace_timestamp(100);
input.set_last_trace_timestamp(100);
ASSERT_TRUE(limiter.SaveStateConcrete(input));
args.allow_user_build_tracing = true;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(99);
EXPECT_CALL(limiter, LoadState(_));
EXPECT_CALL(limiter, SaveState(_));
ASSERT_FALSE(limiter.ShouldTrace(args));
PerfettoCmdState output{};
ASSERT_TRUE(limiter.LoadStateConcrete(&output));
EXPECT_EQ(output.total_bytes_uploaded(), 0u);
EXPECT_EQ(output.first_trace_timestamp(), 0u);
EXPECT_EQ(output.last_trace_timestamp(), 0u);
}
TEST(RateLimiterTest, DropBox_TooSoon) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
PerfettoCmdState input{};
input.set_first_trace_timestamp(10000);
input.set_last_trace_timestamp(10000);
ASSERT_TRUE(limiter.SaveStateConcrete(input));
args.allow_user_build_tracing = true;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(10000 + 60 * 4);
EXPECT_CALL(limiter, LoadState(_));
ASSERT_FALSE(limiter.ShouldTrace(args));
}
TEST(RateLimiterTest, DropBox_TooMuch) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
PerfettoCmdState input{};
input.set_total_bytes_uploaded(10 * 1024 * 1024 + 1);
ASSERT_TRUE(limiter.SaveStateConcrete(input));
args.allow_user_build_tracing = true;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(60 * 60);
EXPECT_CALL(limiter, LoadState(_));
ASSERT_FALSE(limiter.ShouldTrace(args));
}
TEST(RateLimiterTest, DropBox_TooMuch_Override) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
PerfettoCmdState input{};
input.set_total_bytes_uploaded(10 * 1024 * 1024 + 1);
ASSERT_TRUE(limiter.SaveStateConcrete(input));
args.allow_user_build_tracing = true;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(60 * 60);
args.max_upload_bytes_override = 10 * 1024 * 1024 + 2;
EXPECT_CALL(limiter, LoadState(_));
ASSERT_TRUE(limiter.ShouldTrace(args));
}
TEST(RateLimiterTest, DropBox_TooMuchWasUploaded) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
PerfettoCmdState input{};
input.set_first_trace_timestamp(1);
input.set_last_trace_timestamp(1);
input.set_total_bytes_uploaded(10 * 1024 * 1024 + 1);
ASSERT_TRUE(limiter.SaveStateConcrete(input));
args.is_dropbox = true;
args.current_time = base::TimeSeconds(60 * 60 * 24 + 2);
EXPECT_CALL(limiter, LoadState(_));
ASSERT_TRUE(limiter.ShouldTrace(args));
EXPECT_CALL(limiter, SaveState(_));
ASSERT_TRUE(limiter.OnTraceDone(args, true, 1024 * 1024));
PerfettoCmdState output{};
ASSERT_TRUE(limiter.LoadStateConcrete(&output));
EXPECT_EQ(output.total_bytes_uploaded(), 1024u * 1024u);
EXPECT_EQ(output.first_trace_timestamp(),
static_cast<uint64_t>(args.current_time.count()));
EXPECT_EQ(output.last_trace_timestamp(),
static_cast<uint64_t>(args.current_time.count()));
}
TEST(RateLimiterTest, DropBox_FailedToUpload) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(10000);
EXPECT_CALL(limiter, SaveState(_));
EXPECT_CALL(limiter, LoadState(_));
ASSERT_TRUE(limiter.ShouldTrace(args));
ASSERT_FALSE(limiter.OnTraceDone(args, false, 1024 * 1024));
}
TEST(RateLimiterTest, DropBox_FailedToSave) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(10000);
EXPECT_CALL(limiter, SaveState(_));
EXPECT_CALL(limiter, LoadState(_));
ASSERT_TRUE(limiter.ShouldTrace(args));
EXPECT_CALL(limiter, SaveState(_)).WillOnce(Return(false));
ASSERT_FALSE(limiter.OnTraceDone(args, true, 1024 * 1024));
}
TEST(RateLimiterTest, DropBox_CantTraceOnUser) {
StrictMock<MockRateLimiter> limiter;
RateLimiter::Args args;
args.allow_user_build_tracing = false;
args.is_dropbox = true;
args.current_time = base::TimeSeconds(10000);
EXPECT_CALL(limiter, SaveState(_));
EXPECT_CALL(limiter, LoadState(_));
#if PERFETTO_BUILDFLAG(PERFETTO_ANDROID_USERDEBUG_BUILD) || \
PERFETTO_BUILDFLAG(PERFETTO_STANDALONE_BUILD)
ASSERT_TRUE(limiter.ShouldTrace(args));
#else
// Assuming the only way to test on a userbuild is to build the tests in tree
// targeting a user build.
ASSERT_FALSE(limiter.ShouldTrace(args));
#endif
}
} // namespace
} // namespace perfetto