]> git.proxmox.com Git - ceph.git/blame - ceph/src/test/librgw_file_marker.cc
bump version to 12.2.4-pve1
[ceph.git] / ceph / src / test / librgw_file_marker.cc
CommitLineData
3efd9988
FG
1// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
2// vim: ts=8 sw=2 smarttab
3/*
4 * Ceph - scalable distributed file system
5 *
6 * Copyright (C) 2015 Red Hat, Inc.
7 *
8 * This is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License version 2.1, as published by the Free Software
11 * Foundation. See file COPYING.
12 *
13 */
14
15#include <stdint.h>
16#include <tuple>
17#include <iostream>
18#include <fstream>
19#include <stack>
20
21#include "include/rados/librgw.h"
22#include "include/rados/rgw_file.h"
23#include "rgw/rgw_file.h"
24#include "rgw/rgw_lib_frontend.h" // direct requests
25
26#include "gtest/gtest.h"
27#include "common/backport14.h"
28#include "common/ceph_argparse.h"
29#include "common/debug.h"
30#include "global/global_init.h"
31#include "include/assert.h"
32
33#define dout_subsys ceph_subsys_rgw
34
35namespace {
36
37 using namespace rgw;
38 using std::get;
39 using std::string;
40
41 librgw_t rgw_h = nullptr;
42 string userid("testuser");
43 string access_key("");
44 string secret_key("");
45 struct rgw_fs *fs = nullptr;
46 CephContext* cct = nullptr;
47
48 uint32_t owner_uid = 867;
49 uint32_t owner_gid = 5309;
50
51 uint32_t create_mask = RGW_SETATTR_UID | RGW_SETATTR_GID | RGW_SETATTR_MODE;
52
53 string bucket_name("nfsroot");
54
55 class obj_rec
56 {
57 public:
58 string name;
59 struct rgw_file_handle* fh;
60 struct rgw_file_handle* parent_fh;
61 RGWFileHandle* rgw_fh; // alias into fh
62
63 struct state {
64 bool readdir;
65 state() : readdir(false) {}
66 } state;
67
68 obj_rec(string _name, struct rgw_file_handle* _fh,
69 struct rgw_file_handle* _parent_fh, RGWFileHandle* _rgw_fh)
70 : name(std::move(_name)), fh(_fh), parent_fh(_parent_fh),
71 rgw_fh(_rgw_fh) {}
72
73 void clear() {
74 fh = nullptr;
75 rgw_fh = nullptr;
76 }
77
78 void sync() {
79 if (fh)
80 rgw_fh = get_rgwfh(fh);
81 }
82
83 friend ostream& operator<<(ostream& os, const obj_rec& rec);
84 };
85
86 ostream& operator<<(ostream& os, const obj_rec& rec)
87 {
88 RGWFileHandle* rgw_fh = rec.rgw_fh;
89 if (rgw_fh) {
90 const char* type = rgw_fh->is_dir() ? "DIR " : "FILE ";
91 os << rec.rgw_fh->full_object_name()
92 << " (" << rec.rgw_fh->object_name() << "): "
93 << type;
94 }
95 return os;
96 }
97
98 std::stack<obj_rec> obj_stack;
99 std::deque<obj_rec> cleanup_queue;
100
101 typedef std::vector<obj_rec> obj_vec;
102 typedef std::tuple<obj_rec, obj_vec> dirs1_rec;
103 typedef std::vector<dirs1_rec> dirs1_vec;
104
105 dirs1_vec dirs_vec;
106
107 struct obj_rec_st
108 {
109 const obj_rec& obj;
110 const struct stat& st;
111
112 obj_rec_st(const obj_rec& _obj, const struct stat& _st)
113 : obj(_obj), st(_st) {}
114 };
115
116 ostream& operator<<(ostream& os, const obj_rec_st& rec)
117 {
118 RGWFileHandle* rgw_fh = rec.obj.rgw_fh;
119 if (rgw_fh) {
120 const char* type = rgw_fh->is_dir() ? "DIR " : "FILE ";
121 os << rgw_fh->full_object_name()
122 << " (" << rgw_fh->object_name() << "): "
123 << type;
124 const struct stat& st = rec.st;
125 switch(uint8_t(rgw_fh->is_dir())) {
126 case 1:
127 os << " mode: " << st.st_mode;
128 os << " nlinks: " << st.st_nlink;
129 break;
130 case 0:
131 default:
132 os << " mode: " << st.st_mode;
133 os << " size: " << st.st_size;
134 // xxx
135 break;
136 }
137 }
138 return os;
139 }
140
141 bool do_marker1 = false;
142 bool do_marker2 = true;
143 bool do_create = false;
144 bool do_delete = false;
145 bool verbose = false;
146
147 string marker_dir("nfs_marker");
148 struct rgw_file_handle *bucket_fh = nullptr;
149 struct rgw_file_handle *marker_fh;
150 static constexpr int marker_nobjs = 2*1024;
151 std::deque<obj_rec> marker_objs;
152
153 using dirent_t = std::tuple<std::string, uint64_t>;
154 struct dirent_vec
155 {
156 std::vector<dirent_t> obj_names;
157 uint32_t count;
158 dirent_vec() : count(0) {}
159 };
160
161 struct {
162 int argc;
163 char **argv;
164 } saved_args;
165}
166
167TEST(LibRGW, TVAR) {
168 typedef boost::variant<uint64_t*, const char*> readdir_offset;
169
170 uint64_t i1{64001};
171 std::string s1{"blunderbuss"};
172
173 readdir_offset v1{&i1};
174 readdir_offset v2{s1.c_str()};
175 readdir_offset v3{static_cast<const char*>(nullptr)};
176
177 uint64_t* pi1 = get<uint64_t*>(v1);
178 ASSERT_NE(pi1, nullptr);
179 std::cout << "read i1: " << *pi1 << std::endl;
180
181 const char* ps1 = get<const char*>(v2);
182 ASSERT_NE(ps1, nullptr);
183 std::cout << "read s1: " << ps1 << std::endl;
184
185 const char* ps3 = get<const char*>(v3);
186 ASSERT_EQ(ps3, nullptr);
187 std::cout << "read s3: " << ps3 << std::endl;
188}
189
190TEST(LibRGW, INIT) {
191 int ret = librgw_create(&rgw_h, saved_args.argc, saved_args.argv);
192 ASSERT_EQ(ret, 0);
193 ASSERT_NE(rgw_h, nullptr);
194}
195
196TEST(LibRGW, MOUNT) {
197 int ret = rgw_mount2(rgw_h, userid.c_str(), access_key.c_str(),
198 secret_key.c_str(), "/", &fs, RGW_MOUNT_FLAG_NONE);
199 ASSERT_EQ(ret, 0);
200 ASSERT_NE(fs, nullptr);
201
202 cct = static_cast<RGWLibFS*>(fs->fs_private)->get_context();
203}
204
205TEST(LibRGW, MARKER1_SETUP_BUCKET) {
206 /* "large" directory enumeration test. this one deals only with
207 * file objects */
208 struct stat st;
209 int ret;
210
211 st.st_uid = owner_uid;
212 st.st_gid = owner_gid;
213 st.st_mode = 755;
214
215 (void) rgw_lookup(fs, fs->root_fh, bucket_name.c_str(), &bucket_fh,
216 RGW_LOOKUP_FLAG_NONE);
217 if (! bucket_fh) {
218 if (do_create) {
219 struct stat st;
220
221 st.st_uid = owner_uid;
222 st.st_gid = owner_gid;
223 st.st_mode = 755;
224
225 ret = rgw_mkdir(fs, fs->root_fh, bucket_name.c_str(), &st, create_mask,
226 &bucket_fh, RGW_MKDIR_FLAG_NONE);
227 ASSERT_EQ(ret, 0);
228 }
229 }
230
231 ASSERT_NE(bucket_fh, nullptr);
232
233 (void) rgw_lookup(fs, bucket_fh, marker_dir.c_str(), &marker_fh,
234 RGW_LOOKUP_FLAG_NONE);
235 if (! marker_fh) {
236 if (do_create) {
237 ret = rgw_mkdir(fs, bucket_fh, marker_dir.c_str(), &st, create_mask,
238 &marker_fh, RGW_MKDIR_FLAG_NONE);
239 ASSERT_EQ(ret, 0);
240 }
241 }
242
243 ASSERT_NE(marker_fh, nullptr);
244}
245
246TEST(LibRGW, MARKER1_SETUP_OBJECTS)
247{
248 /* "large" directory enumeration test. this one deals only with
249 * file objects */
250 if (do_create) {
251 int ret;
252
253 for (int ix = 0; ix < marker_nobjs; ++ix) {
254 std::string object_name("f_");
255 object_name += to_string(ix);
256 obj_rec obj{object_name, nullptr, marker_fh, nullptr};
257 // lookup object--all operations are by handle
258 ret = rgw_lookup(fs, marker_fh, obj.name.c_str(), &obj.fh,
259 RGW_LOOKUP_FLAG_CREATE);
260 ASSERT_EQ(ret, 0);
261 obj.rgw_fh = get_rgwfh(obj.fh);
262 // open object--open transaction
263 ret = rgw_open(fs, obj.fh, 0 /* posix flags */, RGW_OPEN_FLAG_NONE);
264 ASSERT_EQ(ret, 0);
265 ASSERT_TRUE(obj.rgw_fh->is_open());
266 // unstable write data
267 size_t nbytes;
268 string data("data for ");
269 data += object_name;
270 int ret = rgw_write(fs, obj.fh, 0, data.length(), &nbytes,
271 (void*) data.c_str(), RGW_WRITE_FLAG_NONE);
272 ASSERT_EQ(ret, 0);
273 ASSERT_EQ(nbytes, data.length());
274 // commit transaction (write on close)
275 ret = rgw_close(fs, obj.fh, 0 /* flags */);
276 ASSERT_EQ(ret, 0);
277 // save for cleanup
278 marker_objs.push_back(obj);
279 }
280 }
281}
282
283extern "C" {
284 static bool r2_cb(const char* name, void *arg, uint64_t offset,
285 uint32_t flags) {
286 dirent_vec& dvec =
287 *(static_cast<dirent_vec*>(arg));
288 lsubdout(cct, rgw, 10) << __func__
289 << " bucket=" << bucket_name
290 << " dir=" << marker_dir
291 << " iv count=" << dvec.count
292 << " called back name=" << name
293 << " flags=" << flags
294 << dendl;
295
296 std::cout << __func__
297 << " bucket=" << bucket_name
298 << " dir=" << marker_dir
299 << " iv count=" << dvec.count
300 << " called back name=" << name
301 << " flags=" << flags
302 << std::endl;
303
304 string name_str{name};
305 if (! ((name_str == ".") ||
306 (name_str == ".."))) {
307 dvec.obj_names.push_back(dirent_t{std::move(name_str), offset});
308 }
309 return true; /* XXX */
310 }
311}
312
313TEST(LibRGW, MARKER1_READDIR)
314{
315 if (do_marker1) {
316 using std::get;
317
318 dirent_vec dvec;
319 uint64_t offset = 0;
320 bool eof = false;
321
322 /* because RGWReaddirRequest::default_max is 1000 (XXX make
323 * configurable?) and marker_nobjs is 5*1024, the number
324 * of required rgw_readdir operations N should be
325 * marker_nobjs/1000 < N < marker_nobjs/1000+1, i.e., 6 when
326 * marker_nobjs==5*1024 */
327 uint32_t max_iterations = marker_nobjs/1000+1;
328
329 do {
330 ASSERT_TRUE(dvec.count <= max_iterations);
331 int ret = rgw_readdir(fs, marker_fh, &offset, r2_cb, &dvec, &eof,
332 RGW_READDIR_FLAG_DOTDOT);
333 ASSERT_EQ(ret, 0);
334 ASSERT_EQ(offset, get<1>(dvec.obj_names.back())); // cookie check
335 ++dvec.count;
336 } while(!eof);
337 std::cout << "Read " << dvec.obj_names.size() << " objects in "
338 << marker_dir.c_str() << std::endl;
339 }
340}
341
342TEST(LibRGW, MARKER2_READDIR)
343{
344 if (do_marker2) {
345 using std::get;
346
347 dirent_vec dvec;
348 std::string marker{""};
349 bool eof = false;
350
351 /* because RGWReaddirRequest::default_max is 1000 (XXX make
352 * configurable?) and marker_nobjs is 5*1024, the number
353 * of required rgw_readdir operations N should be
354 * marker_nobjs/1000 < N < marker_nobjs/1000+1, i.e., 6 when
355 * marker_nobjs==5*1024 */
356 uint32_t max_iterations = marker_nobjs/1000+1;
357
358 do {
359 ASSERT_TRUE(dvec.count <= max_iterations);
360 int ret = rgw_readdir2(fs, marker_fh,
361 (marker.length() > 0) ? marker.c_str() : nullptr,
362 r2_cb, &dvec, &eof,
363 RGW_READDIR_FLAG_DOTDOT);
364 ASSERT_EQ(ret, 0);
365 marker = get<0>(dvec.obj_names.back());
366 ++dvec.count;
367 } while((!eof) && dvec.count < 4);
368 std::cout << "Read " << dvec.obj_names.size() << " objects in "
369 << marker_dir.c_str() << std::endl;
370 }
371}
372
373TEST(LibRGW, MARKER1_OBJ_CLEANUP)
374{
375 int rc;
376 for (auto& obj : marker_objs) {
377 if (obj.fh) {
378 if (do_delete) {
379 if (verbose) {
380 std::cout << "unlinking: " << bucket_name << ":" << obj.name
381 << std::endl;
382 }
383 rc = rgw_unlink(fs, marker_fh, obj.name.c_str(), RGW_UNLINK_FLAG_NONE);
384 }
385 rc = rgw_fh_rele(fs, obj.fh, 0 /* flags */);
386 ASSERT_EQ(rc, 0);
387 }
388 }
389 marker_objs.clear();
390}
391
392TEST(LibRGW, CLEANUP) {
393 int rc;
394
395 if (do_marker1) {
396 cleanup_queue.push_back(
397 obj_rec{bucket_name, bucket_fh, fs->root_fh, get_rgwfh(fs->root_fh)});
398 }
399
400 for (auto& elt : cleanup_queue) {
401 if (elt.fh) {
402 rc = rgw_fh_rele(fs, elt.fh, 0 /* flags */);
403 ASSERT_EQ(rc, 0);
404 }
405 }
406 cleanup_queue.clear();
407}
408
409TEST(LibRGW, UMOUNT) {
410 if (! fs)
411 return;
412
413 int ret = rgw_umount(fs, RGW_UMOUNT_FLAG_NONE);
414 ASSERT_EQ(ret, 0);
415}
416
417TEST(LibRGW, SHUTDOWN) {
418 librgw_shutdown(rgw_h);
419}
420
421int main(int argc, char *argv[])
422{
423 char *v{nullptr};
424 string val;
425 vector<const char*> args;
426
427 argv_to_vec(argc, const_cast<const char**>(argv), args);
428 env_to_vec(args);
429
430 v = getenv("AWS_ACCESS_KEY_ID");
431 if (v) {
432 access_key = v;
433 }
434
435 v = getenv("AWS_SECRET_ACCESS_KEY");
436 if (v) {
437 secret_key = v;
438 }
439
440 for (auto arg_iter = args.begin(); arg_iter != args.end();) {
441 if (ceph_argparse_witharg(args, arg_iter, &val, "--access",
442 (char*) nullptr)) {
443 access_key = val;
444 } else if (ceph_argparse_witharg(args, arg_iter, &val, "--secret",
445 (char*) nullptr)) {
446 secret_key = val;
447 } else if (ceph_argparse_witharg(args, arg_iter, &val, "--userid",
448 (char*) nullptr)) {
449 userid = val;
450 } else if (ceph_argparse_witharg(args, arg_iter, &val, "--bn",
451 (char*) nullptr)) {
452 bucket_name = val;
453 } else if (ceph_argparse_witharg(args, arg_iter, &val, "--uid",
454 (char*) nullptr)) {
455 owner_uid = std::stoi(val);
456 } else if (ceph_argparse_witharg(args, arg_iter, &val, "--gid",
457 (char*) nullptr)) {
458 owner_gid = std::stoi(val);
459 } else if (ceph_argparse_flag(args, arg_iter, "--marker1",
460 (char*) nullptr)) {
461 do_marker1 = true;
462 } else if (ceph_argparse_flag(args, arg_iter, "--create",
463 (char*) nullptr)) {
464 do_create = true;
465 } else if (ceph_argparse_flag(args, arg_iter, "--delete",
466 (char*) nullptr)) {
467 do_delete = true;
468 } else if (ceph_argparse_flag(args, arg_iter, "--verbose",
469 (char*) nullptr)) {
470 verbose = true;
471 } else {
472 ++arg_iter;
473 }
474 }
475
476 /* dont accidentally run as anonymous */
477 if ((access_key == "") ||
478 (secret_key == "")) {
479 std::cout << argv[0] << " no AWS credentials, exiting" << std::endl;
480 return EPERM;
481 }
482
483 saved_args.argc = argc;
484 saved_args.argv = argv;
485
486 ::testing::InitGoogleTest(&argc, argv);
487 return RUN_ALL_TESTS();
488}