From 56828d241a618c94f0526b5fd498b7f3ce540230 Mon Sep 17 00:00:00 2001 From: Paul Cristian Sarbu Date: Mon, 22 Aug 2022 17:54:16 +0200 Subject: Test: Add tests for file locking utility class --- test/utils/cpp/file_locking.test.cpp | 70 ++++++++++++++++++++++++++++++++++++ 1 file changed, 70 insertions(+) create mode 100644 test/utils/cpp/file_locking.test.cpp (limited to 'test/utils/cpp/file_locking.test.cpp') diff --git a/test/utils/cpp/file_locking.test.cpp b/test/utils/cpp/file_locking.test.cpp new file mode 100644 index 00000000..9e3924e8 --- /dev/null +++ b/test/utils/cpp/file_locking.test.cpp @@ -0,0 +1,70 @@ +// Copyright 2022 Huawei Cloud Computing Technology Co., Ltd. +// +// 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. + +#include + +#include "catch2/catch.hpp" +#include "src/buildtool/file_system/file_system_manager.hpp" +#include "src/utils/cpp/atomic.hpp" +#include "src/utils/cpp/file_locking.hpp" + +namespace { +[[nodiscard]] auto GetTestDir() noexcept -> std::filesystem::path { + auto* tmp_dir = std::getenv("TEST_TMPDIR"); + if (tmp_dir != nullptr) { + return tmp_dir; + } + return FileSystemManager::GetCurrentDirectory() / "test/other_tools"; +} + +[[nodiscard]] auto GetLockDirPath(int id) noexcept -> std::filesystem::path { + auto lock_file = std::to_string(id) + std::string{".lock"}; + return GetTestDir() / lock_file; +} +} // namespace + +TEST_CASE("Multi-file locking", "[file_locking]") { + // Test locking and unlocking. Each thread will have one lock. + // setup threading + constexpr auto kNumThreads = 50; // increasing it too much will fail + constexpr auto kNumLocks = 5; + + atomic starting_signal{false}; + std::vector threads{}; + threads.reserve(kNumThreads); + + for (int id{}; id < kNumThreads; ++id) { + threads.emplace_back( + [&starting_signal](int tid) { + starting_signal.wait(false); + // cases based on id + auto flockpath = GetLockDirPath(tid % kNumLocks); + // Get lock + auto lock = LockFile::Acquire(flockpath, /*is_shared=*/false); + REQUIRE(lock); + // Do some "work" + std::this_thread::sleep_for(std::chrono::milliseconds(1)); + // lock released automatically when out of scope + }, + id); + } + + starting_signal = true; + starting_signal.notify_all(); + + // wait for threads to finish + for (auto& thread : threads) { + thread.join(); + } +} -- cgit v1.2.3