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
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
|
from abc import ABCMeta, abstractmethod
from six import StringIO
import json
from .conn import get_gateway_connection, get_gateway_secure_connection
class Cluster:
""" interface to run commands against a distinct ceph cluster """
__metaclass__ = ABCMeta
@abstractmethod
def admin(self, args = None, **kwargs):
""" execute a radosgw-admin command """
pass
class Gateway:
""" interface to control a single radosgw instance """
__metaclass__ = ABCMeta
def __init__(self, host = None, port = None, cluster = None, zone = None, ssl_port = 0):
self.host = host
self.port = port
self.cluster = cluster
self.zone = zone
self.connection = None
self.secure_connection = None
self.ssl_port = ssl_port
@abstractmethod
def start(self, args = []):
""" start the gateway with the given args """
pass
@abstractmethod
def stop(self):
""" stop the gateway """
pass
def endpoint(self):
return 'http://%s:%d' % (self.host, self.port)
class SystemObject:
""" interface for system objects, represented in json format and
manipulated with radosgw-admin commands """
__metaclass__ = ABCMeta
def __init__(self, data = None, uuid = None):
self.data = data
self.id = uuid
if data:
self.load_from_json(data)
@abstractmethod
def build_command(self, command):
""" return the command line for the given command, including arguments
to specify this object """
pass
@abstractmethod
def load_from_json(self, data):
""" update internal state based on json data """
pass
def command(self, cluster, cmd, args = None, **kwargs):
""" run the given command and return the output and retcode """
args = self.build_command(cmd) + (args or [])
return cluster.admin(args, **kwargs)
def json_command(self, cluster, cmd, args = None, **kwargs):
""" run the given command, parse the output and return the resulting
data and retcode """
s, r = self.command(cluster, cmd, args or [], **kwargs)
if r == 0:
output = s[s.find('{'):] # trim extra output before json
data = json.loads(output)
self.load_from_json(data)
self.data = data
return self.data, r
# mixins for supported commands
class Create(object):
def create(self, cluster, args = None, **kwargs):
""" create the object with the given arguments """
return self.json_command(cluster, 'create', args, **kwargs)
class Delete(object):
def delete(self, cluster, args = None, **kwargs):
""" delete the object """
# not json_command() because delete has no output
_, r = self.command(cluster, 'delete', args, **kwargs)
if r == 0:
self.data = None
return r
class Get(object):
def get(self, cluster, args = None, **kwargs):
""" read the object from storage """
kwargs['read_only'] = True
return self.json_command(cluster, 'get', args, **kwargs)
class Set(object):
def set(self, cluster, data, args = None, **kwargs):
""" set the object by json """
kwargs['stdin'] = StringIO(json.dumps(data))
return self.json_command(cluster, 'set', args, **kwargs)
class Modify(object):
def modify(self, cluster, args = None, **kwargs):
""" modify the object with the given arguments """
return self.json_command(cluster, 'modify', args, **kwargs)
class CreateDelete(Create, Delete): pass
class GetSet(Get, Set): pass
class Zone(SystemObject, SystemObject.CreateDelete, SystemObject.GetSet, SystemObject.Modify):
def __init__(self, name, zonegroup = None, cluster = None, data = None, zone_id = None, gateways = None):
self.name = name
self.zonegroup = zonegroup
self.cluster = cluster
self.gateways = gateways or []
super(Zone, self).__init__(data, zone_id)
def zone_arg(self):
""" command-line argument to specify this zone """
return ['--rgw-zone', self.name]
def zone_args(self):
""" command-line arguments to specify this zone/zonegroup/realm """
args = self.zone_arg()
if self.zonegroup:
args += self.zonegroup.zonegroup_args()
return args
def build_command(self, command):
""" build a command line for the given command and args """
return ['zone', command] + self.zone_args()
def load_from_json(self, data):
""" load the zone from json """
self.id = data['id']
self.name = data['name']
def start(self, args = None):
""" start all gateways """
for g in self.gateways:
g.start(args)
def stop(self):
""" stop all gateways """
for g in self.gateways:
g.stop()
def period(self):
return self.zonegroup.period if self.zonegroup else None
def realm(self):
return self.zonegroup.realm() if self.zonegroup else None
def is_read_only(self):
return False
def tier_type(self):
raise NotImplementedError
def has_buckets(self):
return True
def get_conn(self, credentials):
return ZoneConn(self, credentials) # not implemented, but can be used
class ZoneConn(object):
def __init__(self, zone, credentials):
self.zone = zone
self.name = zone.name
""" connect to the zone's first gateway """
if isinstance(credentials, list):
self.credentials = credentials[0]
else:
self.credentials = credentials
if self.zone.gateways is not None:
self.conn = get_gateway_connection(self.zone.gateways[0], self.credentials)
self.secure_conn = get_gateway_secure_connection(self.zone.gateways[0], self.credentials)
def get_connection(self):
return self.conn
def get_bucket(self, bucket_name, credentials):
raise NotImplementedError
def check_bucket_eq(self, zone, bucket_name):
raise NotImplementedError
class ZoneGroup(SystemObject, SystemObject.CreateDelete, SystemObject.GetSet, SystemObject.Modify):
def __init__(self, name, period = None, data = None, zonegroup_id = None, zones = None, master_zone = None):
self.name = name
self.period = period
self.zones = zones or []
self.master_zone = master_zone
super(ZoneGroup, self).__init__(data, zonegroup_id)
self.rw_zones = []
self.ro_zones = []
self.zones_by_type = {}
for z in self.zones:
if z.is_read_only():
self.ro_zones.append(z)
else:
self.rw_zones.append(z)
def zonegroup_arg(self):
""" command-line argument to specify this zonegroup """
return ['--rgw-zonegroup', self.name]
def zonegroup_args(self):
""" command-line arguments to specify this zonegroup/realm """
args = self.zonegroup_arg()
realm = self.realm()
if realm:
args += realm.realm_arg()
return args
def build_command(self, command):
""" build a command line for the given command and args """
return ['zonegroup', command] + self.zonegroup_args()
def zone_by_id(self, zone_id):
""" return the matching zone by id """
for zone in self.zones:
if zone.id == zone_id:
return zone
return None
def load_from_json(self, data):
""" load the zonegroup from json """
self.id = data['id']
self.name = data['name']
master_id = data['master_zone']
if not self.master_zone or master_id != self.master_zone.id:
self.master_zone = self.zone_by_id(master_id)
def add(self, cluster, zone, args = None, **kwargs):
""" add an existing zone to the zonegroup """
args = zone.zone_arg() + (args or [])
data, r = self.json_command(cluster, 'add', args, **kwargs)
if r == 0:
zone.zonegroup = self
self.zones.append(zone)
return data, r
def remove(self, cluster, zone, args = None, **kwargs):
""" remove an existing zone from the zonegroup """
args = zone.zone_arg() + (args or [])
data, r = self.json_command(cluster, 'remove', args, **kwargs)
if r == 0:
zone.zonegroup = None
self.zones.remove(zone)
return data, r
def realm(self):
return self.period.realm if self.period else None
class Period(SystemObject, SystemObject.Get):
def __init__(self, realm = None, data = None, period_id = None, zonegroups = None, master_zonegroup = None):
self.realm = realm
self.zonegroups = zonegroups or []
self.master_zonegroup = master_zonegroup
super(Period, self).__init__(data, period_id)
def zonegroup_by_id(self, zonegroup_id):
""" return the matching zonegroup by id """
for zonegroup in self.zonegroups:
if zonegroup.id == zonegroup_id:
return zonegroup
return None
def build_command(self, command):
""" build a command line for the given command and args """
return ['period', command]
def load_from_json(self, data):
""" load the period from json """
self.id = data['id']
master_id = data['master_zonegroup']
if not self.master_zonegroup or master_id != self.master_zonegroup.id:
self.master_zonegroup = self.zonegroup_by_id(master_id)
def update(self, zone, args = None, **kwargs):
""" run 'radosgw-admin period update' on the given zone """
assert(zone.cluster)
args = zone.zone_args() + (args or [])
if kwargs.pop('commit', False):
args.append('--commit')
return self.json_command(zone.cluster, 'update', args, **kwargs)
def commit(self, zone, args = None, **kwargs):
""" run 'radosgw-admin period commit' on the given zone """
assert(zone.cluster)
args = zone.zone_args() + (args or [])
return self.json_command(zone.cluster, 'commit', args, **kwargs)
class Realm(SystemObject, SystemObject.CreateDelete, SystemObject.GetSet):
def __init__(self, name, period = None, data = None, realm_id = None):
self.name = name
self.current_period = period
super(Realm, self).__init__(data, realm_id)
def realm_arg(self):
""" return the command-line arguments that specify this realm """
return ['--rgw-realm', self.name]
def build_command(self, command):
""" build a command line for the given command and args """
return ['realm', command] + self.realm_arg()
def load_from_json(self, data):
""" load the realm from json """
self.id = data['id']
def pull(self, cluster, gateway, credentials, args = [], **kwargs):
""" pull an existing realm from the given gateway """
args += ['--url', gateway.endpoint()]
args += credentials.credential_args()
return self.json_command(cluster, 'pull', args, **kwargs)
def master_zonegroup(self):
""" return the current period's master zonegroup """
if self.current_period is None:
return None
return self.current_period.master_zonegroup
def meta_master_zone(self):
""" return the current period's metadata master zone """
zonegroup = self.master_zonegroup()
if zonegroup is None:
return None
return zonegroup.master_zone
class Credentials:
def __init__(self, access_key, secret):
self.access_key = access_key
self.secret = secret
def credential_args(self):
return ['--access-key', self.access_key, '--secret', self.secret]
class User(SystemObject):
def __init__(self, uid, data = None, name = None, credentials = None, tenant = None):
self.name = name
self.credentials = credentials or []
self.tenant = tenant
super(User, self).__init__(data, uid)
def user_arg(self):
""" command-line argument to specify this user """
args = ['--uid', self.id]
if self.tenant:
args += ['--tenant', self.tenant]
return args
def build_command(self, command):
""" build a command line for the given command and args """
return ['user', command] + self.user_arg()
def load_from_json(self, data):
""" load the user from json """
self.id = data['user_id']
self.name = data['display_name']
self.credentials = [Credentials(k['access_key'], k['secret_key']) for k in data['keys']]
def create(self, zone, args = None, **kwargs):
""" create the user with the given arguments """
assert(zone.cluster)
args = zone.zone_args() + (args or [])
return self.json_command(zone.cluster, 'create', args, **kwargs)
def info(self, zone, args = None, **kwargs):
""" read the user from storage """
assert(zone.cluster)
args = zone.zone_args() + (args or [])
kwargs['read_only'] = True
return self.json_command(zone.cluster, 'info', args, **kwargs)
def delete(self, zone, args = None, **kwargs):
""" delete the user """
assert(zone.cluster)
args = zone.zone_args() + (args or [])
return self.command(zone.cluster, 'delete', args, **kwargs)
|