/*
* 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());
}