1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
// vim: ts=8 sw=2 smarttab
#include "librbd/image/GetMetadataRequest.h"
#include "cls/rbd/cls_rbd_client.h"
#include "common/dout.h"
#include "common/errno.h"
#include "include/ceph_assert.h"
#include "librbd/ImageCtx.h"
#include "librbd/Utils.h"
#include <boost/algorithm/string/predicate.hpp>
#define dout_subsys ceph_subsys_rbd
#undef dout_prefix
#define dout_prefix *_dout << "librbd::image::GetMetadataRequest: " \
<< this << " " << __func__ << ": "
#define MAX_KEYS 64U
namespace librbd {
namespace image {
namespace {
static const std::string INTERNAL_KEY_PREFIX{".rbd"};
} // anonymous namespace
using util::create_rados_callback;
template <typename I>
GetMetadataRequest<I>::GetMetadataRequest(
IoCtx &io_ctx, const std::string &oid, bool filter_internal,
const std::string& filter_key_prefix, const std::string& last_key,
uint32_t max_results, KeyValues* key_values, Context *on_finish)
: m_io_ctx(io_ctx), m_oid(oid), m_filter_internal(filter_internal),
m_filter_key_prefix(filter_key_prefix), m_last_key(last_key),
m_max_results(max_results), m_key_values(key_values),
m_on_finish(on_finish),
m_cct(reinterpret_cast<CephContext*>(m_io_ctx.cct())) {
}
template <typename I>
void GetMetadataRequest<I>::send() {
metadata_list();
}
template <typename I>
void GetMetadataRequest<I>::metadata_list() {
ldout(m_cct, 15) << "start_key=" << m_last_key << dendl;
m_expected_results = MAX_KEYS;
if (m_max_results > 0) {
m_expected_results = std::min<uint32_t>(
m_expected_results, m_max_results - m_key_values->size());
}
librados::ObjectReadOperation op;
cls_client::metadata_list_start(&op, m_last_key, m_expected_results);
auto aio_comp = create_rados_callback<
GetMetadataRequest<I>, &GetMetadataRequest<I>::handle_metadata_list>(this);
m_out_bl.clear();
m_io_ctx.aio_operate(m_oid, aio_comp, &op, &m_out_bl);
aio_comp->release();
}
template <typename I>
void GetMetadataRequest<I>::handle_metadata_list(int r) {
ldout(m_cct, 15) << "r=" << r << dendl;
KeyValues metadata;
if (r == 0) {
auto it = m_out_bl.cbegin();
r = cls_client::metadata_list_finish(&it, &metadata);
}
if (r == -ENOENT || r == -EOPNOTSUPP) {
finish(0);
return;
} else if (r < 0) {
lderr(m_cct) << "failed to retrieve image metadata: " << cpp_strerror(r)
<< dendl;
finish(r);
return;
}
for (auto it = metadata.begin(); it != metadata.end(); ++it) {
if (m_filter_internal &&
boost::starts_with(it->first, INTERNAL_KEY_PREFIX)) {
continue;
} else if (!m_filter_key_prefix.empty() &&
!boost::starts_with(it->first, m_filter_key_prefix)) {
continue;
}
m_key_values->insert({it->first, std::move(it->second)});
}
if (!metadata.empty()) {
m_last_key = metadata.rbegin()->first;
}
if (metadata.size() == m_expected_results &&
(m_max_results == 0 || m_key_values->size() < m_max_results)) {
metadata_list();
return;
}
finish(0);
}
template <typename I>
void GetMetadataRequest<I>::finish(int r) {
ldout(m_cct, 15) << "r=" << r << dendl;
m_on_finish->complete(r);
delete this;
}
} // namespace image
} // namespace librbd
template class librbd::image::GetMetadataRequest<librbd::ImageCtx>;
|