// Copyright 2018, Google Inc.
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
//     * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
//     * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following disclaimer
// in the documentation and/or other materials provided with the
// distribution.
//     * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

// Google Mock - a framework for writing C++ mock classes.
//
// This file tests the internal preprocessor macro library.
#include "gmock/internal/gmock-pp.h"

#include <string>

#include "gmock/gmock.h"

namespace testing {
namespace {

// Matcher to verify that to strings are identical up to whitespace
// Not 100% correct, because it treats "AB" as equal to "A B".
::testing::Matcher<const std::string&> SameExceptSpaces(const std::string& s) {
  auto remove_spaces = [](std::string to_split) {
    to_split.erase(std::remove(to_split.begin(), to_split.end(), ' '),
                   to_split.end());
    return to_split;
  };
  return ::testing::ResultOf(remove_spaces, remove_spaces(s));
}

// Verify that a macro expands to a given text. Ignores whitespace difference.
// In MSVC, GMOCK_PP_STRINGIZE() returns nothing, rather than "". So concatenate
// with an empty string.
#define EXPECT_EXPANSION(Result, Macro) \
  EXPECT_THAT("" GMOCK_PP_STRINGIZE(Macro), SameExceptSpaces(Result))

TEST(Macros, Cat) {
  EXPECT_EXPANSION("14", GMOCK_PP_CAT(1, 4));
  EXPECT_EXPANSION("+=", GMOCK_PP_CAT(+, =));
}

TEST(Macros, Narg) {
  EXPECT_EXPANSION("1", GMOCK_PP_NARG());
  EXPECT_EXPANSION("1", GMOCK_PP_NARG(x));
  EXPECT_EXPANSION("2", GMOCK_PP_NARG(x, y));
  EXPECT_EXPANSION("3", GMOCK_PP_NARG(x, y, z));
  EXPECT_EXPANSION("4", GMOCK_PP_NARG(x, y, z, w));

  EXPECT_EXPANSION("0", GMOCK_PP_NARG0());
  EXPECT_EXPANSION("1", GMOCK_PP_NARG0(x));
  EXPECT_EXPANSION("2", GMOCK_PP_NARG0(x, y));
}

TEST(Macros, Comma) {
  EXPECT_EXPANSION("0", GMOCK_PP_HAS_COMMA());
  EXPECT_EXPANSION("1", GMOCK_PP_HAS_COMMA(, ));
  EXPECT_EXPANSION("0", GMOCK_PP_HAS_COMMA((, )));
}

TEST(Macros, IsEmpty) {
  EXPECT_EXPANSION("1", GMOCK_PP_IS_EMPTY());
  EXPECT_EXPANSION("0", GMOCK_PP_IS_EMPTY(, ));
  EXPECT_EXPANSION("0", GMOCK_PP_IS_EMPTY(a));
  EXPECT_EXPANSION("0", GMOCK_PP_IS_EMPTY(()));

#define GMOCK_PP_INTERNAL_IS_EMPTY_TEST_1
  EXPECT_EXPANSION("1", GMOCK_PP_IS_EMPTY(GMOCK_PP_INTERNAL_IS_EMPTY_TEST_1));
}

TEST(Macros, If) {
  EXPECT_EXPANSION("1", GMOCK_PP_IF(1, 1, 2));
  EXPECT_EXPANSION("2", GMOCK_PP_IF(0, 1, 2));
}

TEST(Macros, HeadTail) {
  EXPECT_EXPANSION("1", GMOCK_PP_HEAD(1));
  EXPECT_EXPANSION("1", GMOCK_PP_HEAD(1, 2));
  EXPECT_EXPANSION("1", GMOCK_PP_HEAD(1, 2, 3));

  EXPECT_EXPANSION("", GMOCK_PP_TAIL(1));
  EXPECT_EXPANSION("2", GMOCK_PP_TAIL(1, 2));
  EXPECT_EXPANSION("2", GMOCK_PP_HEAD(GMOCK_PP_TAIL(1, 2, 3)));
}

TEST(Macros, Parentheses) {
  EXPECT_EXPANSION("0", GMOCK_PP_IS_BEGIN_PARENS(sss));
  EXPECT_EXPANSION("0", GMOCK_PP_IS_BEGIN_PARENS(sss()));
  EXPECT_EXPANSION("0", GMOCK_PP_IS_BEGIN_PARENS(sss() sss));
  EXPECT_EXPANSION("1", GMOCK_PP_IS_BEGIN_PARENS((sss)));
  EXPECT_EXPANSION("1", GMOCK_PP_IS_BEGIN_PARENS((sss)ss));

  EXPECT_EXPANSION("0", GMOCK_PP_IS_ENCLOSED_PARENS(sss));
  EXPECT_EXPANSION("0", GMOCK_PP_IS_ENCLOSED_PARENS(sss()));
  EXPECT_EXPANSION("0", GMOCK_PP_IS_ENCLOSED_PARENS(sss() sss));
  EXPECT_EXPANSION("1", GMOCK_PP_IS_ENCLOSED_PARENS((sss)));
  EXPECT_EXPANSION("0", GMOCK_PP_IS_ENCLOSED_PARENS((sss)ss));

  EXPECT_EXPANSION("1 + 1", GMOCK_PP_REMOVE_PARENS((1 + 1)));
}

TEST(Macros, Increment) {
  EXPECT_EXPANSION("1", GMOCK_PP_INC(0));
  EXPECT_EXPANSION("2", GMOCK_PP_INC(1));
  EXPECT_EXPANSION("3", GMOCK_PP_INC(2));
  EXPECT_EXPANSION("4", GMOCK_PP_INC(3));
  EXPECT_EXPANSION("5", GMOCK_PP_INC(4));

  EXPECT_EXPANSION("16", GMOCK_PP_INC(15));
}

#define JOINER_CAT(a, b) a##b
#define JOINER(_N, _Data, _Elem) JOINER_CAT(_Data, _N) = _Elem

TEST(Macros, Repeat) {
  EXPECT_EXPANSION("", GMOCK_PP_REPEAT(JOINER, X, 0));
  EXPECT_EXPANSION("X0=", GMOCK_PP_REPEAT(JOINER, X, 1));
  EXPECT_EXPANSION("X0= X1=", GMOCK_PP_REPEAT(JOINER, X, 2));
  EXPECT_EXPANSION("X0= X1= X2=", GMOCK_PP_REPEAT(JOINER, X, 3));
  EXPECT_EXPANSION("X0= X1= X2= X3=", GMOCK_PP_REPEAT(JOINER, X, 4));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4=", GMOCK_PP_REPEAT(JOINER, X, 5));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4= X5=", GMOCK_PP_REPEAT(JOINER, X, 6));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4= X5= X6=",
                   GMOCK_PP_REPEAT(JOINER, X, 7));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4= X5= X6= X7=",
                   GMOCK_PP_REPEAT(JOINER, X, 8));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4= X5= X6= X7= X8=",
                   GMOCK_PP_REPEAT(JOINER, X, 9));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4= X5= X6= X7= X8= X9=",
                   GMOCK_PP_REPEAT(JOINER, X, 10));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4= X5= X6= X7= X8= X9= X10=",
                   GMOCK_PP_REPEAT(JOINER, X, 11));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4= X5= X6= X7= X8= X9= X10= X11=",
                   GMOCK_PP_REPEAT(JOINER, X, 12));
  EXPECT_EXPANSION("X0= X1= X2= X3= X4= X5= X6= X7= X8= X9= X10= X11= X12=",
                   GMOCK_PP_REPEAT(JOINER, X, 13));
  EXPECT_EXPANSION(
      "X0= X1= X2= X3= X4= X5= X6= X7= X8= X9= X10= X11= X12= X13=",
      GMOCK_PP_REPEAT(JOINER, X, 14));
  EXPECT_EXPANSION(
      "X0= X1= X2= X3= X4= X5= X6= X7= X8= X9= X10= X11= X12= X13= X14=",
      GMOCK_PP_REPEAT(JOINER, X, 15));
}
TEST(Macros, ForEach) {
  EXPECT_EXPANSION("", GMOCK_PP_FOR_EACH(JOINER, X, ()));
  EXPECT_EXPANSION("X0=a", GMOCK_PP_FOR_EACH(JOINER, X, (a)));
  EXPECT_EXPANSION("X0=a X1=b", GMOCK_PP_FOR_EACH(JOINER, X, (a, b)));
  EXPECT_EXPANSION("X0=a X1=b X2=c", GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c)));
  EXPECT_EXPANSION("X0=a X1=b X2=c X3=d",
                   GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d)));
  EXPECT_EXPANSION("X0=a X1=b X2=c X3=d X4=e",
                   GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e)));
  EXPECT_EXPANSION("X0=a X1=b X2=c X3=d X4=e X5=f",
                   GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f)));
  EXPECT_EXPANSION("X0=a X1=b X2=c X3=d X4=e X5=f X6=g",
                   GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f, g)));
  EXPECT_EXPANSION("X0=a X1=b X2=c X3=d X4=e X5=f X6=g X7=h",
                   GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f, g, h)));
  EXPECT_EXPANSION("X0=a X1=b X2=c X3=d X4=e X5=f X6=g X7=h X8=i",
                   GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f, g, h, i)));
  EXPECT_EXPANSION(
      "X0=a X1=b X2=c X3=d X4=e X5=f X6=g X7=h X8=i X9=j",
      GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f, g, h, i, j)));
  EXPECT_EXPANSION(
      "X0=a X1=b X2=c X3=d X4=e X5=f X6=g X7=h X8=i X9=j X10=k",
      GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f, g, h, i, j, k)));
  EXPECT_EXPANSION(
      "X0=a X1=b X2=c X3=d X4=e X5=f X6=g X7=h X8=i X9=j X10=k X11=l",
      GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f, g, h, i, j, k, l)));
  EXPECT_EXPANSION(
      "X0=a X1=b X2=c X3=d X4=e X5=f X6=g X7=h X8=i X9=j X10=k X11=l X12=m",
      GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f, g, h, i, j, k, l, m)));
  EXPECT_EXPANSION(
      "X0=a X1=b X2=c X3=d X4=e X5=f X6=g X7=h X8=i X9=j X10=k X11=l X12=m "
      "X13=n",
      GMOCK_PP_FOR_EACH(JOINER, X, (a, b, c, d, e, f, g, h, i, j, k, l, m, n)));
  EXPECT_EXPANSION(
      "X0=a X1=b X2=c X3=d X4=e X5=f X6=g X7=h X8=i X9=j X10=k X11=l X12=m "
      "X13=n X14=o",
      GMOCK_PP_FOR_EACH(JOINER, X,
                        (a, b, c, d, e, f, g, h, i, j, k, l, m, n, o)));
}

}  // namespace
}  // namespace testing