/*############################################################################
# Copyright 2017 Intel Corporation
#
# 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.
############################################################################*/
/// Unit tests for OneTimePad.
/*! \file */

#include <gtest/gtest.h>

#include "epid/member/tiny/math/unittests/onetimepad.h"

namespace {

TEST(OneTimePadTest, GenerateFailsWhenDefaultConstructedWithoutInit) {
  OneTimePad otp;
  EXPECT_EQ(0u, otp.BitsConsumed());
  std::vector<unsigned int> actual({0, 0});
  EXPECT_NE(0, OneTimePad::Generate(actual.data(), 8, &otp));
}

TEST(OneTimePadTest, GeneratesCorrectDataWhenConstructedWithUint8s) {
  std::vector<unsigned int> actual1({0, 0});
  std::vector<unsigned int> actual2({0, 0});
  const std::vector<unsigned int> expected1({0x07050301, 0});
  const std::vector<unsigned int> expected2({0x0e0c0a09, 0});
  OneTimePad otp({0x01, 0x03, 0x05, 0x07, 0x09, 0x0a, 0x0c, 0x0e});
  EXPECT_EQ(0u, otp.BitsConsumed());
  EXPECT_EQ(0, OneTimePad::Generate(actual1.data(), 32, &otp));
  EXPECT_EQ(expected1, actual1);
  EXPECT_EQ(32u, otp.BitsConsumed());
  EXPECT_EQ(0, OneTimePad::Generate(actual2.data(), 32, &otp));
  EXPECT_EQ(expected2, actual2);
  EXPECT_EQ(64u, otp.BitsConsumed());
}

TEST(OneTimePadTest, GeneratesCorrectDataWhenInitilizedWithUint8s) {
  std::vector<unsigned int> actual1({0, 0});
  std::vector<unsigned int> actual2({0, 0});
  const std::vector<unsigned int> expected1({0x07050301, 0});
  const std::vector<unsigned int> expected2({0x0e0c0a09, 0});
  OneTimePad otp;
  otp.InitUint8({0x01, 0x03, 0x05, 0x07, 0x09, 0x0a, 0x0c, 0x0e});
  EXPECT_EQ(0u, otp.BitsConsumed());
  EXPECT_EQ(0, OneTimePad::Generate(actual1.data(), 32, &otp));
  EXPECT_EQ(expected1, actual1);
  EXPECT_EQ(32u, otp.BitsConsumed());
  EXPECT_EQ(0, OneTimePad::Generate(actual2.data(), 32, &otp));
  EXPECT_EQ(expected2, actual2);
  EXPECT_EQ(64u, otp.BitsConsumed());
}
TEST(OneTimePadTest, GeneratesCorrectDataWhenInitilizedWithUint32s) {
  std::vector<uint32_t> actual({0x00});
  const std::vector<uint32_t> expected({0x01});
  std::vector<uint32_t> actual2({0x00});
  const std::vector<uint32_t> expected2({0x01020304});
  OneTimePad otp({0x01, 0x03, 0x05, 0x07, 0x09, 0x0a, 0x0c, 0x0e});
  otp.InitUint32({0x01, 0x01020304});
  EXPECT_EQ(0u, otp.BitsConsumed());
  EXPECT_EQ(0, OneTimePad::Generate(actual.data(), 32, &otp));
  EXPECT_EQ(32u, otp.BitsConsumed());
  EXPECT_EQ(expected, actual);
  EXPECT_EQ(0, OneTimePad::Generate(actual2.data(), 32, &otp));
  EXPECT_EQ(expected2, actual2);
  EXPECT_EQ(64u, otp.BitsConsumed());
}
TEST(OneTimePadTest, GeneratesSingleBytesCorrectly) {
  OneTimePad otp({0x01, 0x03, 0x05, 0x07, 0x09, 0x0a, 0x0c, 0x0e});
  std::vector<uint8_t> expected1({0x01, 0x00, 0x00, 0x00});
  std::vector<uint8_t> expected2({0x03, 0x00, 0x00, 0x00});
  std::vector<uint8_t> expected3({0x05, 0x00, 0x00, 0x00});
  std::vector<uint8_t> actual({0, 0, 0, 0});
  EXPECT_EQ(0, OneTimePad::Generate((uint32_t*)actual.data(), 8, &otp));
  EXPECT_EQ(8u, otp.BitsConsumed());
  EXPECT_EQ(expected1, actual);
  EXPECT_EQ(0, OneTimePad::Generate((uint32_t*)actual.data(), 8, &otp));
  EXPECT_EQ(16u, otp.BitsConsumed());
  EXPECT_EQ(expected2, actual);
  EXPECT_EQ(0, OneTimePad::Generate((uint32_t*)actual.data(), 8, &otp));
  EXPECT_EQ(24u, otp.BitsConsumed());
  EXPECT_EQ(expected3, actual);
}

TEST(OneTimePadTest, GenerateRejectsNullPtr) {
  OneTimePad otp(8);
  EXPECT_NE(0, OneTimePad::Generate(nullptr, 32, &otp));
}
TEST(OneTimePadTest, GenerateRejectsNegativeBits) {
  OneTimePad otp(8);
  std::vector<unsigned int> actual({0, 0});
  EXPECT_NE(0, OneTimePad::Generate(actual.data(), -32, &otp));
}
TEST(OneTimePadTest, GenerateRejectsZeroBits) {
  OneTimePad otp(8);
  std::vector<unsigned int> actual({0, 0});
  EXPECT_NE(0, OneTimePad::Generate(actual.data(), 0, &otp));
}
TEST(OneTimePadTest, GenerateRejectsTooLargeRequest) {
  OneTimePad otp(8);
  std::vector<unsigned int> actual({0, 0});
  EXPECT_EQ(0, OneTimePad::Generate(actual.data(), 32, &otp));
  EXPECT_EQ(0, OneTimePad::Generate(actual.data(), 32, &otp));
  EXPECT_NE(0, OneTimePad::Generate(actual.data(), 32, &otp));
}

TEST(OneTimePadTest, GenerateRejectsUnsupportedBitRequest) {
  OneTimePad otp(8);
  std::vector<unsigned int> actual({0, 0});
  EXPECT_NE(0, OneTimePad::Generate(actual.data(), 31, &otp));
}

TEST(OneTimePadTest, GenerateCoddlesDevelopersWhoDoNotCheckReturnValues) {
  OneTimePad otp(4);
  uint32_t word;
  EXPECT_EQ(0, OneTimePad::Generate(&word, 32, &otp));
  EXPECT_NE(0, OneTimePad::Generate(&word, 32, &otp));
  EXPECT_THROW(OneTimePad::Generate(&word, 32, &otp), std::runtime_error);
}

}  // namespace