/*
 * 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 specific language governing permissions and
 * limitations under the License.
 */

#include <algorithm>
#include <iterator>
#include <iostream>

#include <gtest/gtest.h>

#include "../apdu.h"

using android::CommandApdu;
using android::ResponseApdu;

/* CommandApdu */

TEST(CommandApduTest, Case1) {
    const CommandApdu apdu{1, 2, 3, 4};
    const std::vector<uint8_t> expected{1, 2, 3, 4};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case2s) {
    const CommandApdu apdu{4, 3, 2, 1, 0, 3};
    const std::vector<uint8_t> expected{4, 3, 2, 1, 3};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case2s_maxLe) {
    const CommandApdu apdu{4, 3, 2, 1, 0, 256};
    const std::vector<uint8_t> expected{4, 3, 2, 1, 0};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case2e) {
    const CommandApdu apdu{5, 6, 7, 8, 0, 258};
    const std::vector<uint8_t> expected{5, 6, 7, 8, 0, 1, 2};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case2e_maxLe) {
    const CommandApdu apdu{5, 6, 7, 8, 0, 65536};
    const std::vector<uint8_t> expected{5, 6, 7, 8, 0, 0, 0};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case3s) {
    const CommandApdu apdu{8, 7, 6, 5, 5, 0};
    const std::vector<uint8_t> expected{8, 7, 6, 5, 5, 0, 0, 0, 0, 0};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case3s_data) {
    CommandApdu apdu{8, 7, 6, 5, 3, 0};
    auto it = apdu.dataBegin();
    *it++ = 10;
    *it++ = 11;
    *it++ = 12;
    ASSERT_EQ(apdu.dataEnd(), it);

    const std::vector<uint8_t> expected{8, 7, 6, 5, 3, 10, 11, 12};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case3e) {
    const CommandApdu apdu{8, 7, 6, 5, 256, 0};
    std::vector<uint8_t> expected{8, 7, 6, 5, 0, 1, 0};
    expected.resize(expected.size() + 256, 0);
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case3e_data) {
    CommandApdu apdu{8, 7, 6, 5, 65535, 0};
    ASSERT_EQ(size_t{65535}, apdu.dataSize());
    std::fill(apdu.dataBegin(), apdu.dataEnd(), 7);
    std::vector<uint8_t> expected{8, 7, 6, 5, 0, 255, 255};
    expected.resize(expected.size() + 65535, 7);
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case4s) {
    const CommandApdu apdu{1, 3, 5, 7, 2, 3};
    const std::vector<uint8_t> expected{1, 3, 5, 7, 2, 0, 0, 3};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case4s_data) {
    CommandApdu apdu{1, 3, 5, 7, 1, 90};
    auto it = apdu.dataBegin();
    *it++ = 8;
    ASSERT_EQ(apdu.dataEnd(), it);

    const std::vector<uint8_t> expected{1, 3, 5, 7, 1, 8, 90};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case4s_maxLe) {
    const CommandApdu apdu{1, 3, 5, 7, 2, 256};
    const std::vector<uint8_t> expected{1, 3, 5, 7, 2, 0, 0, 0};
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case4e) {
    const CommandApdu apdu{1, 3, 5, 7, 527, 349};
    std::vector<uint8_t> expected{1, 3, 5, 7, 0, 2, 15};
    expected.resize(expected.size() + 527, 0);
    expected.push_back(1);
    expected.push_back(93);
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

TEST(CommandApduTest, Case4e_maxLe) {
    const CommandApdu apdu{1, 3, 5, 7, 20, 65536};
    std::vector<uint8_t> expected{1, 3, 5, 7, 0, 0, 20};
    expected.resize(expected.size() + 20, 0);
    expected.push_back(0);
    expected.push_back(0);
    ASSERT_EQ(expected.size(), apdu.size());
    ASSERT_TRUE(std::equal(apdu.begin(), apdu.end(), expected.begin(), expected.end()));
}

/* ResponseApdu */

TEST(ResponseApduTest, bad) {
    const std::vector<uint8_t> empty{};
    const ResponseApdu<std::vector<uint8_t>> apdu{empty};
    ASSERT_FALSE(apdu.ok());
}

TEST(ResponseApduTest, statusOnly) {
    const std::vector<uint8_t> statusOnly{0x90, 0x37};
    const ResponseApdu<std::vector<uint8_t>> apdu{statusOnly};
    ASSERT_TRUE(apdu.ok());
    ASSERT_EQ(0x90, apdu.sw1());
    ASSERT_EQ(0x37, apdu.sw2());
    ASSERT_EQ(0x9037, apdu.status());
    ASSERT_EQ(size_t{0}, apdu.dataSize());
}

TEST(ResponseApduTest, data) {
    const std::vector<uint8_t> data{1, 2, 3, 9, 8, 7, 0x3a, 0xbc};
    const ResponseApdu<std::vector<uint8_t>> apdu{data};
    ASSERT_TRUE(apdu.ok());
    ASSERT_EQ(0x3abc, apdu.status());
    ASSERT_EQ(size_t{6}, apdu.dataSize());

    const uint8_t expected[] = {1, 2, 3, 9, 8, 7};
    ASSERT_TRUE(std::equal(apdu.dataBegin(), apdu.dataEnd(),
                           std::begin(expected), std::end(expected)));
}

TEST(ResponseApduTest, remainingBytes) {
    const std::vector<uint8_t> remainingBytes{0x61, 23};
    const ResponseApdu<std::vector<uint8_t>> apdu{remainingBytes};
    ASSERT_EQ(23, apdu.remainingBytes());
    ASSERT_FALSE(apdu.isWarning());
    ASSERT_FALSE(apdu.isExecutionError());
    ASSERT_FALSE(apdu.isCheckingError());
    ASSERT_FALSE(apdu.isError());
}

TEST(ResponseApduTest, warning) {
    const std::vector<uint8_t> warning{0x62, 0};
    const ResponseApdu<std::vector<uint8_t>> apdu{warning};
    ASSERT_TRUE(apdu.isWarning());
    ASSERT_FALSE(apdu.isExecutionError());
    ASSERT_FALSE(apdu.isCheckingError());
    ASSERT_FALSE(apdu.isError());
}

TEST(ResponseApduTest, executionError) {
    const std::vector<uint8_t> executionError{0x66, 0};
    const ResponseApdu<std::vector<uint8_t>> apdu{executionError};
    ASSERT_FALSE(apdu.isWarning());
    ASSERT_TRUE(apdu.isExecutionError());
    ASSERT_FALSE(apdu.isCheckingError());
    ASSERT_TRUE(apdu.isError());
}

TEST(ResponseApduTest, checkingError) {
    const std::vector<uint8_t> checkingError{0x67, 0};
    const ResponseApdu<std::vector<uint8_t>> apdu{checkingError};
    ASSERT_FALSE(apdu.isWarning());
    ASSERT_FALSE(apdu.isExecutionError());
    ASSERT_TRUE(apdu.isCheckingError());
    ASSERT_TRUE(apdu.isError());
}