* Added option for EC2 keypair.
* Added workaround for evaluation * Add support for cluster instances ami generation, only works in us-east-1 for now. e.g. ./create-ebs-ami.py --region us-east --hvm --key rob Cluster instances need to have a 8GB EBS backed image with 1 partition and with grub installed. Also HVM images can (currently) only be generated using the CreateImage API call, which requires a running or stopped instance. svn path=/nixos/trunk/; revision=33827
This commit is contained in:
parent
b4afefcc40
commit
be31aebdcf
@ -9,25 +9,46 @@ import argparse
|
|||||||
|
|
||||||
parser = argparse.ArgumentParser(description='Create an EBS-backed NixOS AMI')
|
parser = argparse.ArgumentParser(description='Create an EBS-backed NixOS AMI')
|
||||||
parser.add_argument('--region', dest='region', required=True, help='EC2 region')
|
parser.add_argument('--region', dest='region', required=True, help='EC2 region')
|
||||||
parser.add_argument('--keep', dest='keep', action='store_true', help='keep Charon machine after use')
|
parser.add_argument('--key', dest='key_name', default="eelco", help='EC2 keypair')
|
||||||
|
parser.add_argument('--keep', dest='keep', action='store_true', help='Keep Charon machine after use')
|
||||||
|
parser.add_argument('--hvm', dest='hvm', action='store_true', help='Create HVM image')
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
instance_type = "cc1.4xlarge" if args.hvm else "m1.small"
|
||||||
|
key_name = args.key_name
|
||||||
|
ebs_size = 8 if args.hvm else 20
|
||||||
|
|
||||||
# Start a NixOS machine in the given region.
|
# Start a NixOS machine in the given region.
|
||||||
f = open("ebs-creator-zone.nix", "w")
|
f = open("ebs-creator-config.nix", "w")
|
||||||
f.write('{{ machine.deployment.ec2.region = "{0}"; }}'.format(args.region))
|
f.write('''{{
|
||||||
|
machine =
|
||||||
|
{{ pkgs, ... }}:
|
||||||
|
{{
|
||||||
|
deployment.ec2.region = "{0}";
|
||||||
|
deployment.ec2.keyPair = pkgs.lib.mkOverride 10 "{1}";
|
||||||
|
deployment.ec2.blockDeviceMapping."/dev/xvdg".size = pkgs.lib.mkOverride 10 {2};
|
||||||
|
}};
|
||||||
|
}}
|
||||||
|
'''.format(args.region, key_name, ebs_size))
|
||||||
f.close()
|
f.close()
|
||||||
|
|
||||||
depl = deployment.Deployment("./ebs-creator.json", create=True, nix_exprs=["./ebs-creator.nix", "./ebs-creator-zone.nix"])
|
depl = deployment.Deployment("./ebs-creator.json", create=True, nix_exprs=["./ebs-creator.nix", "./ebs-creator-config.nix"])
|
||||||
if not args.keep: depl.destroy_vms()
|
if not args.keep: depl.destroy_vms()
|
||||||
depl.deploy()
|
depl.deploy()
|
||||||
|
|
||||||
m = depl.machines['machine']
|
m = depl.machines['machine']
|
||||||
|
|
||||||
# Do the installation.
|
# Do the installation.
|
||||||
|
device="/dev/xvdg"
|
||||||
|
if args.hvm:
|
||||||
|
m.run_command('parted -s /dev/xvdg -- mklabel msdos')
|
||||||
|
m.run_command('parted -s /dev/xvdg -- mkpart primary ext2 1M -1s')
|
||||||
|
device="/dev/xvdg1"
|
||||||
|
|
||||||
m.run_command("if mountpoint -q /mnt; then umount /mnt; fi")
|
m.run_command("if mountpoint -q /mnt; then umount /mnt; fi")
|
||||||
m.run_command("mkfs.ext4 -L nixos /dev/xvdg")
|
m.run_command("mkfs.ext4 -L nixos {0}".format(device))
|
||||||
m.run_command("mkdir -p /mnt")
|
m.run_command("mkdir -p /mnt")
|
||||||
m.run_command("mount /dev/xvdg /mnt")
|
m.run_command("mount {0} /mnt".format(device))
|
||||||
m.run_command("touch /mnt/.ebs")
|
m.run_command("touch /mnt/.ebs")
|
||||||
m.run_command("mkdir -p /mnt/etc/nixos")
|
m.run_command("mkdir -p /mnt/etc/nixos")
|
||||||
m.run_command("nixos-checkout") # FIXME
|
m.run_command("nixos-checkout") # FIXME
|
||||||
@ -37,40 +58,78 @@ version = m.run_command("nixos-option system.nixosVersion", capture_stdout=True)
|
|||||||
print >> sys.stderr, "NixOS version is {0}".format(version)
|
print >> sys.stderr, "NixOS version is {0}".format(version)
|
||||||
m.run_command("nixos-install")
|
m.run_command("nixos-install")
|
||||||
m.run_command("rm -rf /mnt/etc/nixos/nixos/.svn /mnt/etc/nixos/nixpkgs/.svn")
|
m.run_command("rm -rf /mnt/etc/nixos/nixos/.svn /mnt/etc/nixos/nixpkgs/.svn")
|
||||||
|
if args.hvm:
|
||||||
|
m.run_command('cp /nix/store/*-grub-0.97*/lib/grub/i386-pc/* /mnt/boot/grub')
|
||||||
|
m.run_command('sed -i "s|hd0|hd0,0|" /mnt/boot/grub/menu.lst')
|
||||||
|
m.run_command('echo "(hd1) /dev/xvdg" > device.map')
|
||||||
|
m.run_command('echo -e "root (hd1,0)\nsetup (hd1)" | grub --device-map=device.map --batch')
|
||||||
|
|
||||||
|
|
||||||
m.run_command("umount /mnt")
|
m.run_command("umount /mnt")
|
||||||
|
|
||||||
ami_name = "nixos-{0}-x86_64-ebs".format(version)
|
|
||||||
description = "NixOS {0} (x86_64; EBS root)".format(version)
|
|
||||||
|
|
||||||
# Create a snapshot.
|
if args.hvm:
|
||||||
m.connect()
|
ami_name = "nixos-{0}-x86_64-ebs-hvm".format(version)
|
||||||
volume = m._conn.get_all_volumes([], filters={'attachment.instance-id': m._instance_id, 'attachment.device': "/dev/sdg"})[0]
|
description = "NixOS {0} (x86_64; EBS root; hvm)".format(version)
|
||||||
snapshot = volume.create_snapshot(description=description)
|
else:
|
||||||
print >> sys.stderr, "created snapshot {0}".format(snapshot.id)
|
ami_name = "nixos-{0}-x86_64-ebs".format(version)
|
||||||
|
description = "NixOS {0} (x86_64; EBS root)".format(version)
|
||||||
|
|
||||||
|
|
||||||
# Wait for the snapshot to finish.
|
# Wait for the snapshot to finish.
|
||||||
def check():
|
def check():
|
||||||
status = snapshot.update()
|
status = snapshot.update()
|
||||||
print >> sys.stderr, "snapshot status is {0}".format(status)
|
print >> sys.stderr, "snapshot status is {0}".format(status)
|
||||||
return status == '100%'
|
return status == '100%'
|
||||||
charon.util.check_wait(check, max_tries=120)
|
|
||||||
|
|
||||||
m._conn.create_tags([snapshot.id], {'Name': ami_name})
|
m.connect()
|
||||||
|
volume = m._conn.get_all_volumes([], filters={'attachment.instance-id': m._instance_id, 'attachment.device': "/dev/sdg"})[0]
|
||||||
|
if args.hvm:
|
||||||
|
instance = m._conn.run_instances( image_id="ami-6a9e4503"
|
||||||
|
, instance_type=instance_type
|
||||||
|
, key_name=key_name
|
||||||
|
, placement=m._zone
|
||||||
|
, security_groups=["eelco-test"]).instances[0]
|
||||||
|
charon.util.check_wait(lambda: instance.update() == 'running', max_tries=120)
|
||||||
|
instance.stop()
|
||||||
|
charon.util.check_wait(lambda: instance.update() == 'stopped', max_tries=120)
|
||||||
|
old_root_volume = m._conn.get_all_volumes([], filters={'attachment.instance-id': instance.id, 'attachment.device': "/dev/sda1"})[0]
|
||||||
|
old_root_volume.detach()
|
||||||
|
volume.detach()
|
||||||
|
charon.util.check_wait(lambda: volume.update() == 'available', max_tries=120)
|
||||||
|
charon.util.check_wait(lambda: old_root_volume.update() == 'available', max_tries=120)
|
||||||
|
volume.attach(instance.id, '/dev/sda1')
|
||||||
|
charon.util.check_wait(lambda: volume.update() == 'in-use', max_tries=120)
|
||||||
|
|
||||||
if not args.keep: depl.destroy_vms()
|
ami_id = m._conn.create_image(instance.id, ami_name, description)
|
||||||
|
time.sleep(5)
|
||||||
|
image = m._conn.get_all_images([ami_id])[0]
|
||||||
|
charon.util.check_wait(lambda: image.update() == 'available', max_tries=120)
|
||||||
|
instance.terminate()
|
||||||
|
|
||||||
# Register the image.
|
else:
|
||||||
aki = m._conn.get_all_images(filters={'manifest-location': '*pv-grub-hd0_1.03-x86_64*'})[0]
|
# Create a snapshot.
|
||||||
print >> sys.stderr, "using kernel image {0} - {1}".format(aki.id, aki.location)
|
snapshot = volume.create_snapshot(description=description)
|
||||||
|
print >> sys.stderr, "created snapshot {0}".format(snapshot.id)
|
||||||
|
|
||||||
block_map = BlockDeviceMapping()
|
charon.util.check_wait(check, max_tries=120)
|
||||||
block_map['/dev/sda'] = BlockDeviceType(snapshot_id=snapshot.id, delete_on_termination=True)
|
|
||||||
block_map['/dev/sdb'] = BlockDeviceType(ephemeral_name="ephemeral0")
|
|
||||||
block_map['/dev/sdc'] = BlockDeviceType(ephemeral_name="ephemeral1")
|
|
||||||
block_map['/dev/sdd'] = BlockDeviceType(ephemeral_name="ephemeral2")
|
|
||||||
block_map['/dev/sde'] = BlockDeviceType(ephemeral_name="ephemeral3")
|
|
||||||
|
|
||||||
ami_id = m._conn.register_image(
|
m._conn.create_tags([snapshot.id], {'Name': ami_name})
|
||||||
|
|
||||||
|
if not args.keep: depl.destroy_vms()
|
||||||
|
|
||||||
|
# Register the image.
|
||||||
|
aki = m._conn.get_all_images(filters={'manifest-location': '*pv-grub-hd0_1.03-x86_64*'})[0]
|
||||||
|
print >> sys.stderr, "using kernel image {0} - {1}".format(aki.id, aki.location)
|
||||||
|
|
||||||
|
block_map = BlockDeviceMapping()
|
||||||
|
block_map['/dev/sda'] = BlockDeviceType(snapshot_id=snapshot.id, delete_on_termination=True)
|
||||||
|
block_map['/dev/sdb'] = BlockDeviceType(ephemeral_name="ephemeral0")
|
||||||
|
block_map['/dev/sdc'] = BlockDeviceType(ephemeral_name="ephemeral1")
|
||||||
|
block_map['/dev/sdd'] = BlockDeviceType(ephemeral_name="ephemeral2")
|
||||||
|
block_map['/dev/sde'] = BlockDeviceType(ephemeral_name="ephemeral3")
|
||||||
|
|
||||||
|
ami_id = m._conn.register_image(
|
||||||
name=ami_name,
|
name=ami_name,
|
||||||
description=description,
|
description=description,
|
||||||
architecture="x86_64",
|
architecture="x86_64",
|
||||||
@ -94,12 +153,13 @@ f.write(
|
|||||||
{{ network.description = "NixOS EBS test";
|
{{ network.description = "NixOS EBS test";
|
||||||
machine.deployment.targetEnv = "ec2";
|
machine.deployment.targetEnv = "ec2";
|
||||||
machine.deployment.ec2.region = "{0}";
|
machine.deployment.ec2.region = "{0}";
|
||||||
machine.deployment.ec2.instanceType = "m1.small";
|
machine.deployment.ec2.instanceType = "{2}";
|
||||||
machine.deployment.ec2.keyPair = "eelco";
|
machine.deployment.ec2.keyPair = "{3}";
|
||||||
machine.deployment.ec2.securityGroups = [ "eelco-test" ];
|
machine.deployment.ec2.securityGroups = [ "eelco-test" ];
|
||||||
machine.deployment.ec2.ami = "{1}";
|
machine.deployment.ec2.ami = "{1}";
|
||||||
|
machine.fileSystems = [];
|
||||||
}}
|
}}
|
||||||
'''.format(args.region, ami_id))
|
'''.format(args.region, ami_id, instance_type, key_name))
|
||||||
f.close()
|
f.close()
|
||||||
|
|
||||||
test_depl = deployment.Deployment("./ebs-test.json", create=True, nix_exprs=["./ebs-test.nix"])
|
test_depl = deployment.Deployment("./ebs-test.json", create=True, nix_exprs=["./ebs-test.nix"])
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
deployment.ec2.instanceType = "m1.small";
|
deployment.ec2.instanceType = "m1.small";
|
||||||
deployment.ec2.keyPair = "eelco";
|
deployment.ec2.keyPair = "eelco";
|
||||||
deployment.ec2.securityGroups = [ "eelco-test" ];
|
deployment.ec2.securityGroups = [ "eelco-test" ];
|
||||||
deployment.ec2.blockDeviceMapping."/dev/xvdg".size = 20;
|
environment.systemPackages = [ pkgs.parted ];
|
||||||
|
fileSystems = [];
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user