Some times you might need to use a preconfigured image in cloud environment as well as on-premise servers with virtualization. Amazon Web Services support importing VM to Amazon Machine Images and Exporting Imported instances. Please note AWS only officially support for exporting previously imported instances. That means you will not be able to export a regular ec2 instance to Virtual Machine Image.

In this guide we will create a Virtual Machine image in Ova format in our local server and import it to AWS. There are few things that I have tried and failed.

I tried to use the official Virtual Image provided by Canonical and importing it in to AWS. But did not work. Then I tried to download it (the VM image) to my local machine and to create a VM using virtual box. It did work either

What you need

  1. aws cli configured in your local computer
  2. VirtualBox

Create a virtual machine in your local computer

This is a pretty simple step. You just have to download iso image from ubuntu official repo and create a virtual machine with VirtualBox. Then in VirtualBox menu go to File->Export Appliance and Export the VM to ovf format. I have exported my vm as ubuntu_vmi.ova

Upload Ova Image to S3

Once the export is completed. Create a s3 bucket in AWS S3. I have created a bucket named ubuntuvirtualmachineimages. Then copy local ubuntu_vmi.ova to that bucket using aws cli.

aws s3 cp <PATH_TO_LOAL_VIRTUAL_MACHINE_IMAGE> s3://<YOUR_S3_BUCKET_NAME>

This would take some time to upload.

Creating Amazon Machine Image from Ova Image

For this step I have followed this aws guide.

  1. Create trust-policy.json
{
  "Version": "2012-10-17",
  "Statement": [
    {
      "Effect": "Allow",
      "Principal": { "Service": "vmie.amazonaws.com" },
      "Action": "sts:AssumeRole",
      "Condition": {
        "StringEquals": {
          "sts:Externalid": "vmimport"
        }
      }
    }
  ]
}

You can do the same visually with IAM AWS Console, in Roles tab.

  1. Create role-policy.json. You have to replace <YOUR_S3_BUCKET_NAME> with your bucket name.
{
  "Version": "2012-10-17",
  "Statement": [
    {
      "Effect": "Allow",
      "Action": ["s3:GetBucketLocation", "s3:GetObject", "s3:ListBucket"],
      "Resource": [
        "arn:aws:s3:::<YOUR_S3_BUCKET_NAME>",
        "arn:aws:s3:::<YOUR_S3_BUCKET_NAME>/*"
      ]
    },
    {
      "Effect": "Allow",
      "Action": [
        "ec2:ModifySnapshotAttribute",
        "ec2:CopySnapshot",
        "ec2:RegisterImage",
        "ec2:Describe*"
      ],
      "Resource": "*"
    }
  ]
}
  1. Create a role using trust-policy.json
aws iam create-role --role-name vmimport --assume-role-policy-document file://trust-policy.json
  1. Attache policy to role
aws iam put-role-policy --role-name vmimport --policy-name vmimport --policy-document file://role-policy.json
  1. Create containers.json
[
  {
    "Description": "First disk",
    "Format": "ovf",
    "UserBucket": {
      "S3Bucket": "<YOUR_S3_BUCKET_NAME>",
      "S3Key": "<YOUR_IMAGE_NAME>"
    }
  }
]
  1. Import the Image
aws ec2 import-image --description "MyVM" --license-type BYOL --disk-containers file://containers.json
  1. Check Status of the task
aws ec2 describe-conversion-tasks --region

Updated:

Leave a Comment