/* * Copyright (c) 2016-2016, Roland Bock * 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. * * 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 HOLDER 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. */ #include #include "MockDb.h" #include "Sample.h" #include namespace { constexpr auto t = test::TabBar{}; constexpr auto f = test::TabFoo{}; template void print_type_on_error(std::true_type) { } template void print_type_on_error(std::false_type) { T::_print_me_; } template void join_static_check(const Lhs& lhs, const Rhs& rhs) { using CheckResult = sqlpp::check_pre_join_t; using ExpectedCheckResult = std::is_same; print_type_on_error(ExpectedCheckResult{}); static_assert(ExpectedCheckResult::value, "Unexpected check result"); using JoinType = decltype(sqlpp::join(lhs, rhs)); using InnerJoinType = decltype(sqlpp::inner_join(lhs, rhs)); using LeftOuterJoinType = decltype(sqlpp::left_outer_join(lhs, rhs)); using RightOuterJoinType = decltype(sqlpp::right_outer_join(lhs, rhs)); using OuterJoinType = decltype(sqlpp::outer_join(lhs, rhs)); using CrossJoinType = decltype(sqlpp::cross_join(lhs, rhs)); using ExpectedReturnType = sqlpp::logic::all_t< (Assert::value and sqlpp::is_pre_join_t::value and sqlpp::is_pre_join_t::value and sqlpp::is_pre_join_t::value and sqlpp::is_pre_join_t::value and sqlpp::is_pre_join_t::value and sqlpp::is_join_t::value) xor (std::is_same::value and std::is_same::value and std::is_same::value and std::is_same::value and std::is_same::value and std::is_same::value)>; print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); static_assert(ExpectedReturnType::value, "Unexpected return type"); } template void on_static_check(const Lhs& lhs, const Rhs& rhs) { using CheckResult = sqlpp::check_join_on_t; using ExpectedCheckResult = std::is_same; print_type_on_error(ExpectedCheckResult{}); static_assert(ExpectedCheckResult::value, "Unexpected check result"); using ResultType = decltype(lhs.on(rhs)); using ExpectedReturnType = sqlpp::logic::all_t<(Assert::value and sqlpp::is_join_t::value) xor std::is_same::value>; print_type_on_error(ExpectedReturnType{}); static_assert(ExpectedReturnType::value, "Unexpected return type"); } void static_join() { // Prepare a few table aliases for tests const auto ta = t.as(sqlpp::alias::a); const auto tb = t.as(sqlpp::alias::b); const auto fa = f.as(sqlpp::alias::a); const auto fb = f.as(sqlpp::alias::b); // OK: Join two different tables join_static_check(t, f); join_static_check(t, fa); join_static_check(ta, fb); // OK: Self join join_static_check(ta, tb); join_static_check(t, tb); join_static_check(ta, t); // Prepare a join for tests: const auto j = cross_join(ta, tb); // OK: Add a third table join_static_check(j, f); join_static_check(j, t.as(sqlpp::alias::c)); join_static_check(j, t); // Try a bunch of non-tables join_static_check(t, 7); join_static_check(t, t.alpha); join_static_check(t, t.beta); join_static_check(t, t.gamma); join_static_check(t, t.delta); join_static_check(7, t); join_static_check(t.alpha, t); join_static_check(t.beta, t); join_static_check(t.gamma, t); join_static_check(t.delta, t); // Try to join with join (rhs) join_static_check(t, j); join_static_check(f, j); join_static_check(t.as(sqlpp::alias::left), j); // Try to join identical table names join_static_check(t, t); join_static_check(f, f); join_static_check(t.as(f), f); join_static_check(t, f.as(t)); join_static_check(ta, fa); join_static_check(j, fa); join_static_check(j, fb); join_static_check(j, ta); join_static_check(j, tb); // Prepare a pre_joins for tests: const auto t_f = join(t, f); const auto f_t = join(f, t); const auto t_t = join(ta, tb); const auto f_f = join(fa, fb); // OK join.on() on_static_check(t_f, t.alpha > f.omega); on_static_check(f_t, t.alpha < f.omega); on_static_check(f_f, fa.omega == fb.omega); on_static_check(t_t, ta.alpha == tb.alpha); on_static_check(t_f, t.gamma); // Try join.on(non-expression) on_static_check(t_f, true); on_static_check(t_f, 7); on_static_check(t_f, t); // Try join.on(non-boolean) on_static_check(t_f, t.alpha); on_static_check(t_f, t.beta); on_static_check(t_f, f.omega); // Try join.on(foreign-table) on_static_check(t_f, ta.alpha != 0); on_static_check(t_t, t.gamma); on_static_check(f_f, f.omega > fa.omega); } template void join_dynamic_check(const Table& table) { using CheckResult = sqlpp::check_dynamic_pre_join_t; using ExpectedCheckResult = std::is_same; print_type_on_error(ExpectedCheckResult{}); static_assert(ExpectedCheckResult::value, "Unexpected check result"); using JoinType = decltype(sqlpp::dynamic_join(table)); using InnerJoinType = decltype(sqlpp::dynamic_inner_join(table)); using LeftOuterJoinType = decltype(sqlpp::dynamic_left_outer_join(table)); using RightOuterJoinType = decltype(sqlpp::dynamic_right_outer_join(table)); using OuterJoinType = decltype(sqlpp::dynamic_outer_join(table)); using CrossJoinType = decltype(sqlpp::dynamic_cross_join(table)); using ExpectedReturnType = sqlpp::logic::all_t<(Assert::value and sqlpp::is_dynamic_pre_join_t::value and sqlpp::is_dynamic_pre_join_t::value and sqlpp::is_dynamic_pre_join_t::value and sqlpp::is_dynamic_pre_join_t::value and sqlpp::is_dynamic_pre_join_t::value and sqlpp::is_dynamic_join_t::value) xor (std::is_same::value and std::is_same::value and std::is_same::value and std::is_same::value and std::is_same::value and std::is_same::value)>; print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); print_type_on_error(ExpectedReturnType{}); static_assert(ExpectedReturnType::value, "Unexpected return type"); } template void on_dynamic_check(const Lhs& lhs, const Rhs& rhs) { using CheckResult = sqlpp::check_dynamic_join_on_t; using ExpectedCheckResult = std::is_same; print_type_on_error(ExpectedCheckResult{}); static_assert(ExpectedCheckResult::value, "Unexpected check result"); using ResultType = decltype(lhs.on(rhs)); using ExpectedReturnType = sqlpp::logic::all_t<(Assert::value and sqlpp::is_dynamic_join_t::value) xor std::is_same::value>; print_type_on_error(ExpectedReturnType{}); static_assert(ExpectedReturnType::value, "Unexpected return type"); } void dynamic_join() { // Prepare a few table aliases for tests const auto ta = t.as(sqlpp::alias::a); const auto fa = f.as(sqlpp::alias::a); // OK join_dynamic_check(t); join_dynamic_check(f); join_dynamic_check(ta); join_dynamic_check(fa); join_dynamic_check(sqlpp::verbatim_table("tab_sample")); join_dynamic_check(sqlpp::verbatim_table("tab_sample").as(sqlpp::alias::a)); // Try a bunch of non-tables join_dynamic_check(7); join_dynamic_check(t.alpha); join_dynamic_check(t.beta); join_dynamic_check(t.gamma); join_dynamic_check(t.delta); // Try (pre) joins join_dynamic_check(t.join(f)); join_dynamic_check(t.cross_join(f)); // Prepare a dynamic_pre_joins for tests: const auto tj = dynamic_join(t); const auto fj = dynamic_join(f); const auto vj = dynamic_join(sqlpp::verbatim_table("tab_sample")); // OK dynamic_join.on() on_dynamic_check(tj, t.alpha > f.omega); on_dynamic_check(fj, t.alpha < f.omega); // Try dynamic_join.on(non-expression) on_dynamic_check(tj, true); on_dynamic_check(tj, 7); on_dynamic_check(tj, t); // Try dynamic_join.on(non-boolean) on_dynamic_check(tj, t.alpha); on_dynamic_check(tj, t.beta); on_dynamic_check(tj, f.omega); // OK dynamic_join.on(foreign-table) on_dynamic_check(tj, ta.alpha != 0); on_dynamic_check(tj, t.gamma); on_dynamic_check(tj, f.omega > fa.omega); on_dynamic_check(vj, t.alpha < f.omega); } } int main(int, char* []) { static_join(); dynamic_join(); }