]> git.proxmox.com Git - pve-storage.git/blame - PVE/Storage/RBDPlugin.pm
rbd : volume_snapshot : parse volname
[pve-storage.git] / PVE / Storage / RBDPlugin.pm
CommitLineData
0509010d
AD
1package PVE::Storage::RBDPlugin;
2
3use strict;
4use warnings;
5use IO::File;
6use PVE::Tools qw(run_command trim);
7use PVE::Storage::Plugin;
8use PVE::JSONSchema qw(get_standard_option);
9
10use base qw(PVE::Storage::Plugin);
11
249cb647
SP
12sub rbd_unittobytes {
13 {
14 "M" => 1024*1024,
15 "G" => 1024*1024*1024,
16 "T" => 1024*1024*1024*1024,
17 }
18}
19
411476cd
DM
20my $rbd_cmd = sub {
21 my ($scfg, $storeid, $op, @options) = @_;
0509010d 22
e5427b00 23 my $monhost = $scfg->{monhost};
0509010d
AD
24 $monhost =~ s/;/,/g;
25
411476cd
DM
26 my $cmd = ['/usr/bin/rbd', '-p', $scfg->{pool}, '-m', $monhost, '-n',
27 "client.$scfg->{username}",
28 '--keyring', "/etc/pve/priv/ceph/${storeid}.keyring",
29 '--auth_supported', $scfg->{authsupported}, $op];
3e195ccc 30
411476cd 31 push @$cmd, @options if scalar(@options);
3e195ccc 32
411476cd
DM
33 return $cmd;
34};
0509010d 35
69589444
AD
36my $rados_cmd = sub {
37 my ($scfg, $storeid, $op, @options) = @_;
38
39 my $monhost = $scfg->{monhost};
40 $monhost =~ s/;/,/g;
41
42 my $cmd = ['/usr/bin/rados', '-p', $scfg->{pool}, '-m', $monhost, '-n',
43 "client.$scfg->{username}",
44 '--keyring', "/etc/pve/priv/ceph/${storeid}.keyring",
45 '--auth_supported', $scfg->{authsupported}, $op];
46
47 push @$cmd, @options if scalar(@options);
48
49 return $cmd;
50};
51
411476cd
DM
52sub rbd_ls {
53 my ($scfg, $storeid) = @_;
d70e7f6c 54
7cb2889a 55 my $cmd = &$rbd_cmd($scfg, $storeid, 'ls', '-l');
d70e7f6c 56
411476cd 57 my $list = {};
0509010d 58
8c3abf12 59 my $parser = sub {
411476cd 60 my $line = shift;
0509010d 61
ca1e168a
AD
62 if ($line =~ m/^((vm|base)-(\d+)-disk-\d+)\s+(\d+)(M|G|T)\s((\S+)\/((vm|base)-\d+-\S+@\S+))?/) {
63 my ($image, $owner, $size, $unit, $parent) = ($1, $3, $4, $5, $8);
0509010d 64
411476cd
DM
65 $list->{$scfg->{pool}}->{$image} = {
66 name => $image,
249cb647 67 size => $size*rbd_unittobytes()->{$unit},
62b98a65 68 parent => $parent,
411476cd
DM
69 vmid => $owner
70 };
71 }
8c3abf12
DM
72 };
73
74 eval {
75 run_command($cmd, errmsg => "rbd error", errfunc => sub {}, outfunc => $parser);
76 };
77 my $err = $@;
78
79 die $err if $err && $err !~ m/doesn't contain rbd images/ ;
411476cd
DM
80
81 return $list;
0509010d
AD
82}
83
62b98a65 84sub rbd_volume_info {
992e6835
AD
85 my ($scfg, $storeid, $volname, $snap) = @_;
86
87 my $cmd = undef;
88
89 if($snap){
90 $cmd = &$rbd_cmd($scfg, $storeid, 'info', $volname, '--snap', $snap);
91 }else{
92 $cmd = &$rbd_cmd($scfg, $storeid, 'info', $volname);
93 }
e110213e 94
e110213e 95 my $size = undef;
62b98a65 96 my $parent = undef;
992e6835
AD
97 my $format = undef;
98 my $protected = undef;
62b98a65 99
e110213e
AD
100 my $parser = sub {
101 my $line = shift;
102
249cb647
SP
103 if ($line =~ m/size (\d+) (M|G|T)B in (\d+) objects/) {
104 $size = $1 * rbd_unittobytes()->{$2} if ($1);
62b98a65
AD
105 } elsif ($line =~ m/parent:\s(\S+)\/(\S+)/) {
106 $parent = $2;
992e6835
AD
107 } elsif ($line =~ m/format:\s(\d+)/) {
108 $format = $1;
109 } elsif ($line =~ m/protected:\s(\S+)/) {
110 $protected = 1 if $1 eq "True";
e110213e 111 }
992e6835 112
e110213e
AD
113 };
114
115 run_command($cmd, errmsg => "rbd error", errfunc => sub {}, outfunc => $parser);
116
992e6835 117 return ($size, $parent, $format, $protected);
e110213e
AD
118}
119
0509010d
AD
120sub addslashes {
121 my $text = shift;
122 $text =~ s/;/\\;/g;
123 $text =~ s/:/\\:/g;
124 return $text;
125}
126
e5427b00 127# Configuration
0509010d 128
e5427b00
AD
129PVE::JSONSchema::register_format('pve-storage-monhost', \&parse_monhost);
130sub parse_monhost {
0509010d
AD
131 my ($name, $noerr) = @_;
132
133 if ($name !~ m/^[a-z][a-z0-9\-\_\.]*[a-z0-9]$/i) {
134 return undef if $noerr;
135 die "lvm name '$name' contains illegal characters\n";
136 }
137
138 return $name;
139}
140
0509010d
AD
141sub type {
142 return 'rbd';
143}
144
145sub plugindata {
146 return {
147 content => [ {images => 1}, { images => 1 }],
148 };
149}
150
151sub properties {
152 return {
e5427b00 153 monhost => {
0509010d 154 description => "Monitors daemon ips.",
e5427b00 155 type => 'string',
0509010d 156 },
e5427b00
AD
157 pool => {
158 description => "Pool.",
0509010d
AD
159 type => 'string',
160 },
e5427b00
AD
161 username => {
162 description => "RBD Id.",
0509010d
AD
163 type => 'string',
164 },
e5427b00 165 authsupported => {
0509010d
AD
166 description => "Authsupported.",
167 type => 'string',
168 },
169 };
170}
171
172sub options {
173 return {
35d6dfaf
AD
174 nodes => { optional => 1 },
175 disable => { optional => 1 },
e5427b00 176 monhost => { fixed => 1 },
35d6dfaf 177 pool => { fixed => 1 },
e5427b00 178 username => { fixed => 1 },
35d6dfaf 179 authsupported => { fixed => 1 },
0509010d
AD
180 content => { optional => 1 },
181 };
182}
183
184# Storage implementation
185
186sub parse_volname {
187 my ($class, $volname) = @_;
188
d04c7e55
AD
189 if ($volname =~ m/^((base-(\d+)-\S+)\/)?((base)?(vm)?-(\d+)-\S+)$/) {
190 return ('images', $4, $7, $2, $3, $5);
0509010d
AD
191 }
192
193 die "unable to parse rbd volume name '$volname'\n";
194}
195
196sub path {
e5427b00 197 my ($class, $scfg, $volname, $storeid) = @_;
0509010d
AD
198
199 my ($vtype, $name, $vmid) = $class->parse_volname($volname);
200
e5427b00
AD
201 my $monhost = addslashes($scfg->{monhost});
202 my $pool = $scfg->{pool};
203 my $username = $scfg->{username};
204 my $authsupported = addslashes($scfg->{authsupported});
205
4e2d3bc8 206 my $path = "rbd:$pool/$name:id=$username:auth_supported=$authsupported:keyring=/etc/pve/priv/ceph/$storeid.keyring:mon_host=$monhost";
0509010d
AD
207
208 return ($path, $vmid, $vtype);
209}
210
5b9b9b14
AD
211my $find_free_diskname = sub {
212 my ($storeid, $scfg, $vmid) = @_;
213
214 my $rbd = rbd_ls($scfg, $storeid);
215 my $disk_ids = {};
216 my $dat = $rbd->{$scfg->{pool}};
217
218 foreach my $image (keys %$dat) {
219 my $volname = $dat->{$image}->{name};
220 if ($volname =~ m/(vm|base)-$vmid-disk-(\d+)/){
221 $disk_ids->{$2} = 1;
222 }
223 }
224 #fix: can we search in $rbd hash key with a regex to find (vm|base) ?
225 for (my $i = 1; $i < 100; $i++) {
226 if (!$disk_ids->{$i}) {
227 return "vm-$vmid-disk-$i";
228 }
229 }
230
231 die "unable to allocate an image name for VM $vmid in storage '$storeid'\n";
232};
233
5eab0272
DM
234sub create_base {
235 my ($class, $storeid, $scfg, $volname) = @_;
236
992e6835
AD
237 my $snap = '__base__';
238
239 my ($vtype, $name, $vmid, $basename, $basevmid, $isBase) =
240 $class->parse_volname($volname);
241
242 die "create_base not possible with base image\n" if $isBase;
243
244 my ($size, $parent, $format, undef) = rbd_volume_info($scfg, $storeid, $name);
245 die "rbd volume info on '$name' failed\n" if !($size);
246
247 die "rbd image must be at format V2" if $format ne "2";
248
249 die "volname '$volname' contains wrong information about parent $parent $basename\n"
250 if $basename && (!$parent || $parent ne $basename."@".$snap);
251
252 my $newname = $name;
253 $newname =~ s/^vm-/base-/;
254
255 my $newvolname = $basename ? "$basename/$newname" : "$newname";
256
257 my $cmd = &$rbd_cmd($scfg, $storeid, 'rename', $name, $newname);
258 run_command($cmd, errmsg => "rbd rename $name' error", errfunc => sub {});
259
260 my $running = undef; #fixme : is create_base always offline ?
261
262 $class->volume_snapshot($scfg, $storeid, $newname, $snap, $running);
263
264 my (undef, undef, undef, $protected) = rbd_volume_info($scfg, $storeid, $newname, $snap);
265
266 if (!$protected){
267 my $cmd = &$rbd_cmd($scfg, $storeid, 'snap', 'protect', $newname, '--snap', $snap);
268 run_command($cmd, errmsg => "rbd protect $newname snap $snap' error", errfunc => sub {});
269 }
270
271 return $newvolname;
272
5eab0272
DM
273}
274
275sub clone_image {
276 my ($class, $scfg, $storeid, $volname, $vmid) = @_;
277
f2708285
AD
278 my $snap = '__base__';
279
280 my ($vtype, $basename, $basevmid, undef, undef, $isBase) =
281 $class->parse_volname($volname);
282
283 die "clone_image onyl works on base images\n" if !$isBase;
284
285 my $name = &$find_free_diskname($storeid, $scfg, $vmid);
286
287 warn "clone $volname: $basename to $name\n";
288
289 my $newvol = "$basename/$name";
290
291 my $cmd = &$rbd_cmd($scfg, $storeid, 'clone', $basename, '--snap', $snap, $name);
292 run_command($cmd, errmsg => "rbd clone $basename' error", errfunc => sub {});
293
294 return $newvol;
5eab0272
DM
295}
296
0509010d
AD
297sub alloc_image {
298 my ($class, $storeid, $scfg, $vmid, $fmt, $name, $size) = @_;
299
300
e5427b00 301 die "illegal name '$name' - sould be 'vm-$vmid-*'\n"
0509010d 302 if $name && $name !~ m/^vm-$vmid-/;
0509010d 303
8b3ae518 304 $name = &$find_free_diskname($storeid, $scfg, $vmid);
0509010d 305
166196ac 306 my $cmd = &$rbd_cmd($scfg, $storeid, 'create', '--format' , 2, '--size', ($size/1024), $name);
411476cd 307 run_command($cmd, errmsg => "rbd create $name' error", errfunc => sub {});
0509010d
AD
308
309 return $name;
310}
311
312sub free_image {
32437ed2 313 my ($class, $storeid, $scfg, $volname, $isBase) = @_;
0509010d 314
42d07b9a
AD
315 my ($vtype, $name, $vmid, undef, undef, undef) =
316 $class->parse_volname($volname);
317
318 if ($isBase) {
319 my $snap = '__base__';
320 my (undef, undef, undef, $protected) = rbd_volume_info($scfg, $storeid, $name, $snap);
321 if ($protected){
322 my $cmd = &$rbd_cmd($scfg, $storeid, 'snap', 'unprotect', $name, '--snap', $snap);
323 run_command($cmd, errmsg => "rbd unprotect $name snap $snap' error", errfunc => sub {});
324 }
325 }
326
327 my $cmd = &$rbd_cmd($scfg, $storeid, 'snap', 'purge', $name);
c30470a3
AD
328 run_command($cmd, errmsg => "rbd snap purge $volname' error", outfunc => sub {}, errfunc => sub {});
329
42d07b9a 330 $cmd = &$rbd_cmd($scfg, $storeid, 'rm', $name);
411476cd 331 run_command($cmd, errmsg => "rbd rm $volname' error", outfunc => sub {}, errfunc => sub {});
0509010d
AD
332
333 return undef;
334}
335
336sub list_images {
337 my ($class, $storeid, $scfg, $vmid, $vollist, $cache) = @_;
338
e5427b00 339 $cache->{rbd} = rbd_ls($scfg, $storeid) if !$cache->{rbd};
411476cd 340
0509010d
AD
341 my $res = [];
342
411476cd 343 if (my $dat = $cache->{rbd}->{$scfg->{pool}}) {
0509010d
AD
344 foreach my $image (keys %$dat) {
345
346 my $volname = $dat->{$image}->{name};
347
348 my $volid = "$storeid:$volname";
349
0509010d
AD
350 my $owner = $dat->{$volname}->{vmid};
351 if ($vollist) {
352 my $found = grep { $_ eq $volid } @$vollist;
353 next if !$found;
354 } else {
355 next if defined ($vmid) && ($owner ne $vmid);
356 }
357
358 my $info = $dat->{$volname};
359 $info->{volid} = $volid;
411476cd 360 $info->{format} = 'raw';
0509010d
AD
361
362 push @$res, $info;
363 }
364 }
365
411476cd 366 return $res;
0509010d
AD
367}
368
0509010d
AD
369sub status {
370 my ($class, $storeid, $scfg, $cache) = @_;
371
69589444
AD
372 my $cmd = &$rados_cmd($scfg, $storeid, 'df');
373
374 my $stats = {};
375
376 my $parser = sub {
377 my $line = shift;
378 if ($line =~ m/^\s+total\s(\S+)\s+(\d+)/) {
379 $stats->{$1} = $2;
380 }
381 };
382
383 eval {
384 run_command($cmd, errmsg => "rados error", errfunc => sub {}, outfunc => $parser);
385 };
386
387 my $total = $stats->{space} ? $stats->{space}*1024 : 0;
388 my $free = $stats->{avail} ? $stats->{avail}*1024 : 0;
389 my $used = $stats->{used} ? $stats->{used}*1024: 0;
0509010d 390 my $active = 1;
0509010d 391
411476cd 392 return ($total, $free, $used, $active);
0509010d
AD
393}
394
395sub activate_storage {
396 my ($class, $storeid, $scfg, $cache) = @_;
397 return 1;
398}
399
400sub deactivate_storage {
401 my ($class, $storeid, $scfg, $cache) = @_;
402 return 1;
403}
404
405sub activate_volume {
406 my ($class, $storeid, $scfg, $volname, $exclusive, $cache) = @_;
407 return 1;
408}
409
410sub deactivate_volume {
411 my ($class, $storeid, $scfg, $volname, $exclusive, $cache) = @_;
412 return 1;
413}
414
0002d9cc
AD
415sub volume_size_info {
416 my ($class, $scfg, $storeid, $volname, $timeout) = @_;
417
81d1d017
AD
418 my ($vtype, $name, $vmid) = $class->parse_volname($volname);
419 my ($size, undef) = rbd_volume_info($scfg, $storeid, $name);
62b98a65 420 return $size;
0002d9cc
AD
421}
422
e7a42a76
AD
423sub volume_resize {
424 my ($class, $scfg, $storeid, $volname, $size, $running) = @_;
425
426 return 1 if $running;
427
478fc06c
AD
428 my ($vtype, $name, $vmid) = $class->parse_volname($volname);
429
430 my $cmd = &$rbd_cmd($scfg, $storeid, 'resize', '--size', ($size/1024/1024), $name);
e7a42a76
AD
431 run_command($cmd, errmsg => "rbd resize $volname' error", errfunc => sub {});
432 return undef;
433}
434
788dd8e1
AD
435sub volume_snapshot {
436 my ($class, $scfg, $storeid, $volname, $snap, $running) = @_;
437
438 return 1 if $running;
439
9af33ed0
AD
440 my ($vtype, $name, $vmid) = $class->parse_volname($volname);
441
442 my $cmd = &$rbd_cmd($scfg, $storeid, 'snap', 'create', '--snap', $snap, $name);
788dd8e1
AD
443 run_command($cmd, errmsg => "rbd snapshot $volname' error", errfunc => sub {});
444 return undef;
445}
446
5a2b2e2f
AD
447sub volume_snapshot_rollback {
448 my ($class, $scfg, $storeid, $volname, $snap) = @_;
449
450 my $cmd = &$rbd_cmd($scfg, $storeid, 'snap', 'rollback', '--snap', $snap, $volname);
451 run_command($cmd, errmsg => "rbd snapshot $volname to $snap' error", errfunc => sub {});
452}
453
cce29bcd
AD
454sub volume_snapshot_delete {
455 my ($class, $scfg, $storeid, $volname, $snap, $running) = @_;
456
457 return 1 if $running;
458
459 my $cmd = &$rbd_cmd($scfg, $storeid, 'snap', 'rm', '--snap', $snap, $volname);
460 run_command($cmd, errmsg => "rbd snapshot $volname' error", errfunc => sub {});
461 return undef;
462}
463
774f21b9
AD
464sub volume_has_feature {
465 my ($class, $scfg, $feature, $storeid, $volname, $snapname, $running) = @_;
466
467 my $features = {
468 snapshot => { current => 1, snap => 1},
469 clone => { snap => 1},
470 };
471
472 my $snap = $snapname ? 'snap' : 'current';
473 return 1 if $features->{$feature}->{$snap};
474
475 return undef;
476}
477
0509010d 4781;