/*
 * Copyright (C) 2017 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 specic language governing permissions and
 * limitations under the License.
 */

#include <algorithm>
#include <thread>

#include <android-base/file.h>

#include <gtest/gtest.h>

#include "perfmgr/FileNode.h"
#include "perfmgr/NodeLooperThread.h"

namespace android {
namespace perfmgr {

using namespace std::chrono_literals;

constexpr auto kSLEEP_TOLERANCE_MS = 50ms;

class NodeLooperThreadTest : public ::testing::Test {
  protected:
    virtual void SetUp() {
        std::unique_ptr<TemporaryFile> tf = std::make_unique<TemporaryFile>();
        nodes_.emplace_back(new FileNode(
            "n0", tf->path, {{"n0_value0"}, {"n0_value1"}, {"n0_value2"}}, 2,
            false));
        files_.emplace_back(std::move(tf));
        tf = std::make_unique<TemporaryFile>();
        nodes_.emplace_back(new FileNode(
            "n1", tf->path, {{"n1_value0"}, {"n1_value1"}, {"n1_value2"}}, 2,
            true));
        files_.emplace_back(std::move(tf));
    }

    virtual void TearDown() {
        nodes_.clear();
        files_.clear();
    }
    std::vector<std::unique_ptr<Node>> nodes_;
    std::vector<std::unique_ptr<TemporaryFile>> files_;
};

static inline void _VerifyPathValue(const std::string& path,
                                    const std::string& value) {
    std::string s;
    EXPECT_TRUE(android::base::ReadFileToString(path, &s)) << strerror(errno);
    EXPECT_EQ(value, s);
}

// Test default value init
TEST_F(NodeLooperThreadTest, InitRunTest) {
    sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    EXPECT_TRUE(th->isRunning());
    _VerifyPathValue(files_[0]->path, "");
    _VerifyPathValue(files_[1]->path, "n1_value2");
    th->Stop();
    EXPECT_FALSE(th->isRunning());
}

// Test add request
TEST_F(NodeLooperThreadTest, AddRequest) {
    sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_));
    EXPECT_TRUE(th->isRunning());
    // Dummy LAUNCH boost actions:
    // Node0, value0, 200ms
    // Node1, value1, 400ms
    std::vector<NodeAction> actions{{0, 0, 200ms}, {1, 1, 400ms}};
    EXPECT_TRUE(th->Request(actions, "LAUNCH"));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    _VerifyPathValue(files_[0]->path, "n0_value0");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    std::this_thread::sleep_for(200ms);
    _VerifyPathValue(files_[0]->path, "n0_value2");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    std::this_thread::sleep_for(200ms);
    _VerifyPathValue(files_[0]->path, "n0_value2");
    _VerifyPathValue(files_[1]->path, "n1_value2");
    th->Stop();
    EXPECT_FALSE(th->isRunning());
}

// Test request to override expire time
TEST_F(NodeLooperThreadTest, AddRequestOverride) {
    sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_));
    EXPECT_TRUE(th->isRunning());
    // Dummy LAUNCH boost actions:
    // Node0, value0, 200ms
    // Node1, value1, 500ms
    std::vector<NodeAction> actions{{0, 0, 200ms}, {1, 1, 500ms}};
    EXPECT_TRUE(th->Request(actions, "LAUNCH"));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    _VerifyPathValue(files_[0]->path, "n0_value0");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    // Dummy LAUNCH boost actions:
    // Node0, value0, 300ms will extend
    // Node1, value1, 100ms will not extend
    actions = std::vector<NodeAction>{{0, 0, 300ms}, {1, 1, 100ms}};
    EXPECT_TRUE(th->Request(actions, "LAUNCH"));
    std::this_thread::sleep_for(200ms);
    _VerifyPathValue(files_[0]->path, "n0_value0");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    std::this_thread::sleep_for(150ms);
    // Node0 value0 expired
    _VerifyPathValue(files_[0]->path, "n0_value2");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    std::this_thread::sleep_for(150ms);
    _VerifyPathValue(files_[0]->path, "n0_value2");
    _VerifyPathValue(files_[1]->path, "n1_value2");
    th->Stop();
    EXPECT_FALSE(th->isRunning());
}

// Test cancel request
TEST_F(NodeLooperThreadTest, CancelRequest) {
    sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_));
    EXPECT_TRUE(th->isRunning());
    // Dummy LAUNCH boost actions:
    // Node0, value0, forever
    // Node1, value1, forever
    std::vector<NodeAction> actions{{0, 0, 0ms}, {1, 1, 0ms}};
    EXPECT_TRUE(th->Request(actions, "LAUNCH"));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    _VerifyPathValue(files_[0]->path, "n0_value0");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    EXPECT_TRUE(th->Cancel(actions, "LAUNCH"));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    _VerifyPathValue(files_[0]->path, "n0_value2");
    _VerifyPathValue(files_[1]->path, "n1_value2");
    th->Stop();
    EXPECT_FALSE(th->isRunning());
}

// Test multiple request
TEST_F(NodeLooperThreadTest, MultipleRequest) {
    sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_));
    EXPECT_TRUE(th->isRunning());
    // Dummy LAUNCH boost actions:
    // Node0, value1, 800ms
    // Node1, value1, forever
    std::vector<NodeAction> actions_interaction{{0, 1, 800ms}, {1, 1, 0ms}};
    EXPECT_TRUE(th->Request(actions_interaction, "INTERACTION"));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    _VerifyPathValue(files_[0]->path, "n0_value1");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    // Dummy LAUNCH boost actions:
    // Node0, value0, forever
    // Node1, value0, 400ms
    std::vector<NodeAction> actions_launch{{0, 0, 0ms}, {1, 0, 400ms}};
    EXPECT_TRUE(th->Request(actions_launch, "LAUNCH"));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    _VerifyPathValue(files_[0]->path, "n0_value0");
    _VerifyPathValue(files_[1]->path, "n1_value0");
    std::this_thread::sleep_for(400ms);
    // "LAUNCH" node1 expired
    _VerifyPathValue(files_[0]->path, "n0_value0");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    EXPECT_TRUE(th->Cancel(actions_launch, "LAUNCH"));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    // "LAUNCH" canceled
    _VerifyPathValue(files_[0]->path, "n0_value1");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    std::this_thread::sleep_for(400ms);
    // "INTERACTION" node0 expired
    _VerifyPathValue(files_[0]->path, "n0_value2");
    _VerifyPathValue(files_[1]->path, "n1_value1");
    EXPECT_TRUE(th->Cancel(actions_interaction, "INTERACTION"));
    std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS);
    // "INTERACTION" canceled
    _VerifyPathValue(files_[0]->path, "n0_value2");
    _VerifyPathValue(files_[1]->path, "n1_value2");
    th->Stop();
    EXPECT_FALSE(th->isRunning());
}

}  // namespace perfmgr
}  // namespace android