#!/usr/bin/env python
import sys
-import os.path
+import time
+import os
+import random
+import string
from subprocess import call, PIPE, Popen
from invirt.config import structs as config
def error():
print >>sys.stderr, "Error removing LV %s\n" % lvname
sys.exit(1)
- # I know this is the wrong answer, but sometimes the first
- # lvchange -a n fails for no particularly good reason, so this is
- # a pretty good workaround
- call(["/sbin/lvchange", "-a", "n", lvpath])
- rv = call(["/sbin/lvchange", "-a", "n", lvpath])
- if rv != 0:
- error()
- rv = call(["/sbin/lvchange", "-a", "ey", lvpath])
- if rv != 0:
- error()
- rv = call(["/sbin/lvremove", "--force", lvpath])
- if rv != 0:
- error()
+
+ # Rename the LV to something else so we can wipe it before reusing
+ # the space
+ while True:
+ new_lvname = "old_%s_%s" % (lvname, ''.join(random.choice(string.ascii_letters) for i in xrange(6)))
+ new_lvpath = "/dev/%s/%s" % (vg, new_lvname)
+ p = Popen(["/sbin/lvrename", lvpath, new_lvpath], stdout=PIPE, stderr=PIPE)
+ rv = p.wait()
+ if rv == 5 and 'already exists in volume group' in p.stderr.read():
+ continue
+ elif rv != 0:
+ error()
+ else:
+ break
ensureoff(machine)
+
+ # Fork. The child process wipes the LV and then deletes
+ # it. There's not really anything sane to do with errors (since
+ # this is running non-interactively), so let's just drop them on
+ # the floor for now.
+ if os.fork() != 0:
+ call(["/bin/dd", "if=/dev/zero", "of=%s" % new_lvpath])
+ call(["/sbin/lvchange", "-a", "n", new_lvpath])
+ call(["/sbin/lvchange", "-a", "ey", new_lvpath])
+ call(["/sbin/lvremove", "--force", new_lvpath])
elif subcommand == "lvresize":
size = sys.argv[4]
ensureoff(machine)
if rv != 0:
print >>sys.stderr, "Error creating LV %s\n" %(lvname,)
sys.exit(1)
-