-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathupdate_rbd_directory.py
executable file
·150 lines (123 loc) · 5.41 KB
/
update_rbd_directory.py
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
#!/usr/bin/env python3
"""
Filename: update_rbd_directory.py
Written by: Michael J. Kidd aka linuxkidd
Version: 2.1
Last modified: 2022-03-16
Changelog:
- 2022-03-15 v1.0, python2 only version released (vis gist)
- 2022-03-16 v2.0, python3 and cli arguments parsing added
- 2022-03-16 v2.1, fixed bytes conversion glitch, added '-f' option
Usage help:
./update_rbd_directory.py --help
Purpose:
This script processes the 'rados ls' output, looking for 'rbd_id.$name'
entries, where $name is the name of an RBD.
Once found, it checks the 'rbd_directory' object to ensure the expected
'name_$name' and 'id_$id' entries are present for the found RBD.
If, the 'name_$name' or 'id_$id' omap entries are missing, the script
will stage the missing entries until the end of the object loop, at
which point, the missing entries will be added to the rbd_directory
omap data.
NOTE:
- The 'name_$name' entry contains the ID value
- The 'id_$id' entry contains the NAME value
-- Otherwise, it wouldn't be much of a mapping, would it?
"""
import argparse,rados,rbd,struct
needed_count={ 'name':0, 'id': 0}
keys = []
values = []
antifield={ "name":"id", "id":"name" }
fields={}
"""
appendOmap(fields,myfield):
- fields == dictionary with 'name' and 'id' values
- myfield == the current field, either 'name' or 'id', being checked
The key == value pair are appended to the 'keys' / 'values'
global variables for addition by addOmap() function.
"""
def appendOmap(fields,myfield):
needed_count[myfield]+=1
keys.append("{}_{}".format(myfield,fields[myfield]))
values.append("{}{}".format(struct.pack('<I',len(fields[antifield[myfield]])).decode('ascii'),fields[antifield[myfield]]))
"""
checkIter(iterator, fields, myfield)
- iterator == ioctx iterator returned from 'get_omap_vals_by_keys'
- fields == dictionary with 'name' and 'id' values
- myfield == the current field, either 'name' or 'id', being checked
Try to iterate the omap key, if there are no values, the StopIteration
exception is thrown, meaning the omap key is missing and needs to be
created.
"""
def checkIter(iter,fields,myfield):
try:
mykey = iter.__next__()
except StopIteration:
# Always report the name, since the ID is harder to understand.
print("Missing {} entry {}".format(myfield,fields['name']))
appendOmap(fields,myfield)
"""
loopRados(ioctx)
- ioctx == IO Control provided from cluster.open_ioctx() in __main__
Loops over the rados pool specified by args.pool ( via ioctx ), looking
for objects of name 'rbd_id.$name' where $name is the name of an RBD in
the pool. Then, calls 'checkIter()' to confirm if the name and id keys
for the RBD are present. If not, the needed keys/values are appended
to keys / values lists for later addition via 'addOmap()'.
"""
def loopRados(ioctx):
for object in ioctx.list_objects():
if object.key[:7]=="rbd_id.":
myrbd = rbd.Image(ioctx,name=object.key[7:])
fields={ "name": myrbd.get_name(), "id": myrbd.id() }
if args.force:
for myfield in antifield.keys():
print("Forcing {} entry for {}".format(myfield,fields['name']))
appendOmap(fields,myfield)
else:
with rados.ReadOpCtx() as read_op:
for myfield in antifield.keys():
iter, ret = ioctx.get_omap_vals_by_keys(read_op, tuple(["{}_{}".format(myfield,fields[myfield]),]))
ioctx.operate_read_op(read_op,"rbd_directory")
checkIter(iter,fields,myfield)
"""
addOmap(ioctx)
- ioctx == IO Control provided from cluster.open_ioctx() in __main__
Adds missing omap 'key = value' pairs to 'rbd_directory' object in the
rados pool specified by args.pool ( via ioctx ). These values are
populated by checkIter() called from loopRados().
"""
def addOmap(ioctx):
if needed_count['name']>0 or needed_count['id']>0:
with rados.WriteOpCtx() as write_op:
ioctx.set_omap(write_op, tuple(keys), tuple(values))
write_op.set_flags(rados.LIBRADOS_OPERATION_SKIPRWLOCKS)
ioctx.operate_write_op(write_op, "rbd_directory")
print("Added {} name entries, and {} id entries.".format(needed_count['name'],needed_count['id']))
else:
print("No missing entries.")
ioctx.close()
if __name__ == "__main__":
parser = argparse.ArgumentParser()
parser.add_argument("-c", "--conf", default = "/etc/ceph/ceph.conf", help="Ceph config file to use ( default: /etc/ceph/ceph.conf )")
parser.add_argument("-f", "--force", action = 'store_true', help="Force refresh of all OMAP entries ( default: not-set )")
parser.add_argument("-p", "--pool", default = "rbd", help="Pool to use for RBD directory correction ( default: rbd )")
args = parser.parse_args()
try:
cluster = rados.Rados(conffile=args.conf)
except:
print("Failed to read {}, do you need to specify a different conf file with -c ?".format(args.conf))
exit(1)
try:
cluster.connect()
except:
print("Failed to connect to cluster using: {}, do you need to specify a different conf file with -c ?".format(args.conf))
exit(1)
try:
ioctx = cluster.open_ioctx(args.pool)
except:
print("Failed to open pool {}, do you need to specify a different pool with -p ?".format(args.pool))
exit(1)
loopRados(ioctx)
addOmap(ioctx)