/*************************************************************************** * Copyright (c) Wolf Vollprecht, Johan Mabille and Sylvain Corlay * * Copyright (c) QuantStack * * * * Distributed under the terms of the BSD 3-Clause License. * * * * The full license is in the file LICENSE, distributed with this software. * ****************************************************************************/ #include "gtest/gtest.h" #include "xtensor-python/pyarray.hpp" #include "xtensor/containers/xarray.hpp" #include "xtensor/views/xview.hpp" #include "test_common.hpp" namespace xt { using container_type = std::vector; template using ndarray = pyarray; void test1 (ndarrayconst& x) { ndarray y = x; ndarray z = xt::zeros({10}); } double compute(ndarray const& xs) { auto v = xt::view (xs, 0, xt::all()); return v(0); } TEST(pyarray, initializer_constructor) { pyarray r {{{ 0, 1, 2}, { 3, 4, 5}, { 6, 7, 8}}, {{ 9, 10, 11}, {12, 13, 14}, {15, 16, 17}}}; EXPECT_EQ(r.layout(), xt::layout_type::row_major); EXPECT_EQ(r.dimension(), 3); EXPECT_EQ(r(0, 0, 1), 1); EXPECT_EQ(r.shape()[0], 2); pyarray c {{{ 0, 1, 2}, { 3, 4, 5}, { 6, 7, 8}}, {{ 9, 10, 11}, {12, 13, 14}, {15, 16, 17}}}; EXPECT_EQ(c.layout(), xt::layout_type::column_major); EXPECT_EQ(c.dimension(), 3); EXPECT_EQ(c(0, 0, 1), 1); EXPECT_EQ(c.shape()[0], 2); pyarray d {{{ 0, 1, 2}, { 3, 4, 5}, { 6, 7, 8}}, {{ 9, 10, 11}, {12, 13, 14}, {15, 16, 17}}}; EXPECT_EQ(d.layout(), xt::layout_type::row_major); EXPECT_EQ(d.dimension(), 3); EXPECT_EQ(d(0, 0, 1), 1); EXPECT_EQ(d.shape()[0], 2); } TEST(pyarray, expression) { pyarray a = xt::empty({}); EXPECT_EQ(a.layout(), xt::layout_type::row_major); EXPECT_EQ(a.dimension(), 0); EXPECT_EQ(a.size(), 1); pyarray b = xt::empty({5}); EXPECT_EQ(b.layout(), xt::layout_type::row_major); EXPECT_EQ(b.dimension(), 1); EXPECT_EQ(b.size(), 5); pyarray c = xt::empty({5, 3}); EXPECT_EQ(c.layout(), xt::layout_type::row_major); EXPECT_EQ(c.dimension(), 2); EXPECT_EQ(c.size(), 15); EXPECT_EQ(c.shape(0), 5); EXPECT_EQ(c.shape(1), 3); } TEST(pyarray, shaped_constructor) { { SCOPED_TRACE("row_major constructor"); row_major_result<> rm; pyarray ra(rm.m_shape); compare_shape(ra, rm); EXPECT_EQ(layout_type::row_major, ra.layout()); } { SCOPED_TRACE("column_major constructor"); column_major_result<> cm; pyarray ca(cm.m_shape, layout_type::column_major); compare_shape(ca, cm); EXPECT_EQ(layout_type::column_major, ca.layout()); } } TEST(pyarray, from_shape) { auto arr = pyarray::from_shape({5, 2, 6}); auto exp_shape = std::vector{5, 2, 6}; EXPECT_TRUE(std::equal(arr.shape().begin(), arr.shape().end(), exp_shape.begin())); EXPECT_EQ(arr.shape().size(), 3); EXPECT_EQ(arr.size(), 5 * 2 * 6); } TEST(pyarray, strided_constructor) { central_major_result<> cmr; pyarray cma(cmr.m_shape, cmr.m_strides); compare_shape(cma, cmr); } TEST(pyarray, valued_constructor) { { SCOPED_TRACE("row_major valued constructor"); row_major_result<> rm; int value = 2; pyarray ra(rm.m_shape, value); compare_shape(ra, rm); std::vector vec(ra.size(), value); EXPECT_TRUE(std::equal(vec.cbegin(), vec.cend(), ra.storage().cbegin())); } { SCOPED_TRACE("column_major valued constructor"); column_major_result<> cm; int value = 2; pyarray ca(cm.m_shape, value, layout_type::column_major); compare_shape(ca, cm); std::vector vec(ca.size(), value); EXPECT_TRUE(std::equal(vec.cbegin(), vec.cend(), ca.storage().cbegin())); } } TEST(pyarray, strided_valued_constructor) { central_major_result<> cmr; int value = 2; pyarray cma(cmr.m_shape, cmr.m_strides, value); compare_shape(cma, cmr); std::vector vec(cma.size(), value); EXPECT_TRUE(std::equal(vec.cbegin(), vec.cend(), cma.storage().cbegin())); } TEST(pyarray, copy_semantic) { central_major_result<> res; int value = 2; pyarray a(res.m_shape, res.m_strides, value); { SCOPED_TRACE("copy constructor"); pyarray b(a); compare_shape(a, b); EXPECT_EQ(a.storage(), b.storage()); a.data()[0] += 1; EXPECT_NE(a.storage()[0], b.storage()[0]); } { SCOPED_TRACE("assignment operator"); row_major_result<> r; pyarray c(r.m_shape, 0); EXPECT_NE(a.storage(), c.storage()); c = a; compare_shape(a, c); EXPECT_EQ(a.storage(), c.storage()); a.data()[0] += 1; EXPECT_NE(a.storage()[0], c.storage()[0]); } } TEST(pyarray, move_semantic) { central_major_result<> res; int value = 2; pyarray a(res.m_shape, res.m_strides, value); { SCOPED_TRACE("move constructor"); pyarray tmp(a); pyarray b(std::move(tmp)); compare_shape(a, b); EXPECT_EQ(a.storage(), b.storage()); } { SCOPED_TRACE("move assignment"); row_major_result<> r; pyarray c(r.m_shape, 0); EXPECT_NE(a.storage(), c.storage()); pyarray tmp(a); c = std::move(tmp); compare_shape(a, c); EXPECT_EQ(a.storage(), c.storage()); } } TEST(pyarray, extended_constructor) { xt::xarray a1 = { { 1, 2 },{ 3, 4 } }; xt::xarray a2 = { { 1, 2 },{ 3, 4 } }; pyarray c = a1 + a2; EXPECT_EQ(c(0, 0), a1(0, 0) + a2(0, 0)); EXPECT_EQ(c(0, 1), a1(0, 1) + a2(0, 1)); EXPECT_EQ(c(1, 0), a1(1, 0) + a2(1, 0)); EXPECT_EQ(c(1, 1), a1(1, 1) + a2(1, 1)); pyarray d = a1 + a2; EXPECT_EQ(d(0, 0), a1(0, 0) + a2(0, 0)); EXPECT_EQ(d(0, 1), a1(0, 1) + a2(0, 1)); EXPECT_EQ(d(1, 0), a1(1, 0) + a2(1, 0)); EXPECT_EQ(d(1, 1), a1(1, 1) + a2(1, 1)); pyarray e = a1 + a2; EXPECT_EQ(e(0, 0), a1(0, 0) + a2(0, 0)); EXPECT_EQ(e(0, 1), a1(0, 1) + a2(0, 1)); EXPECT_EQ(e(1, 0), a1(1, 0) + a2(1, 0)); EXPECT_EQ(e(1, 1), a1(1, 1) + a2(1, 1)); } TEST(pyarray, resize) { pyarray a; test_resize(a); pyarray b = { {1, 2}, {3, 4} }; a.resize(b.shape()); EXPECT_EQ(a.shape(), b.shape()); } TEST(pyarray, transpose) { pyarray a; test_transpose(a); } TEST(pyarray, access) { pyarray a; test_access(a); } TEST(pyarray, indexed_access) { pyarray a; test_indexed_access(a); } TEST(pyarray, broadcast_shape) { pyarray a; test_broadcast(a); test_broadcast2(a); } TEST(pyarray, iterator) { pyarray a; pyarray b; test_iterator(a, b); pyarray c; bool truthy = std::is_same::value; EXPECT_TRUE(truthy); } TEST(pyarray, initializer_list) { pyarray a0(1); pyarray a1({1, 2}); pyarray a2({{1, 2}, {2, 4}, {5, 6}}); EXPECT_EQ(1, a0()); EXPECT_EQ(2, a1(1)); EXPECT_EQ(4, a2(1, 1)); } TEST(pyarray, zerod) { pyarray a; EXPECT_EQ(0, a()); } TEST(pyarray, reshape) { pyarray a = {{1,2,3}, {4,5,6}}; auto ptr = a.data(); a.reshape({1, 6}); std::vector sc1({1, 6}); EXPECT_TRUE(std::equal(sc1.begin(), sc1.end(), a.shape().begin()) && a.shape().size() == 2); EXPECT_EQ(ptr, a.data()); a.reshape({6}); std::vector sc2 = {6}; EXPECT_TRUE(std::equal(sc2.begin(), sc2.end(), a.shape().begin()) && a.shape().size() == 1); EXPECT_EQ(ptr, a.data()); } TEST(pyarray, view) { xt::pyarray arr = xt::zeros({ 10 }); auto v = xt::view(arr, xt::all()); EXPECT_EQ(v(0), 0.); } TEST(pyarray, zerod_copy) { xt::pyarray arr = 2; xt::pyarray arr2(arr); EXPECT_EQ(arr(), arr2()); } }