forked from jsmartin/raidformer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
raidformer.py
executable file
·221 lines (173 loc) · 8.61 KB
/
raidformer.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
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
#! /usr/bin/python
import time
import optparse
import sys
from boto import ec2
import boto.utils
from subprocess import Popen, PIPE, STDOUT
import os.path
import glob
def runcmd(cmd):
p = Popen(cmd, shell=True, stdin=PIPE, stdout=PIPE, stderr=STDOUT, close_fds=True)
output = p.stdout.read()
output.strip()
return output
def attached_name(item):
item = item.replace("sd","xvd")
return item
def get_options():
""" command-line options """
usage = "usage: %prog [options]"
OptionParser = optparse.OptionParser
parser = OptionParser(usage)
parser.add_option("-a", "--attach", action="store_true",
dest="attach", default=False, help="Do the volume creation and attachment.")
parser.add_option("-c", "--count", action="store", type="int",
dest="count", default=1, help="Number of EBS volumes ")
parser.add_option("-d", "--device", action="store", type="string",
dest="device", default="/dev/xvdf", help="block device to start with")
parser.add_option("-f", "--filesystem", action="store", type="string",
dest="filesystem", default="ext4", help="filesystem type")
parser.add_option("-l", "--logvol", action="store", type="string",
dest="logvol", default="LogVolData", help="Logical Volume Name")
parser.add_option("-m", "--mountpoint", action="store", type="string",
dest="mountpoint", help="Mountpoint")
parser.add_option("", "--md", action="store", type="string",
dest="md_device", default="/dev/md0", help="md device name ")
parser.add_option("-r", "--raidlevel", action="store", type="int",
dest="raidlevel", help="RAID level")
parser.add_option("-s", "--size", action="store", type="int",
dest="size", help="Size of EBS volumes ")
parser.add_option("-t", "--test", action="store_true",
dest="test", default=False, help="Does a dry run of the mdadm lvm commands.")
parser.add_option("", "--tag", action="store", type="string",
dest="tag", default="ebs_raid", help="Tag name for the ebs devices")
parser.add_option("-v", "--volgroup", action="store", type="string",
dest="volgroup", default="VolGroupData", help="Volume Group Name")
parser.add_option("-w", "--wipe", action="store_true",
dest="wipe", default=False, help="Format the new filesystem")
parser.add_option("", "--from-snapshot", action="store",
dest="snapshot", default=None,
help="Create volumes from one or more ebs snapshots. Order preservation is essential")
parser.add_option("-i", "--iops", action="store", type="int",
dest="iops", default=None, help="Specify provisioned iops")
parser.add_option("", "--type", action="store", type="string",
dest="voltype", default="standard", help="Type of volume to create: gp2 for General Purpose (SSD) volumes, io1 for Provisioned IOPS (SSD) volumes, or standard for Magnetic volumes")
options, args = parser.parse_args()
return options, args
def initialize_raid( cmds, md_device, raidlevel, count, attached_devices ):
cmds.append("echo Y | mdadm --verbose --create %s --level=%s --chunk=256 --raid-devices=%s %s" % ( md_device, str(raidlevel), str(count), ' '.join(attached_devices) ) )
cmds.append("mdadm --detail --scan > /etc/mdadm.conf")
cmds.append("dd if=/dev/zero of=%s bs=512 count=1" % md_device )
cmds.append("pvcreate %s" % md_device )
return cmds
def initialize_raid_from_snapshot( cmds, md_device, attached_devices ):
cmds.append("echo Y | mdadm --verbose --assemble %s %s" % ( md_device, ' '.join(attached_devices) ) )
cmds.append("mdadm --detail --scan > /etc/mdadm.conf")
cmds.append("pvcreate %s" % md_device )
return cmds
def initialize_filesystem(cmds, wipe, md_device, volgroup, logvol, format_cmds, filesystem, mountpoint):
device_short = os.path.basename(md_device).title()
cmds.append("vgcreate %s %s" % ( volgroup, md_device ) )
cmds.append("lvcreate -l 100%%vg -n %s%s %s" % (logvol, device_short, volgroup) )
if wipe is True:
cmds.append("%s /dev/%s/%s%s" % (format_cmds[filesystem], volgroup, logvol, device_short))
cmds.append('echo "/dev/%s/%s%s %s %s %s 1 1" >> /etc/fstab' % (volgroup, logvol, device_short, mountpoint, filesystem, format_fstab_settings[filesystem]) )
cmds.append('mount %s' % mountpoint)
return cmds
options, args = get_options()
format_cmds = {
"ext4": "mkfs.ext4 -j",
"xfs": "mkfs.xfs"
}
format_fstab_settings = {
"ext4": "defaults",
"xfs": "noatime,nodiratime"
}
if options.test:
print "TEST MODE - The following operations will be performed:"
#device lettering
#http://docs.amazonwebservices.com/AWSEC2/latest/UserGuide/ebs-attaching-volume.html
devices = [ '/dev/sdf', '/dev/sdg', '/dev/sdh', '/dev/sdi', '/dev/sdj', '/dev/sdk', '/dev/sdl', '/dev/sdm', '/dev/sdn', '/dev/sdo', '/dev/sdp' ]
if not options.device in devices:
print "You must use a valid device. See http://docs.amazonwebservices.com/AWSEC2/latest/UserGuide/ebs-attaching-volume.html."
sys.exit(1)
if options.attach is True:
if len(glob.glob(options.device)) > 0:
print "You already have devices that start with %s." % options.device
sys.exit(1)
if os.path.exists(options.md_device):
print "Device %s already exists." % options.md_device
sys.exit(1)
my_snapshots = []
if options.snapshot:
my_snapshots = options.snapshot.split(',')
options.count = len(my_snapshots)
if options.wipe:
print "Cowardly will not wipe volumes created from snapshots. Loose the --wipe option if you want to restore from snapshots."
sys.exit(1)
even_number_raidlevels = [10]
if options.raidlevel in even_number_raidlevels and options.count % 2 != 0:
print "Number of volumes to be created in not compatible with raid level %s" % options.raidlevel
sys.exit(1)
instance_data = boto.utils.get_instance_metadata()
# connect to region of the current instance rather than default of us-east-1
zone = instance_data['placement']['availability-zone']
region_name = zone[:-1]
print "Connecting to region %s" % region_name
ec2conn = ec2.connect_to_region(region_name)
# trim the devices list to the ones we want to create
devindex = devices.index(options.device)
devices[0:devindex] = []
devices = devices[:options.count]
# map ebs devices ids to ubuntu devices
attached_devices = map(attached_name, devices)
vol_ids = []
if (options.attach or options.snapshot) and not options.test:
for key, device in enumerate(devices):
if my_snapshots:
snapshot = my_snapshots[key]
print "Restoring snapshot %s to device %s" % (device, snapshot)
vol = ec2conn.create_volume(options.size, instance_data['placement']['availability-zone'], snapshot=snapshot, volume_type=options.voltype, iops=options.iops)
else:
print "Creating new volume on device %s" % device
vol = ec2conn.create_volume(options.size, instance_data['placement']['availability-zone'], volume_type=options.voltype, iops=options.iops)
print "Created volume: ", vol.id
print "Waiting for %s to change to state available" % vol.id
ready = False
while not ready:
created = ec2conn.get_all_volumes([vol.id])[0]
if created.status == "available":
ready = True
print "Volume %s is available" % vol.id
ec2conn.attach_volume(vol.id, instance_data['instance-id'], device)
print "Attached volume: ", vol.id
vol_ids.append(vol.id)
vol.add_tag("Name", options.tag)
vol.add_tag("raidformer_created", '1')
for device in attached_devices:
found = False
while found is False:
print "Waiting for %s to become available." % device
if os.path.exists(device):
print "%s has been found." % device
break
else:
time.sleep(10)
if not os.path.isdir(options.mountpoint):
print "Creating mount point: %s" % options.mountpoint
if not options.test:
os.makedirs(options.mountpoint)
else:
print "WARNING mount point already exists: %s" % options.mountpoint
commands = list()
if options.snapshot:
commands = initialize_raid_from_snapshot(commands, options.md_device, attached_devices )
else:
commands = initialize_raid(commands, options.md_device, options.raidlevel, options.count, attached_devices )
commands = initialize_filesystem(commands, options.wipe, options.md_device, options.volgroup, options.logvol, format_cmds, options.filesystem, options.mountpoint)
for cmd in commands:
print 'Running:', cmd
if options.test is False:
output = runcmd(cmd)
print output