2021-08-05 22:05:09 +00:00
|
|
|
#include "settings.hpp"
|
|
|
|
|
|
|
|
#include <components/detournavigator/asyncnavmeshupdater.hpp>
|
2022-09-22 18:26:05 +00:00
|
|
|
#include <components/detournavigator/dbrefgeometryobject.hpp>
|
2021-08-05 22:05:09 +00:00
|
|
|
#include <components/detournavigator/makenavmesh.hpp>
|
2022-09-22 18:26:05 +00:00
|
|
|
#include <components/detournavigator/navmeshdbutils.hpp>
|
2021-08-05 22:05:09 +00:00
|
|
|
#include <components/detournavigator/serialization.hpp>
|
|
|
|
#include <components/loadinglistener/loadinglistener.hpp>
|
|
|
|
|
2022-06-04 17:48:28 +00:00
|
|
|
#include <BulletCollision/CollisionShapes/btBoxShape.h>
|
|
|
|
|
2021-08-05 22:05:09 +00:00
|
|
|
#include <DetourNavMesh.h>
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
2022-03-10 17:34:35 +00:00
|
|
|
#include <limits>
|
2022-09-22 18:26:05 +00:00
|
|
|
#include <map>
|
2021-08-05 22:05:09 +00:00
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
using namespace testing;
|
|
|
|
using namespace DetourNavigator;
|
|
|
|
using namespace DetourNavigator::Tests;
|
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
void addHeightFieldPlane(
|
|
|
|
TileCachedRecastMeshManager& recastMeshManager, const osg::Vec2i cellPosition = osg::Vec2i(0, 0))
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
const int cellSize = 8192;
|
2022-09-22 18:26:05 +00:00
|
|
|
recastMeshManager.addHeightfield(cellPosition, cellSize, HeightfieldPlane{ 0 }, nullptr);
|
2021-08-05 22:05:09 +00:00
|
|
|
}
|
|
|
|
|
2021-12-16 21:40:02 +00:00
|
|
|
void addObject(const btBoxShape& shape, TileCachedRecastMeshManager& recastMeshManager)
|
|
|
|
{
|
|
|
|
const ObjectId id(&shape);
|
|
|
|
osg::ref_ptr<Resource::BulletShape> bulletShape(new Resource::BulletShape);
|
2024-09-23 22:14:18 +00:00
|
|
|
constexpr VFS::Path::NormalizedView test("test.nif");
|
|
|
|
bulletShape->mFileName = test;
|
2021-12-16 21:40:02 +00:00
|
|
|
bulletShape->mFileHash = "test_hash";
|
|
|
|
ObjectTransform objectTransform;
|
|
|
|
std::fill(std::begin(objectTransform.mPosition.pos), std::end(objectTransform.mPosition.pos), 0.1f);
|
|
|
|
std::fill(std::begin(objectTransform.mPosition.rot), std::end(objectTransform.mPosition.rot), 0.2f);
|
|
|
|
objectTransform.mScale = 3.14f;
|
|
|
|
const CollisionShape collisionShape(
|
2022-09-22 18:26:05 +00:00
|
|
|
osg::ref_ptr<Resource::BulletShapeInstance>(new Resource::BulletShapeInstance(bulletShape)), shape,
|
|
|
|
objectTransform);
|
2022-09-05 07:23:14 +00:00
|
|
|
recastMeshManager.addObject(id, collisionShape, btTransform::getIdentity(), AreaType_ground, nullptr);
|
2021-12-16 21:40:02 +00:00
|
|
|
}
|
|
|
|
|
2021-08-05 22:05:09 +00:00
|
|
|
struct DetourNavigatorAsyncNavMeshUpdaterTest : Test
|
|
|
|
{
|
|
|
|
Settings mSettings = makeSettings();
|
2022-09-22 18:26:05 +00:00
|
|
|
TileCachedRecastMeshManager mRecastMeshManager{ mSettings.mRecast };
|
|
|
|
OffMeshConnectionsManager mOffMeshConnectionsManager{ mSettings.mRecast };
|
|
|
|
const AgentBounds mAgentBounds{ CollisionShapeType::Aabb, { 29, 29, 66 } };
|
|
|
|
const TilePosition mPlayerTile{ 0, 0 };
|
2024-05-19 12:26:28 +00:00
|
|
|
const ESM::RefId mWorldspace = ESM::RefId::stringRefId("sys::default");
|
2022-09-22 18:26:05 +00:00
|
|
|
const btBoxShape mBox{ btVector3(100, 100, 20) };
|
2021-08-05 22:05:09 +00:00
|
|
|
Loading::Listener mListener;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, for_all_jobs_done_when_empty_wait_should_terminate)
|
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
AsyncNavMeshUpdater updater{ mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr };
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2021-08-05 22:05:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, for_required_tiles_present_when_empty_wait_should_terminate)
|
|
|
|
{
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::requiredTilesPresent, &mListener);
|
2021-08-05 22:05:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_should_generate_navmesh_tile)
|
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2021-08-05 22:05:09 +00:00
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-09-22 18:26:05 +00:00
|
|
|
const std::map<TilePosition, ChangeType> changedTiles{ { TilePosition{ 0, 0 }, ChangeType::add } };
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2022-07-01 14:43:58 +00:00
|
|
|
EXPECT_NE(navMeshCacheItem->lockConst()->getImpl().getTileRefAt(0, 0, 0), 0u);
|
2021-08-05 22:05:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, repeated_post_should_lead_to_cache_hit)
|
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2021-08-05 22:05:09 +00:00
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-09-22 18:26:05 +00:00
|
|
|
const std::map<TilePosition, ChangeType> changedTiles{ { TilePosition{ 0, 0 }, ChangeType::add } };
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
ASSERT_EQ(stats.mCache.mGetCount, 1);
|
|
|
|
ASSERT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
}
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mCache.mGetCount, 2);
|
|
|
|
EXPECT_EQ(stats.mCache.mHitCount, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_for_update_change_type_should_not_update_cache)
|
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2021-08-05 22:05:09 +00:00
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-09-22 18:26:05 +00:00
|
|
|
const std::map<TilePosition, ChangeType> changedTiles{ { TilePosition{ 0, 0 }, ChangeType::update } };
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
ASSERT_EQ(stats.mCache.mGetCount, 1);
|
|
|
|
ASSERT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
}
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mCache.mGetCount, 2);
|
|
|
|
EXPECT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_should_write_generated_tile_to_db)
|
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2021-08-05 22:05:09 +00:00
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
2021-12-16 21:40:02 +00:00
|
|
|
addObject(mBox, mRecastMeshManager);
|
2022-03-10 17:34:35 +00:00
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max());
|
2021-08-05 22:05:09 +00:00
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-09-22 18:26:05 +00:00
|
|
|
const TilePosition tilePosition{ 0, 0 };
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles{ { tilePosition, ChangeType::add } };
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2022-03-10 17:34:35 +00:00
|
|
|
updater.stop();
|
2021-08-05 22:05:09 +00:00
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
2022-09-22 18:26:05 +00:00
|
|
|
ShapeId nextShapeId{ 1 };
|
2021-08-05 22:05:09 +00:00
|
|
|
const std::vector<DbRefGeometryObject> objects = makeDbRefGeometryObjects(recastMesh->getMeshSources(),
|
2022-09-22 18:26:05 +00:00
|
|
|
[&](const MeshSource& v) { return resolveMeshSource(*dbPtr, v, nextShapeId); });
|
|
|
|
const auto tile = dbPtr->findTile(
|
|
|
|
mWorldspace, tilePosition, serialize(mSettings.mRecast, mAgentBounds, *recastMesh, objects));
|
2021-08-05 22:05:09 +00:00
|
|
|
ASSERT_TRUE(tile.has_value());
|
|
|
|
EXPECT_EQ(tile->mTileId, 1);
|
2022-07-01 12:25:23 +00:00
|
|
|
EXPECT_EQ(tile->mVersion, navMeshFormatVersion);
|
2021-08-05 22:05:09 +00:00
|
|
|
}
|
|
|
|
|
2021-12-16 21:40:02 +00:00
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_when_writing_to_db_disabled_should_not_write_tiles)
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2021-08-05 22:05:09 +00:00
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
2021-12-16 21:40:02 +00:00
|
|
|
addObject(mBox, mRecastMeshManager);
|
2022-03-10 17:34:35 +00:00
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max());
|
2021-08-05 22:05:09 +00:00
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
mSettings.mWriteToNavMeshDb = false;
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-09-22 18:26:05 +00:00
|
|
|
const TilePosition tilePosition{ 0, 0 };
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles{ { tilePosition, ChangeType::add } };
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2022-03-10 17:34:35 +00:00
|
|
|
updater.stop();
|
2021-08-05 22:05:09 +00:00
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
2022-09-22 18:26:05 +00:00
|
|
|
ShapeId nextShapeId{ 1 };
|
2021-08-05 22:05:09 +00:00
|
|
|
const std::vector<DbRefGeometryObject> objects = makeDbRefGeometryObjects(recastMesh->getMeshSources(),
|
2022-09-22 18:26:05 +00:00
|
|
|
[&](const MeshSource& v) { return resolveMeshSource(*dbPtr, v, nextShapeId); });
|
|
|
|
const auto tile = dbPtr->findTile(
|
|
|
|
mWorldspace, tilePosition, serialize(mSettings.mRecast, mAgentBounds, *recastMesh, objects));
|
2021-08-05 22:05:09 +00:00
|
|
|
ASSERT_FALSE(tile.has_value());
|
|
|
|
}
|
|
|
|
|
2021-12-16 21:40:02 +00:00
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_when_writing_to_db_disabled_should_not_write_shapes)
|
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2021-12-16 21:40:02 +00:00
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
addObject(mBox, mRecastMeshManager);
|
2022-03-10 17:34:35 +00:00
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max());
|
2021-12-16 21:40:02 +00:00
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
mSettings.mWriteToNavMeshDb = false;
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-09-22 18:26:05 +00:00
|
|
|
const TilePosition tilePosition{ 0, 0 };
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles{ { tilePosition, ChangeType::add } };
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2022-03-10 17:34:35 +00:00
|
|
|
updater.stop();
|
2021-12-16 21:40:02 +00:00
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
2022-09-22 18:26:05 +00:00
|
|
|
const auto objects = makeDbRefGeometryObjects(
|
|
|
|
recastMesh->getMeshSources(), [&](const MeshSource& v) { return resolveMeshSource(*dbPtr, v); });
|
2023-05-13 12:55:20 +00:00
|
|
|
EXPECT_TRUE(std::holds_alternative<MeshSource>(objects));
|
2021-12-16 21:40:02 +00:00
|
|
|
}
|
|
|
|
|
2021-08-05 22:05:09 +00:00
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_should_read_from_db_on_cache_miss)
|
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2021-08-05 22:05:09 +00:00
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
mSettings.mMaxNavMeshTilesCacheSize = 0;
|
2022-03-10 17:34:35 +00:00
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager,
|
2022-09-22 18:26:05 +00:00
|
|
|
std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max()));
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-09-22 18:26:05 +00:00
|
|
|
const std::map<TilePosition, ChangeType> changedTiles{ { TilePosition{ 0, 0 }, ChangeType::add } };
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
ASSERT_EQ(stats.mCache.mGetCount, 1);
|
|
|
|
ASSERT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
ASSERT_TRUE(stats.mDb.has_value());
|
|
|
|
ASSERT_EQ(stats.mDb->mGetTileCount, 1);
|
|
|
|
ASSERT_EQ(stats.mDbGetTileHits, 0);
|
|
|
|
}
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mCache.mGetCount, 2);
|
|
|
|
EXPECT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
ASSERT_TRUE(stats.mDb.has_value());
|
|
|
|
EXPECT_EQ(stats.mDb->mGetTileCount, 2);
|
|
|
|
EXPECT_EQ(stats.mDbGetTileHits, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, on_changing_player_tile_post_should_remove_tiles_out_of_range)
|
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2021-08-05 22:05:09 +00:00
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-09-22 18:26:05 +00:00
|
|
|
const std::map<TilePosition, ChangeType> changedTilesAdd{ { TilePosition{ 0, 0 }, ChangeType::add } };
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTilesAdd);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2022-07-01 14:43:58 +00:00
|
|
|
ASSERT_NE(navMeshCacheItem->lockConst()->getImpl().getTileRefAt(0, 0, 0), 0u);
|
2022-09-22 18:26:05 +00:00
|
|
|
const std::map<TilePosition, ChangeType> changedTilesRemove{ { TilePosition{ 0, 0 }, ChangeType::remove } };
|
2021-08-05 22:05:09 +00:00
|
|
|
const TilePosition playerTile(100, 100);
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, playerTile, mWorldspace, changedTilesRemove);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2022-07-01 14:43:58 +00:00
|
|
|
EXPECT_EQ(navMeshCacheItem->lockConst()->getImpl().getTileRefAt(0, 0, 0), 0u);
|
2021-08-05 22:05:09 +00:00
|
|
|
}
|
2022-03-10 17:34:35 +00:00
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, should_stop_writing_to_db_when_size_limit_is_reached)
|
|
|
|
{
|
2022-09-05 07:23:14 +00:00
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
2022-03-10 17:34:35 +00:00
|
|
|
for (int x = -1; x <= 1; ++x)
|
|
|
|
for (int y = -1; y <= 1; ++y)
|
|
|
|
addHeightFieldPlane(mRecastMeshManager, osg::Vec2i(x, y));
|
|
|
|
addObject(mBox, mRecastMeshManager);
|
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", 4097);
|
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
2023-02-17 13:55:05 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
2022-03-10 17:34:35 +00:00
|
|
|
std::map<TilePosition, ChangeType> changedTiles;
|
|
|
|
for (int x = -5; x <= 5; ++x)
|
|
|
|
for (int y = -5; y <= 5; ++y)
|
2022-09-22 18:26:05 +00:00
|
|
|
changedTiles.emplace(TilePosition{ x, y }, ChangeType::add);
|
2022-06-16 22:28:44 +00:00
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
2022-09-05 07:23:14 +00:00
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
2022-03-10 17:34:35 +00:00
|
|
|
updater.stop();
|
2024-08-06 22:01:47 +00:00
|
|
|
|
|
|
|
std::size_t present = 0;
|
|
|
|
|
2022-03-10 17:34:35 +00:00
|
|
|
for (int x = -5; x <= 5; ++x)
|
2024-08-06 22:01:47 +00:00
|
|
|
{
|
2022-03-10 17:34:35 +00:00
|
|
|
for (int y = -5; y <= 5; ++y)
|
|
|
|
{
|
|
|
|
const TilePosition tilePosition(x, y);
|
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
2023-05-13 12:55:20 +00:00
|
|
|
const auto objects = makeDbRefGeometryObjects(
|
2022-09-22 18:26:05 +00:00
|
|
|
recastMesh->getMeshSources(), [&](const MeshSource& v) { return resolveMeshSource(*dbPtr, v); });
|
2023-05-13 12:55:20 +00:00
|
|
|
if (std::holds_alternative<MeshSource>(objects))
|
2022-03-10 17:34:35 +00:00
|
|
|
continue;
|
2024-08-06 22:01:47 +00:00
|
|
|
present += dbPtr
|
|
|
|
->findTile(mWorldspace, tilePosition,
|
|
|
|
serialize(mSettings.mRecast, mAgentBounds, *recastMesh,
|
|
|
|
std::get<std::vector<DbRefGeometryObject>>(objects)))
|
|
|
|
.has_value();
|
2022-03-10 17:34:35 +00:00
|
|
|
}
|
2024-08-06 22:01:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
EXPECT_EQ(present, 11);
|
2022-03-10 17:34:35 +00:00
|
|
|
}
|
2024-02-07 10:14:31 +00:00
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, next_tile_id_should_be_updated_on_duplicate)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
addObject(mBox, mRecastMeshManager);
|
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max());
|
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
|
|
|
|
|
|
|
const TileId nextTileId(dbPtr->getMaxTileId() + 1);
|
2024-05-19 12:26:28 +00:00
|
|
|
ASSERT_EQ(dbPtr->insertTile(nextTileId, mWorldspace, TilePosition{}, TileVersion{ 1 }, {}, {}), 1);
|
2024-02-07 10:14:31 +00:00
|
|
|
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
|
|
|
const TilePosition tilePosition{ 0, 0 };
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles{ { tilePosition, ChangeType::add } };
|
|
|
|
|
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
|
|
|
|
|
|
|
const AgentBounds agentBounds{ CollisionShapeType::Cylinder, { 29, 29, 66 } };
|
|
|
|
updater.post(agentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
|
|
|
|
|
|
|
updater.stop();
|
|
|
|
|
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
|
|
|
ShapeId nextShapeId{ 1 };
|
|
|
|
const std::vector<DbRefGeometryObject> objects = makeDbRefGeometryObjects(recastMesh->getMeshSources(),
|
|
|
|
[&](const MeshSource& v) { return resolveMeshSource(*dbPtr, v, nextShapeId); });
|
|
|
|
const auto tile = dbPtr->findTile(
|
|
|
|
mWorldspace, tilePosition, serialize(mSettings.mRecast, agentBounds, *recastMesh, objects));
|
|
|
|
ASSERT_TRUE(tile.has_value());
|
|
|
|
EXPECT_EQ(tile->mTileId, 2);
|
|
|
|
EXPECT_EQ(tile->mVersion, navMeshFormatVersion);
|
|
|
|
}
|
2024-04-05 23:10:48 +00:00
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, repeated_tile_updates_should_be_delayed)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace, nullptr);
|
|
|
|
|
|
|
|
mSettings.mMaxTilesNumber = 9;
|
|
|
|
mSettings.mMinUpdateInterval = std::chrono::milliseconds(250);
|
|
|
|
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(1, mSettings);
|
|
|
|
|
|
|
|
std::map<TilePosition, ChangeType> changedTiles;
|
|
|
|
|
|
|
|
for (int x = -3; x <= 3; ++x)
|
|
|
|
for (int y = -3; y <= 3; ++y)
|
|
|
|
changedTiles.emplace(TilePosition{ x, y }, ChangeType::update);
|
|
|
|
|
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
|
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
|
|
|
|
|
|
|
{
|
|
|
|
const AsyncNavMeshUpdaterStats stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mJobs, 0);
|
|
|
|
EXPECT_EQ(stats.mWaiting.mDelayed, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
updater.post(mAgentBounds, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
|
|
|
|
{
|
|
|
|
const AsyncNavMeshUpdaterStats stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mJobs, 49);
|
|
|
|
EXPECT_EQ(stats.mWaiting.mDelayed, 49);
|
|
|
|
}
|
|
|
|
|
|
|
|
updater.wait(WaitConditionType::allJobsDone, &mListener);
|
|
|
|
|
|
|
|
{
|
|
|
|
const AsyncNavMeshUpdaterStats stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mJobs, 0);
|
|
|
|
EXPECT_EQ(stats.mWaiting.mDelayed, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct DetourNavigatorSpatialJobQueueTest : Test
|
|
|
|
{
|
|
|
|
const AgentBounds mAgentBounds{ CollisionShapeType::Aabb, osg::Vec3f(1, 1, 1) };
|
|
|
|
const std::shared_ptr<GuardedNavMeshCacheItem> mNavMeshCacheItemPtr;
|
|
|
|
const std::weak_ptr<GuardedNavMeshCacheItem> mNavMeshCacheItem = mNavMeshCacheItemPtr;
|
2024-05-19 12:26:28 +00:00
|
|
|
const ESM::RefId mWorldspace = ESM::RefId::stringRefId("worldspace");
|
2024-04-05 23:10:48 +00:00
|
|
|
const TilePosition mChangedTile{ 0, 0 };
|
|
|
|
const std::chrono::steady_clock::time_point mProcessTime{};
|
|
|
|
const TilePosition mPlayerTile{ 0, 0 };
|
|
|
|
const int mMaxTiles = 9;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorSpatialJobQueueTest, should_store_multiple_jobs_per_tile)
|
|
|
|
{
|
|
|
|
std::list<Job> jobs;
|
|
|
|
SpatialJobQueue queue;
|
|
|
|
|
2024-05-19 12:26:28 +00:00
|
|
|
const ESM::RefId worldspace1 = ESM::RefId::stringRefId("worldspace1");
|
|
|
|
const ESM::RefId worldspace2 = ESM::RefId::stringRefId("worldspace2");
|
|
|
|
|
|
|
|
queue.push(jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, worldspace1, mChangedTile, ChangeType::remove, mProcessTime));
|
|
|
|
queue.push(jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, worldspace2, mChangedTile, ChangeType::update, mProcessTime));
|
2024-04-05 23:10:48 +00:00
|
|
|
|
|
|
|
ASSERT_EQ(queue.size(), 2);
|
|
|
|
|
|
|
|
const auto job1 = queue.pop(mChangedTile);
|
|
|
|
ASSERT_TRUE(job1.has_value());
|
2024-05-19 12:26:28 +00:00
|
|
|
EXPECT_EQ((*job1)->mWorldspace, worldspace1);
|
2024-04-05 23:10:48 +00:00
|
|
|
|
|
|
|
const auto job2 = queue.pop(mChangedTile);
|
|
|
|
ASSERT_TRUE(job2.has_value());
|
2024-05-19 12:26:28 +00:00
|
|
|
EXPECT_EQ((*job2)->mWorldspace, worldspace2);
|
2024-04-05 23:10:48 +00:00
|
|
|
|
|
|
|
EXPECT_EQ(queue.size(), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct DetourNavigatorJobQueueTest : DetourNavigatorSpatialJobQueueTest
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, pop_should_return_nullptr_from_empty)
|
|
|
|
{
|
|
|
|
JobQueue queue;
|
|
|
|
ASSERT_FALSE(queue.hasJob());
|
|
|
|
ASSERT_FALSE(queue.pop(mPlayerTile).has_value());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, push_on_change_type_remove_should_add_to_removing)
|
|
|
|
{
|
|
|
|
const std::chrono::steady_clock::time_point processTime{};
|
|
|
|
|
|
|
|
std::list<Job> jobs;
|
|
|
|
const JobIt job = jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, mChangedTile, ChangeType::remove, processTime);
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(job);
|
|
|
|
|
|
|
|
EXPECT_EQ(queue.getStats().mRemoving, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, pop_should_return_last_removing)
|
|
|
|
{
|
|
|
|
std::list<Job> jobs;
|
|
|
|
JobQueue queue;
|
|
|
|
|
|
|
|
queue.push(jobs.emplace(jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, TilePosition(0, 0),
|
|
|
|
ChangeType::remove, mProcessTime));
|
|
|
|
queue.push(jobs.emplace(jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, TilePosition(1, 0),
|
|
|
|
ChangeType::remove, mProcessTime));
|
|
|
|
|
|
|
|
ASSERT_TRUE(queue.hasJob());
|
|
|
|
const auto job = queue.pop(mPlayerTile);
|
|
|
|
ASSERT_TRUE(job.has_value());
|
|
|
|
EXPECT_EQ((*job)->mChangedTile, TilePosition(1, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, push_on_change_type_not_remove_should_add_to_updating)
|
|
|
|
{
|
|
|
|
std::list<Job> jobs;
|
|
|
|
const JobIt job = jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, mChangedTile, ChangeType::update, mProcessTime);
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(job);
|
|
|
|
|
|
|
|
EXPECT_EQ(queue.getStats().mUpdating, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, pop_should_return_nearest_to_player_tile)
|
|
|
|
{
|
|
|
|
std::list<Job> jobs;
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(jobs.emplace(jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, TilePosition(0, 0),
|
|
|
|
ChangeType::update, mProcessTime));
|
|
|
|
queue.push(jobs.emplace(jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, TilePosition(1, 0),
|
|
|
|
ChangeType::update, mProcessTime));
|
|
|
|
|
|
|
|
ASSERT_TRUE(queue.hasJob());
|
|
|
|
const auto job = queue.pop(TilePosition(1, 0));
|
|
|
|
ASSERT_TRUE(job.has_value());
|
|
|
|
EXPECT_EQ((*job)->mChangedTile, TilePosition(1, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, push_on_processing_time_more_than_now_should_add_to_delayed)
|
|
|
|
{
|
|
|
|
const std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now();
|
|
|
|
const std::chrono::steady_clock::time_point processTime = now + std::chrono::seconds(1);
|
|
|
|
|
|
|
|
std::list<Job> jobs;
|
|
|
|
const JobIt job = jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, mChangedTile, ChangeType::update, processTime);
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(job, now);
|
|
|
|
|
|
|
|
EXPECT_EQ(queue.getStats().mDelayed, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, pop_should_return_when_delayed_job_is_ready)
|
|
|
|
{
|
|
|
|
const std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now();
|
|
|
|
const std::chrono::steady_clock::time_point processTime = now + std::chrono::seconds(1);
|
|
|
|
|
|
|
|
std::list<Job> jobs;
|
|
|
|
const JobIt job = jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, mChangedTile, ChangeType::update, processTime);
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(job, now);
|
|
|
|
|
|
|
|
ASSERT_FALSE(queue.hasJob(now));
|
|
|
|
ASSERT_FALSE(queue.pop(mPlayerTile, now).has_value());
|
|
|
|
|
|
|
|
ASSERT_TRUE(queue.hasJob(processTime));
|
|
|
|
EXPECT_TRUE(queue.pop(mPlayerTile, processTime).has_value());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, update_should_move_ready_delayed_to_updating)
|
|
|
|
{
|
|
|
|
const std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now();
|
|
|
|
const std::chrono::steady_clock::time_point processTime = now + std::chrono::seconds(1);
|
|
|
|
|
|
|
|
std::list<Job> jobs;
|
|
|
|
const JobIt job = jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, mChangedTile, ChangeType::update, processTime);
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(job, now);
|
|
|
|
|
|
|
|
ASSERT_EQ(queue.getStats().mDelayed, 1);
|
|
|
|
|
|
|
|
queue.update(mPlayerTile, mMaxTiles, processTime);
|
|
|
|
|
|
|
|
EXPECT_EQ(queue.getStats().mDelayed, 0);
|
|
|
|
EXPECT_EQ(queue.getStats().mUpdating, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, update_should_move_ready_delayed_to_removing_when_out_of_range)
|
|
|
|
{
|
|
|
|
const std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now();
|
|
|
|
const std::chrono::steady_clock::time_point processTime = now + std::chrono::seconds(1);
|
|
|
|
|
|
|
|
std::list<Job> jobs;
|
|
|
|
const JobIt job = jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, mChangedTile, ChangeType::update, processTime);
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(job, now);
|
|
|
|
|
|
|
|
ASSERT_EQ(queue.getStats().mDelayed, 1);
|
|
|
|
|
|
|
|
queue.update(TilePosition(10, 10), mMaxTiles, processTime);
|
|
|
|
|
|
|
|
EXPECT_EQ(queue.getStats().mDelayed, 0);
|
|
|
|
EXPECT_EQ(queue.getStats().mRemoving, 1);
|
|
|
|
EXPECT_EQ(job->mChangeType, ChangeType::remove);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, update_should_move_updating_to_removing_when_out_of_range)
|
|
|
|
{
|
|
|
|
std::list<Job> jobs;
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(jobs.emplace(
|
|
|
|
jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, mChangedTile, ChangeType::update, mProcessTime));
|
|
|
|
queue.push(jobs.emplace(jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, TilePosition(10, 10),
|
|
|
|
ChangeType::update, mProcessTime));
|
|
|
|
|
|
|
|
ASSERT_EQ(queue.getStats().mUpdating, 2);
|
|
|
|
|
|
|
|
queue.update(TilePosition(10, 10), mMaxTiles);
|
|
|
|
|
|
|
|
EXPECT_EQ(queue.getStats().mUpdating, 1);
|
|
|
|
EXPECT_EQ(queue.getStats().mRemoving, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorJobQueueTest, clear_should_remove_all)
|
|
|
|
{
|
|
|
|
const std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now();
|
|
|
|
const std::chrono::steady_clock::time_point processTime = now + std::chrono::seconds(1);
|
|
|
|
|
|
|
|
std::list<Job> jobs;
|
|
|
|
const JobIt removing = jobs.emplace(jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace,
|
|
|
|
TilePosition(0, 0), ChangeType::remove, mProcessTime);
|
|
|
|
const JobIt updating = jobs.emplace(jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace,
|
|
|
|
TilePosition(1, 0), ChangeType::update, mProcessTime);
|
|
|
|
const JobIt delayed = jobs.emplace(jobs.end(), mAgentBounds, mNavMeshCacheItem, mWorldspace, TilePosition(2, 0),
|
|
|
|
ChangeType::update, processTime);
|
|
|
|
|
|
|
|
JobQueue queue;
|
|
|
|
queue.push(removing);
|
|
|
|
queue.push(updating);
|
|
|
|
queue.push(delayed, now);
|
|
|
|
|
|
|
|
ASSERT_EQ(queue.getStats().mRemoving, 1);
|
|
|
|
ASSERT_EQ(queue.getStats().mUpdating, 1);
|
|
|
|
ASSERT_EQ(queue.getStats().mDelayed, 1);
|
|
|
|
|
|
|
|
queue.clear();
|
|
|
|
|
|
|
|
EXPECT_EQ(queue.getStats().mRemoving, 0);
|
|
|
|
EXPECT_EQ(queue.getStats().mUpdating, 0);
|
|
|
|
EXPECT_EQ(queue.getStats().mDelayed, 0);
|
|
|
|
}
|
2021-08-05 22:05:09 +00:00
|
|
|
}
|