]> git.apps.os.sepia.ceph.com Git - ceph-ci.git/commitdiff
test: add tests for memstore clone_range
authorCasey Bodley <cbodley@redhat.com>
Mon, 19 Sep 2016 20:46:23 +0000 (16:46 -0400)
committerCasey Bodley <cbodley@redhat.com>
Tue, 20 Sep 2016 19:40:49 +0000 (15:40 -0400)
Signed-off-by: Casey Bodley <cbodley@redhat.com>
src/test/objectstore/CMakeLists.txt
src/test/objectstore/test_memstore_clone.cc [new file with mode: 0644]

index 198f2644f4b1e579dac90e5cb9ec5c6af2d18062..25633d2575bdf915c1f80c0343960570cf4279f8 100644 (file)
@@ -137,3 +137,7 @@ add_executable(unittest_transaction
 add_ceph_unittest(unittest_transaction ${CMAKE_RUNTIME_OUTPUT_DIRECTORY}/unittest_transaction)
 target_link_libraries(unittest_transaction os common)
 
+# unittest_memstore_clone
+add_executable(unittest_memstore_clone test_memstore_clone.cc)
+add_ceph_unittest(unittest_memstore_clone ${CMAKE_RUNTIME_OUTPUT_DIRECTORY}/unittest_memstore_clone)
+target_link_libraries(unittest_memstore_clone os global)
diff --git a/src/test/objectstore/test_memstore_clone.cc b/src/test/objectstore/test_memstore_clone.cc
new file mode 100644 (file)
index 0000000..bf5b63c
--- /dev/null
@@ -0,0 +1,224 @@
+// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
+// vim: ts=8 sw=2 smarttab
+/*
+ * Ceph - scalable distributed file system
+ *
+ * Copyright (C) 2015 Red Hat
+ *
+ * This is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License version 2.1, as published by the Free Software
+ * Foundation. See file COPYING.
+ *
+ */
+#include <boost/intrusive_ptr.hpp>
+#include "global/global_init.h"
+#include "common/ceph_argparse.h"
+#include "os/ObjectStore.h"
+#include <gtest/gtest.h>
+#include "include/assert.h"
+#include "common/errno.h"
+
+namespace {
+
+ObjectStore *g_store{nullptr};
+const coll_t cid;
+
+ghobject_t make_ghobject(const char *oid)
+{
+  return ghobject_t{hobject_t{oid, "", CEPH_NOSNAP, 0, 0, ""}};
+}
+
+} // anonymous namespace
+
+// src 11[11 11 11 11]11
+// dst 22 22 22 22 22 22
+// res 22 11 11 11 11 22
+TEST(MemStore, CloneRangeAllocated)
+{
+  ASSERT_TRUE(g_store);
+
+  const auto src = make_ghobject("src1");
+  const auto dst = make_ghobject("dst1");
+
+  bufferlist srcbl, dstbl, result, expected;
+  srcbl.append("111111111111");
+  dstbl.append("222222222222");
+  expected.append("221111111122");
+
+  ObjectStore::Transaction t;
+  t.write(cid, src, 0, 12, srcbl);
+  t.write(cid, dst, 0, 12, dstbl);
+  t.clone_range(cid, src, dst, 2, 8, 2);
+  ASSERT_EQ(0u, g_store->apply_transaction(nullptr, std::move(t)));
+  ASSERT_EQ(12, g_store->read(cid, dst, 0, 12, result));
+  ASSERT_EQ(expected, result);
+}
+
+// src __[__ __ __ __]__ 11 11
+// dst 22 22 22 22 22 22
+// res 22 00 00 00 00 22
+TEST(MemStore, CloneRangeHole)
+{
+  ASSERT_TRUE(g_store);
+
+  const auto src = make_ghobject("src2");
+  const auto dst = make_ghobject("dst2");
+
+  bufferlist srcbl, dstbl, result, expected;
+  srcbl.append("1111");
+  dstbl.append("222222222222");
+  expected.append("22\000\000\000\000\000\000\000\00022", 12);
+
+  ObjectStore::Transaction t;
+  t.write(cid, src, 12, 4, srcbl);
+  t.write(cid, dst, 0, 12, dstbl);
+  t.clone_range(cid, src, dst, 2, 8, 2);
+  ASSERT_EQ(0u, g_store->apply_transaction(nullptr, std::move(t)));
+  ASSERT_EQ(12, g_store->read(cid, dst, 0, 12, result));
+  ASSERT_EQ(expected, result);
+}
+
+// src __[__ __ __ 11]11
+// dst 22 22 22 22 22 22
+// res 22 00 00 00 11 22
+TEST(MemStore, CloneRangeHoleStart)
+{
+  ASSERT_TRUE(g_store);
+
+  const auto src = make_ghobject("src3");
+  const auto dst = make_ghobject("dst3");
+
+  bufferlist srcbl, dstbl, result, expected;
+  srcbl.append("1111");
+  dstbl.append("222222222222");
+  expected.append("22\000\000\000\000\000\0001122", 12);
+
+  ObjectStore::Transaction t;
+  t.write(cid, src, 8, 4, srcbl);
+  t.write(cid, dst, 0, 12, dstbl);
+  t.clone_range(cid, src, dst, 2, 8, 2);
+  ASSERT_EQ(0u, g_store->apply_transaction(nullptr, std::move(t)));
+  ASSERT_EQ(12, g_store->read(cid, dst, 0, 12, result));
+  ASSERT_EQ(expected, result);
+}
+
+// src 11[11 __ __ 11]11
+// dst 22 22 22 22 22 22
+// res 22 11 00 00 11 22
+TEST(MemStore, CloneRangeHoleMiddle)
+{
+  ASSERT_TRUE(g_store);
+
+  const auto src = make_ghobject("src4");
+  const auto dst = make_ghobject("dst4");
+
+  bufferlist srcbl, dstbl, result, expected;
+  srcbl.append("1111");
+  dstbl.append("222222222222");
+  expected.append("2211\000\000\000\0001122", 12);
+
+  ObjectStore::Transaction t;
+  t.write(cid, src, 0, 4, srcbl);
+  t.write(cid, src, 8, 4, srcbl);
+  t.write(cid, dst, 0, 12, dstbl);
+  t.clone_range(cid, src, dst, 2, 8, 2);
+  ASSERT_EQ(0u, g_store->apply_transaction(nullptr, std::move(t)));
+  ASSERT_EQ(12, g_store->read(cid, dst, 0, 12, result));
+  ASSERT_EQ(expected, result);
+}
+
+// src 11[11 __ __ __]__ 11 11
+// dst 22 22 22 22 22 22
+// res 22 11 00 00 00 22
+TEST(MemStore, CloneRangeHoleEnd)
+{
+  ASSERT_TRUE(g_store);
+
+  const auto src = make_ghobject("src5");
+  const auto dst = make_ghobject("dst5");
+
+  bufferlist srcbl, dstbl, result, expected;
+  srcbl.append("1111");
+  dstbl.append("222222222222");
+  expected.append("2211\000\000\000\000\000\00022", 12);
+
+  ObjectStore::Transaction t;
+  t.write(cid, src, 0, 4, srcbl);
+  t.write(cid, src, 12, 4, srcbl);
+  t.write(cid, dst, 0, 12, dstbl);
+  t.clone_range(cid, src, dst, 2, 8, 2);
+  ASSERT_EQ(0u, g_store->apply_transaction(nullptr, std::move(t)));
+  ASSERT_EQ(12, g_store->read(cid, dst, 0, 12, result));
+  ASSERT_EQ(expected, result);
+}
+
+// enable boost::intrusive_ptr<CephContext>
+void intrusive_ptr_add_ref(CephContext *cct) { cct->get(); }
+void intrusive_ptr_release(CephContext *cct) { cct->put(); }
+
+int main(int argc, char** argv)
+{
+  // default to memstore
+  vector<const char*> defaults{
+    "--osd_objectstore", "memstore",
+    "--osd_data", "memstore_clone_temp_dir",
+    "--memstore_page_size", "4",
+  };
+
+  vector<const char*> args;
+  argv_to_vec(argc, (const char **)argv, args);
+
+  global_init(&defaults, args, CEPH_ENTITY_TYPE_CLIENT,
+              CODE_ENVIRONMENT_UTILITY, 0);
+  common_init_finish(g_ceph_context);
+
+  // release g_ceph_context on exit
+  boost::intrusive_ptr<CephContext> cct{g_ceph_context, false};
+
+  // create and mount the objectstore
+  std::unique_ptr<ObjectStore> store{ObjectStore::create(
+      g_ceph_context,
+      g_conf->osd_objectstore,
+      g_conf->osd_data,
+      g_conf->osd_journal,
+      g_conf->osd_os_flags)};
+  if (!store) {
+    derr << "failed to create osd_objectstore=" << g_conf->osd_objectstore << dendl;
+    return EXIT_FAILURE;
+  }
+
+  int r = store->mkfs();
+  if (r < 0) {
+    derr << "failed to mkfs with " << cpp_strerror(r) << dendl;
+    return EXIT_FAILURE;
+  }
+
+  r = store->mount();
+  if (r < 0) {
+    derr << "failed to mount with " << cpp_strerror(r) << dendl;
+    return EXIT_FAILURE;
+  }
+  g_store = store.get();
+
+  ObjectStore::Transaction t;
+  t.create_collection(cid, 4);
+  r = store->apply_transaction(nullptr, std::move(t));
+  if (r < 0) {
+    derr << "failed to create collection with " << cpp_strerror(r) << dendl;
+    return EXIT_FAILURE;
+  }
+
+  // unmount the store on exit
+  auto umount = [] (ObjectStore *store) {
+    int r = store->umount();
+    if (r < 0) {
+      derr << "failed to unmount with " << cpp_strerror(r) << dendl;
+    }
+    g_store = nullptr;
+  };
+  std::unique_ptr<ObjectStore, decltype(umount)> umounter{store.get(), umount};
+
+  ::testing::InitGoogleTest(&argc, argv);
+  return RUN_ALL_TESTS();
+}