ResInsight/ThirdParty/custom-opm-flowdiagnostics/opm-flowdiagnostics/tests/test_assembledconnections.cpp

480 lines
12 KiB
C++

/*
Copyright 2016 SINTEF ICT, Applied Mathematics.
Copyright 2016 Statoil ASA.
This file is part of the Open Porous Media Project (OPM).
OPM is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
OPM is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with OPM. If not, see <http://www.gnu.org/licenses/>.
*/
#if HAVE_CONFIG_H
#include <config.h>
#endif // HAVE_CONFIG_H
#if HAVE_DYNAMIC_BOOST_TEST
#define BOOST_TEST_DYN_LINK
#endif
#define NVERBOSE
#define BOOST_TEST_MODULE TEST_ASSEMBLED_CONNECTIONS
#include <opm/common/utility/platform_dependent/disable_warnings.h>
#include <boost/test/unit_test.hpp>
#include <opm/common/utility/platform_dependent/reenable_warnings.h>
#include <opm/utility/graph/AssembledConnections.hpp>
#include <exception>
#include <stdexcept>
namespace {
template <class Collection1, class Collection2>
void check_is_close(const Collection1& c1, const Collection2& c2)
{
BOOST_REQUIRE_EQUAL(c1.size(), c2.size());
if (! c1.empty()) {
auto i1 = c1.begin(), e1 = c1.end();
auto i2 = c2.begin();
for (; i1 != e1; ++i1, ++i2) {
BOOST_CHECK_CLOSE(*i1, *i2, 1.0e-10);
}
}
}
} // Namespace Anonymous
BOOST_AUTO_TEST_SUITE(Two_By_Two)
BOOST_AUTO_TEST_CASE (Constructor)
{
auto g = Opm::AssembledConnections{};
}
BOOST_AUTO_TEST_CASE (Zero_To_One)
{
auto g = Opm::AssembledConnections{};
g.addConnection(0, 1);
g.compress(4);
BOOST_CHECK_EQUAL(g.numRows(), 4);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 1, 1, 1, 1 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{ 1 };
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
}
BOOST_AUTO_TEST_CASE (Zero_To_One_Two)
{
auto g = Opm::AssembledConnections{};
g.addConnection(0, 2);
g.addConnection(0, 1);
g.compress(4);
BOOST_CHECK_EQUAL(g.numRows(), 4);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 2, 2, 2, 2 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{ 1, 2 };
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
}
BOOST_AUTO_TEST_CASE (Zero_To_One_Two_Duplicate)
{
auto g = Opm::AssembledConnections{};
for (auto i = 0, n = 3; i < n; ++i) {
g.addConnection(0, 2);
}
g.addConnection(0, 1);
for (auto i = 0, n = 3; i < n; ++i) {
g.addConnection(0, 2);
}
for (auto i = 0, n = 3; i < n; ++i) {
g.addConnection(0, 1);
}
g.compress(4);
BOOST_CHECK_EQUAL(g.numRows(), 4);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 2, 2, 2, 2 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{ 1, 2 };
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
}
BOOST_AUTO_TEST_CASE (No_Out_Edge_From_High_Vertex)
{
// Vertex of highest numerical ID not referenced as source vertex in
// connection list. We must still produce a complete adjacency
// representation that is aware of all vertices when the expected size
// is provided.
auto g = Opm::AssembledConnections{};
g.addConnection(0, 1, 1.0);
g.addConnection(0, 2, 1.0);
g.addConnection(1, 3, 1.0);
g.addConnection(2, 3, 1.0);
g.compress(4);
BOOST_CHECK_EQUAL(g.numRows(), 4);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 2, 3, 4, 4 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{ 1, 2, 3, 3 };
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
{
const auto w = g.connectionWeight();
const auto expect_w = std::vector<double>{
1.0, 1.0, 1.0, 1.0
};
check_is_close(w, expect_w);
}
}
BOOST_AUTO_TEST_CASE (Unexpected_Vertex_ID)
{
// We expect a lower number of vertices than are actually present. This
// is an error.
auto g = Opm::AssembledConnections{};
g.addConnection(0, 1);
g.addConnection(0, 2);
g.addConnection(1, 3);
g.addConnection(3, 2);
BOOST_CHECK_THROW(g.compress(3), std::invalid_argument);
}
BOOST_AUTO_TEST_CASE (Isolated_Node)
{
auto g = Opm::AssembledConnections{};
g.addConnection(2, 0);
g.addConnection(0, 2);
g.addConnection(0, 0);
g.compress(3);
BOOST_CHECK_EQUAL(g.numRows(), 3);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 2, 2, 3 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{ 0, 2, 0 };
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
}
BOOST_AUTO_TEST_CASE (All_To_All)
{
auto g = Opm::AssembledConnections{};
const auto n = 4;
for (auto i = 0*n; i < n; ++i) {
for (auto j = 0*n; j < n; ++j) {
g.addConnection(j, i);
}
}
g.compress(n);
BOOST_CHECK_EQUAL(g.numRows(), n);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 4, 8, 12, 16 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{
0, 1, 2, 3,
0, 1, 2, 3,
0, 1, 2, 3,
0, 1, 2, 3,
};
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
}
BOOST_AUTO_TEST_CASE (All_To_All_Duplicate)
{
auto g = Opm::AssembledConnections{};
const auto n = 4;
for (auto k = 0*n; k < n; ++k) {
for (auto i = 0*n; i < n; ++i) {
for (auto j = 0*n; j < n; ++j) {
g.addConnection(i, i);
g.addConnection(i, j);
g.addConnection(j, i);
g.addConnection(j, j);
}
}
}
g.compress(n);
BOOST_CHECK_EQUAL(g.numRows(), n);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 4, 8, 12, 16 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{
0, 1, 2, 3,
0, 1, 2, 3,
0, 1, 2, 3,
0, 1, 2, 3,
};
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
}
BOOST_AUTO_TEST_CASE (Weighted_Graph_Single)
{
auto g = Opm::AssembledConnections{};
g.addConnection(0, 2, 0.2);
g.addConnection(0, 1, - 0.1);
g.addConnection(1, 3, 13.0);
g.addConnection(2, 3, -23.0);
g.compress(4);
BOOST_CHECK_EQUAL(g.numRows(), 4);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 2, 3, 4, 4 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{ 1, 2, 3, 3 };
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
{
const auto w = g.connectionWeight();
const auto expect_w = std::vector<double>{
- 0.1, 0.2,
13.0,
-23.0,
};
check_is_close(w, expect_w);
}
{
const int num_cells = g.startPointers().size() - 1;
int num_conn = 0;
double weight_sum = 0.0;
for (int cell = 0; cell < num_cells; ++cell) {
for (const auto& conn : g.cellNeighbourhood(cell)) {
++num_conn;
weight_sum += conn.weight;
}
}
BOOST_CHECK_EQUAL(num_conn, 4);
BOOST_CHECK_CLOSE(weight_sum, -9.9, 1e-10);
}
}
BOOST_AUTO_TEST_CASE (Weighted_Graph_Multiple)
{
auto g = Opm::AssembledConnections{};
const auto count = 4;
for (auto i = 0*count; i < count; ++i) {
g.addConnection(0, 2, 0.2);
g.addConnection(0, 1, - 0.1);
g.addConnection(1, 3, 13.0);
g.addConnection(2, 3, -23.0);
}
g.compress(4);
BOOST_CHECK_EQUAL(g.numRows(), 4);
{
const auto start = g.startPointers();
const auto expect_ia = std::vector<int>{ 0, 2, 3, 4, 4 };
BOOST_CHECK_EQUAL_COLLECTIONS(start .begin(), start .end(),
expect_ia.begin(), expect_ia.end());
}
{
const auto neigh = g.neighbourhood();
const auto expect_ja = std::vector<int>{ 1, 2, 3, 3 };
BOOST_CHECK_EQUAL_COLLECTIONS(neigh .begin(), neigh .end(),
expect_ja.begin(), expect_ja.end());
}
{
const auto w = g.connectionWeight();
const auto expect_w = std::vector<double>{
count * (- 0.1), count * 0.2,
count * 13.0 ,
count * (-23.0),
};
check_is_close(w, expect_w);
}
{
const int num_cells = g.startPointers().size() - 1;
int num_conn = 0;
double weight_sum = 0.0;
for (int cell = 0; cell < num_cells; ++cell) {
for (const auto& conn : g.cellNeighbourhood(cell)) {
++num_conn;
weight_sum += conn.weight;
}
}
BOOST_CHECK_EQUAL(num_conn, 4);
BOOST_CHECK_CLOSE(weight_sum, count*(-9.9), 1e-10);
}
}
BOOST_AUTO_TEST_CASE (Compress_Invalid_Throw)
{
auto g = Opm::AssembledConnections{};
g.addConnection(0, 1);
g.addConnection(0, 2, 1.0);
// Can't mix weighted and unweighted edges.
BOOST_CHECK_THROW(g.compress(3), std::logic_error);
}
BOOST_AUTO_TEST_SUITE_END()