diff --git a/objectstore/Backend.hpp b/objectstore/Backend.hpp index a9d0b906ca4142b3c14924aba03a94b0d0223076..7b31b139786191c22de12c65153a26ce35154466 100644 --- a/objectstore/Backend.hpp +++ b/objectstore/Backend.hpp @@ -97,6 +97,12 @@ public: * @return pointer to the newly created representation. */ virtual Parameters * getParams() = 0; + + /** + * Return the name of the class. Mostly usefull in tests + * @return name of the class + */ + virtual std::string typeName() = 0; }; }} // end of cta::objectstore diff --git a/objectstore/BackendAbstractTests.cpp b/objectstore/BackendAbstractTests.cpp new file mode 100644 index 0000000000000000000000000000000000000000..015e5fb5c1df8649e7d75a5a0d8142fb92e9d6af --- /dev/null +++ b/objectstore/BackendAbstractTests.cpp @@ -0,0 +1,35 @@ +#include "BackendAbstractTests.hpp" +#include "BackendVFS.hpp" +#include "BackendRados.hpp" + +TEST_P(BackendAbstractTest, BasicReadWrite) { + std::cout << "Type=" << m_os->typeName() << std::endl; + const std::string testValue = "1234"; + const std::string testSecondValue = "1234"; + const std::string testObjectName = "testObject"; + // Check we can verify the absence of an object + ASSERT_EQ(false, m_os->exists(testObjectName)); + // Check that an update attempt fails on a non-existing object + ASSERT_THROW(m_os->atomicOverwrite(testObjectName, testSecondValue), cta::exception::Exception); + // Check the creation of the obecjt + m_os->create(testObjectName, testValue); + // Check that re-creating an existing object throws exception + ASSERT_THROW(m_os->create(testObjectName, testValue), cta::exception::Exception); + // Check we can validate the presence of the object + ASSERT_EQ(true, m_os->exists(testObjectName)); + // Check that we can read back after creation + ASSERT_EQ(testValue, m_os->read(testObjectName)); + m_os->atomicOverwrite(testObjectName, testSecondValue); + // Check that an update goes through + ASSERT_EQ(testSecondValue, m_os->read(testObjectName)); + // Check that we read back the value + ASSERT_EQ(testSecondValue, m_os->read(testObjectName)); + // Check we can delete the object + ASSERT_NO_THROW(m_os->remove(testObjectName)); + // Check that the object is actually gone + ASSERT_EQ(false, m_os->exists(testObjectName)); +} + +cta::objectstore::BackendVFS osVFS; +cta::objectstore::BackendRados osRados("tapetest", "tapetest"); +INSTANTIATE_TEST_CASE_P(BackendTest, BackendAbstractTest, ::testing::Values(&osVFS, &osRados)); \ No newline at end of file diff --git a/objectstore/BackendAbstractTests.hpp b/objectstore/BackendAbstractTests.hpp new file mode 100644 index 0000000000000000000000000000000000000000..94acd8eca9c1ef24d902fc8ab256531c7eee4776 --- /dev/null +++ b/objectstore/BackendAbstractTests.hpp @@ -0,0 +1,13 @@ +#pragma once + +#include <gtest/gtest.h> +#include "Backend.hpp" + +class BackendAbstractTest: public ::testing::TestWithParam<cta::objectstore::Backend *> { +protected: + BackendAbstractTest() {} + virtual void SetUp() { + m_os = GetParam(); + } + cta::objectstore::Backend * m_os; +}; diff --git a/objectstore/BackendRados.hpp b/objectstore/BackendRados.hpp index 1252e056209291310c3e4590b4e24d707317a172..92dfbf809891e416647300b27e005b6acad34f07 100644 --- a/objectstore/BackendRados.hpp +++ b/objectstore/BackendRados.hpp @@ -1,12 +1,24 @@ #pragma once #include "Backend.hpp" +#include "exception/Errnum.hpp" +#include <rados/librados.hpp> +#include <sys/syscall.h> +#include <errno.h> namespace cta { namespace objectstore { - -class ObjectStoreRados: public Backend { +/** + * An implementation of the object store primitives, using Rados. + */ +class BackendRados: public Backend { public: - ObjectStoreRados(std::string path, std::string userId, std::string pool): + /** + * The constructor, connecting to the storage pool 'pool' using the user id + * 'userId' + * @param userId + * @param pool + */ + BackendRados(std::string userId, std::string pool): m_user(userId), m_pool(pool), m_cluster(), m_radosCtx() { cta::exception::Errnum::throwOnNonZero(m_cluster.init(userId.c_str()), "In ObjectStoreRados::ObjectStoreRados, failed to m_cluster.init"); @@ -24,7 +36,7 @@ public: throw; } } - virtual ~ObjectStoreRados() { + virtual ~BackendRados() { m_radosCtx.close(); m_cluster.shutdown(); } @@ -37,14 +49,25 @@ public: virtual void create(std::string name, std::string content) { - atomicOverwrite(name, content); + librados::ObjectWriteOperation wop; + const bool createExclusive = true; + wop.create(createExclusive); + ceph::bufferlist bl; + bl.append(content.c_str(), content.size()); + wop.write_full(bl); + cta::exception::Errnum::throwOnNonZero(m_radosCtx.operate(name, &wop), + std::string("In ObjectStoreRados::create, failed to create exclusively or write ") + + name); } virtual void atomicOverwrite(std::string name, std::string content) { + librados::ObjectWriteOperation wop; + wop.assert_exists(); ceph::bufferlist bl; bl.append(content.c_str(), content.size()); - cta::exception::Errnum::throwOnNonZero(m_radosCtx.write_full(name, bl), - std::string("In ObjectStoreRados::atomicOverwrite, failed to write_full ") + wop.write_full(bl); + cta::exception::Errnum::throwOnNonZero(m_radosCtx.operate(name, &wop), + std::string("In ObjectStoreRados::atomicOverwrite, failed to assert existence or write ") + name); } @@ -67,7 +90,43 @@ public: cta::exception::Errnum::throwOnNegative(m_radosCtx.remove(name)); } - virtual void lockExclusive(std::string name, ContextHandle & context, std::string where) { + virtual bool exists(std::string name) { + uint64_t size; + time_t date; + if (m_radosCtx.stat(name, &size, &date)) { + return false; + } else { + return true; + } + } + + class ScopedLock: public Backend::ScopedLock { + friend class BackendRados; + public: + virtual void release() { + if (!m_lockSet) return; + cta::exception::Errnum::throwOnReturnedErrno( + -m_context.unlock(m_oid, "lock", m_clientId), + std::string("In cta::objectstore::ScopedLock::release, failed unlock ")+ + m_oid); + m_lockSet = false; + } + virtual ~ScopedLock() { release(); } + private: + ScopedLock(librados::IoCtx & ioCtx): m_lockSet(false), m_context(ioCtx) {} + void set(const std::string & oid, const std::string clientId) { + m_oid = oid; + m_clientId = clientId;\ + m_lockSet = true; + } + bool m_lockSet; + librados::IoCtx & m_context; + std::string m_clientId; + std::string m_oid; + }; + +private: + std::string createUniqueClientId() { // Build a unique client name: host:thread char buff[200]; cta::exception::Errnum::throwOnMinusOne(gethostname(buff, sizeof(buff)), @@ -75,60 +134,73 @@ public: pid_t tid = syscall(SYS_gettid); std::stringstream client; client << buff << ":" << tid; + return client.str(); + } + +public: + virtual ScopedLock * lockExclusive(std::string name) { + std::string client = createUniqueClientId(); struct timeval tv; tv.tv_usec = 0; tv.tv_sec = 10; int rc; + std::auto_ptr<ScopedLock> ret(new ScopedLock(m_radosCtx)); do { - rc=m_radosCtx.lock_exclusive(name, "lock", client.str(), "", &tv, 0); + rc=m_radosCtx.lock_exclusive(name, "lock", client, "", &tv, 0); } while (-EBUSY==rc); cta::exception::Errnum::throwOnReturnedErrno(-rc, std::string("In ObjectStoreRados::lockExclusive, failed to librados::IoCtx::lock_exclusive: ")+ - name + "/" + "lock" + "/" + client.str() + "//"); - context.set(); - //std::cout << "LockedExclusive: " << name << "/" << "lock" << "/" << client.str() << "//@" << where << std::endl; + name + "/" + "lock" + "/" + client + "//"); + ret->set(name, client); + return ret.release(); } - virtual void lockShared(std::string name, ContextHandle & context, std::string where) { - // Build a unique client name: host:thread - char buff[200]; - cta::exception::Errnum::throwOnMinusOne(gethostname(buff, sizeof(buff)), - "In ObjectStoreRados::lockExclusive: failed to gethostname"); - pid_t tid = syscall(SYS_gettid); - std::stringstream client; - client << buff << ":" << tid; + virtual ScopedLock * lockShared(std::string name) { + std::string client = createUniqueClientId(); struct timeval tv; tv.tv_usec = 0; tv.tv_sec = 10; int rc; + std::auto_ptr<ScopedLock> ret(new ScopedLock(m_radosCtx)); do { - rc=m_radosCtx.lock_shared(name, "lock", client.str(), "", "", &tv, 0); + rc=m_radosCtx.lock_shared(name, "lock", client, "", "", &tv, 0); } while (-EBUSY==rc); cta::exception::Errnum::throwOnReturnedErrno(-rc, std::string("In ObjectStoreRados::lockShared, failed to librados::IoCtx::lock_shared: ")+ - name + "/" + "lock" + "/" + client.str() + "//"); - context.set(); - //std::cout << "LockedShared: " << name << "/" << "lock" << "/" << client.str() << "//@" << where << std::endl; + name + "/" + "lock" + "/" + client + "//"); + ret->set(name, client); + return ret.release(); } - - virtual void unlock(std::string name, ContextHandle & context, std::string where) { - // Build a unique client name: host:thread - char buff[200]; - cta::exception::Errnum::throwOnMinusOne(gethostname(buff, sizeof(buff)), - "In ObjectStoreRados::lockExclusive: failed to gethostname"); - pid_t tid = syscall(SYS_gettid); - std::stringstream client; - client << buff << ":" << tid; - cta::exception::Errnum::throwOnReturnedErrno( - -m_radosCtx.unlock(name, "lock", client.str()), - std::string("In ObjectStoreRados::lockExclusive, failed to lock_exclusive ")+ - name); - context.reset(); - //std::cout << "Unlocked: " << name << "/" << "lock" << "/" << client.str() << "//@" << where << std::endl; + + class Parameters: public Backend::Parameters { + friend class BackendRados; + public: + /** + * The standard-issue params to string for logging + * @return a string representation of the parameters for logging + */ + virtual std::string toStr() { + std::stringstream ret; + ret << "userId=" << m_userId << " pool=" << m_pool; + return ret.str(); + } + private: + std::string m_userId; + std::string m_pool; + }; + + virtual Parameters * getParams() { + std::auto_ptr<Parameters> ret (new Parameters); + ret->m_pool = m_pool; + ret->m_userId = m_user; + return ret.release(); } - + virtual std::string typeName() { + return "cta::objectstore::BackendRados"; + } + private: std::string m_user; std::string m_pool; diff --git a/objectstore/BackendVFS.cpp b/objectstore/BackendVFS.cpp index b736271266ced8911f16c0adefd9945009578c1e..39de51d984e2a500be2cda8e5d35e137c9b22ea3 100644 --- a/objectstore/BackendVFS.cpp +++ b/objectstore/BackendVFS.cpp @@ -61,25 +61,29 @@ BackendVFS::~BackendVFS() { void BackendVFS::create(std::string name, std::string content) { std::string path = m_root + "/" + name; std::string lockPath = m_root + "/." + name + ".lock"; + bool fileCreated = false; + bool lockCreated = false; try { - int fd = ::creat(path.c_str(), S_IRWXU); + int fd = ::open(path.c_str(), O_WRONLY | O_CREAT | O_EXCL, S_IRWXU); // Create and fill up the path cta::exception::Errnum::throwOnMinusOne(fd, - "In ObjectStoreVFS::create, failed to creat the file"); + "In ObjectStoreVFS::create, failed to open the file"); + fileCreated = true; cta::exception::Errnum::throwOnMinusOne( ::write(fd, content.c_str(), content.size()), "In ObjectStoreVFS::create, failed to write to file"); cta::exception::Errnum::throwOnMinusOne(::close(fd), "In ObjectStoreVFS::create, failed to close the file"); // Create the lock file - int fdLock = ::creat(lockPath.c_str(), S_IRWXU); + int fdLock = ::open(lockPath.c_str(), O_WRONLY | O_CREAT | O_EXCL, S_IRWXU); + lockCreated = true; cta::exception::Errnum::throwOnMinusOne(fdLock, "In ObjectStoreVFS::create, failed to creat the lock file"); cta::exception::Errnum::throwOnMinusOne(::close(fdLock), "In ObjectStoreVFS::create, failed to close the lock file"); } catch (...) { - unlink(path.c_str()); - unlink(lockPath.c_str()); + if (fileCreated) unlink(path.c_str()); + if (lockCreated) unlink(lockPath.c_str()); throw; } } diff --git a/objectstore/BackendVFS.hpp b/objectstore/BackendVFS.hpp index 40893fb05b56e0f45516b8194bf212478fd0161d..b2dbd1cc3c3ffe6f0a2bcee48740724f51f27f2e 100644 --- a/objectstore/BackendVFS.hpp +++ b/objectstore/BackendVFS.hpp @@ -87,6 +87,12 @@ public: }; virtual Parameters * getParams(); + + + virtual std::string typeName() { + return "cta::objectstore::BackendVFS"; + } + private: std::string m_root; diff --git a/objectstore/CMakeLists.txt b/objectstore/CMakeLists.txt index c7fc916c336faec15e1886b6b7209a57b88cd350..8d295b6fe7b602da004789e7b729fa17b62e8667 100644 --- a/objectstore/CMakeLists.txt +++ b/objectstore/CMakeLists.txt @@ -49,6 +49,11 @@ add_library (CTAObjectStore # target_link_libraries(createEnvironment # protobuf rados CTAObjectStore) -add_executable(unitTests unitTests.cpp BackendVFSTest.cpp) +set(ObjectStoreUnitTests + BackendVFSTest.cpp + BackendAbstractTests.cpp) +# RootEntryTest.cpp) + +add_executable(unitTests unitTests.cpp ${ObjectStoreUnitTests}) target_link_libraries(unitTests protobuf rados CTAObjectStore gtest gmock) \ No newline at end of file diff --git a/objectstore/ObjectOps.hpp b/objectstore/ObjectOps.hpp index f2cfa959de1aa1ec13e02fcfd3ce8d325ca0e256..fe21f96069b243dc232a9f2f24afe4d15617a41a 100644 --- a/objectstore/ObjectOps.hpp +++ b/objectstore/ObjectOps.hpp @@ -1,6 +1,6 @@ #pragma once -#include "BackendStore.hpp.hpp" +#include "Backend.hpp" namespace cta { namespace objectstore { diff --git a/objectstore/ObjectStoreChoice.hpp b/objectstore/ObjectStoreChoice.hpp index c1c704ede306e52abcfebca9cae23cd4c83b190c..c87d6d3dc07a28b3831c8119b3fbe79e12b856ce 100644 --- a/objectstore/ObjectStoreChoice.hpp +++ b/objectstore/ObjectStoreChoice.hpp @@ -3,7 +3,7 @@ #include "ObjectOps.hpp" #define USE_RADOS 1 #if USE_RADOS -typedef cta::objectstore::ObjectStoreRados myOS; +typedef cta::objectstore::BackendRados myOS; #else typedef cta::objectstore::BackendVFS myOS; #endif \ No newline at end of file diff --git a/objectstore/RootEntry.hpp b/objectstore/RootEntry.hpp index c1e0f336a88306b3494709d9aaa2e8c1f5707858..a39e154fb31306df2ede0806811345fac9d2438d 100644 --- a/objectstore/RootEntry.hpp +++ b/objectstore/RootEntry.hpp @@ -2,7 +2,7 @@ #include "objectstore/cta.pb.h" -#include "BackendStore.hpp.hpp" +#include "Backend.hpp" #include "ObjectOps.hpp" #include "Agent.hpp" diff --git a/objectstore/RootEntryTest.cpp b/objectstore/RootEntryTest.cpp index f80310a0124c10d741db82e7b8a7b252e57b2f24..49c16246d69b5ea2d2c3557a832c428d12f284af 100644 --- a/objectstore/RootEntryTest.cpp +++ b/objectstore/RootEntryTest.cpp @@ -3,6 +3,6 @@ #include "exception/Exception.hpp" #include "RootEntry.hpp" -TEST(RootEntry.BasicAccess) { +TEST(RootEntry, BasicAccess) { -} \ No newline at end of file +}; \ No newline at end of file