/* * Copyright 2018 Uber Technologies, Inc. * * 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. */ /** @file * @brief tests H3 distance function. * * usage: `testH3Distance` */ #include #include #include #include "algos.h" #include "baseCells.h" #include "constants.h" #include "h3Index.h" #include "h3api.h" #include "localij.h" #include "stackAlloc.h" #include "test.h" #include "utility.h" static const int MAX_DISTANCES[] = {1, 2, 5, 12, 19, 26}; static void h3Distance_identity_assertions(H3Index h3) { t_assert(H3_EXPORT(h3Distance)(h3, h3) == 0, "distance to self is 0"); } static void h3Distance_kRing_assertions(H3Index h3) { int r = H3_GET_RESOLUTION(h3); t_assert(r <= 5, "resolution supported by test function (kRing)"); int maxK = MAX_DISTANCES[r]; int sz = H3_EXPORT(maxKringSize)(maxK); STACK_ARRAY_CALLOC(H3Index, neighbors, sz); STACK_ARRAY_CALLOC(int, distances, sz); H3_EXPORT(kRingDistances)(h3, maxK, neighbors, distances); for (int i = 0; i < sz; i++) { if (neighbors[i] == 0) { continue; } int calculatedDistance = H3_EXPORT(h3Distance)(h3, neighbors[i]); // Don't consider indexes where h3Distance reports failure to // generate t_assert(calculatedDistance == distances[i] || calculatedDistance == -1, "kRingDistances matches h3Distance"); } } SUITE(h3Distance) { // Some indexes that represent base cells. Base cells // are hexagons except for `pent1`. H3Index bc1 = H3_INIT; setH3Index(&bc1, 0, 15, 0); H3Index bc2 = H3_INIT; setH3Index(&bc2, 0, 8, 0); H3Index bc3 = H3_INIT; setH3Index(&bc3, 0, 31, 0); H3Index pent1 = H3_INIT; setH3Index(&pent1, 0, 4, 0); TEST(testIndexDistance) { H3Index bc = 0; setH3Index(&bc, 1, 17, 0); H3Index p = 0; setH3Index(&p, 1, 14, 0); H3Index p2; setH3Index(&p2, 1, 14, 2); H3Index p3; setH3Index(&p3, 1, 14, 3); H3Index p4; setH3Index(&p4, 1, 14, 4); H3Index p5; setH3Index(&p5, 1, 14, 5); H3Index p6; setH3Index(&p6, 1, 14, 6); t_assert(H3_EXPORT(h3Distance)(bc, p) == 3, "distance onto pentagon"); t_assert(H3_EXPORT(h3Distance)(bc, p2) == 2, "distance onto p2"); t_assert(H3_EXPORT(h3Distance)(bc, p3) == 3, "distance onto p3"); // TODO works correctly but is rejected due to possible pentagon // distortion. // t_assert(H3_EXPORT(h3Distance)(bc, p4) == 3, "distance onto p4"); // t_assert(H3_EXPORT(h3Distance)(bc, p5) == 4, "distance onto p5"); t_assert(H3_EXPORT(h3Distance)(bc, p6) == 2, "distance onto p6"); } TEST(testIndexDistance2) { H3Index origin = 0x820c4ffffffffffL; // Destination is on the other side of the pentagon H3Index destination = 0x821ce7fffffffffL; // TODO doesn't work because of pentagon distortion. Both should be 5. t_assert(H3_EXPORT(h3Distance)(destination, origin) == -1, "distance in res 2 across pentagon"); t_assert(H3_EXPORT(h3Distance)(origin, destination) == -1, "distance in res 2 across pentagon (reversed)"); } TEST(h3Distance_identity) { iterateAllIndexesAtRes(0, h3Distance_identity_assertions); iterateAllIndexesAtRes(1, h3Distance_identity_assertions); iterateAllIndexesAtRes(2, h3Distance_identity_assertions); } TEST(h3Distance_kRing) { iterateAllIndexesAtRes(0, h3Distance_kRing_assertions); iterateAllIndexesAtRes(1, h3Distance_kRing_assertions); iterateAllIndexesAtRes(2, h3Distance_kRing_assertions); // Don't iterate all of res 3, to save time iterateAllIndexesAtResPartial(3, h3Distance_kRing_assertions, 27); // Further resolutions aren't tested to save time. } TEST(h3DistanceBaseCells) { t_assert(H3_EXPORT(h3Distance)(bc1, pent1) == 1, "distance to neighbor is 1 (15, 4)"); t_assert(H3_EXPORT(h3Distance)(bc1, bc2) == 1, "distance to neighbor is 1 (15, 8)"); t_assert(H3_EXPORT(h3Distance)(bc1, bc3) == 1, "distance to neighbor is 1 (15, 31)"); t_assert(H3_EXPORT(h3Distance)(pent1, bc3) == -1, "distance to neighbor is invalid"); } TEST(ijkDistance) { CoordIJK z = {0, 0, 0}; CoordIJK i = {1, 0, 0}; CoordIJK ik = {1, 0, 1}; CoordIJK ij = {1, 1, 0}; CoordIJK j2 = {0, 2, 0}; t_assert(ijkDistance(&z, &z) == 0, "identity distance 0,0,0"); t_assert(ijkDistance(&i, &i) == 0, "identity distance 1,0,0"); t_assert(ijkDistance(&ik, &ik) == 0, "identity distance 1,0,1"); t_assert(ijkDistance(&ij, &ij) == 0, "identity distance 1,1,0"); t_assert(ijkDistance(&j2, &j2) == 0, "identity distance 0,2,0"); t_assert(ijkDistance(&z, &i) == 1, "0,0,0 to 1,0,0"); t_assert(ijkDistance(&z, &j2) == 2, "0,0,0 to 0,2,0"); t_assert(ijkDistance(&z, &ik) == 1, "0,0,0 to 1,0,1"); t_assert(ijkDistance(&i, &ik) == 1, "1,0,0 to 1,0,1"); t_assert(ijkDistance(&ik, &j2) == 3, "1,0,1 to 0,2,0"); t_assert(ijkDistance(&ij, &ik) == 2, "1,0,1 to 1,1,0"); } TEST(h3DistanceResolutionMismatch) { t_assert( H3_EXPORT(h3Distance)(0x832830fffffffffL, 0x822837fffffffffL) == -1, "cannot compare at different resolutions"); } TEST(h3DistanceEdge) { H3Index origin = 0x832830fffffffffL; H3Index dest = 0x832834fffffffffL; H3Index edge = H3_EXPORT(getH3UnidirectionalEdge(origin, dest)); t_assert(0 != edge, "test edge is valid"); t_assert(H3_EXPORT(h3Distance)(edge, origin) == 0, "edge has zero distance to origin"); t_assert(H3_EXPORT(h3Distance)(origin, edge) == 0, "origin has zero distance to edge"); t_assert(H3_EXPORT(h3Distance)(edge, dest) == 1, "edge has distance to destination"); t_assert(H3_EXPORT(h3Distance)(edge, dest) == 1, "destination has distance to edge"); } }