2019-10-25 06:42:40 +00:00
|
|
|
import json
|
2019-11-18 17:39:57 +00:00
|
|
|
import os
|
2019-10-25 06:42:40 +00:00
|
|
|
import subprocess
|
2019-12-21 09:36:55 +00:00
|
|
|
import sys
|
2019-10-25 06:42:40 +00:00
|
|
|
|
2019-12-21 09:36:55 +00:00
|
|
|
from os.path import isdir
|
2019-11-25 06:52:36 +00:00
|
|
|
from os.path import join as join_path
|
2019-12-08 13:11:19 +00:00
|
|
|
from ucloud.config import etcd_client, config, image_storage_handler
|
2019-12-03 10:40:41 +00:00
|
|
|
from ucloud.imagescanner import logger
|
2019-10-25 06:42:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
def qemu_img_type(path):
|
|
|
|
qemu_img_info_command = ["qemu-img", "info", "--output", "json", path]
|
|
|
|
try:
|
|
|
|
qemu_img_info = subprocess.check_output(qemu_img_info_command)
|
|
|
|
except Exception as e:
|
2019-11-18 17:39:57 +00:00
|
|
|
logger.exception(e)
|
2019-10-25 06:42:40 +00:00
|
|
|
return None
|
|
|
|
else:
|
|
|
|
qemu_img_info = json.loads(qemu_img_info.decode("utf-8"))
|
|
|
|
return qemu_img_info["format"]
|
|
|
|
|
2019-12-07 13:25:21 +00:00
|
|
|
def check():
|
|
|
|
""" check whether settings are sane, refuse to start if they aren't """
|
2019-12-21 09:36:55 +00:00
|
|
|
if config['storage']['storage_backend'] == 'filesystem' and not isdir(config['storage']['image_dir']):
|
|
|
|
sys.exit("You have set STORAGE_BACKEND to filesystem, but "
|
|
|
|
"{} does not exist. Refusing to start".format(config['storage']['image_dir'])
|
|
|
|
)
|
2019-12-07 13:25:21 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
subprocess.check_output(['which', 'qemu-img'])
|
|
|
|
except Exception:
|
|
|
|
print("qemu-img missing")
|
|
|
|
sys.exit(1)
|
|
|
|
|
2019-10-25 06:42:40 +00:00
|
|
|
|
2019-11-02 15:42:24 +00:00
|
|
|
def main():
|
|
|
|
# We want to get images entries that requests images to be created
|
2019-12-14 15:23:31 +00:00
|
|
|
images = etcd_client.get_prefix(config['etcd']['image_prefix'], value_in_json=True)
|
2019-11-02 15:42:24 +00:00
|
|
|
images_to_be_created = list(filter(lambda im: im.value['status'] == 'TO_BE_CREATED', images))
|
|
|
|
|
|
|
|
for image in images_to_be_created:
|
|
|
|
try:
|
|
|
|
image_uuid = image.key.split('/')[-1]
|
|
|
|
image_owner = image.value['owner']
|
|
|
|
image_filename = image.value['filename']
|
|
|
|
image_store_name = image.value['store_name']
|
2019-12-14 15:23:31 +00:00
|
|
|
image_full_path = join_path(config['storage']['file_dir'], image_owner, image_filename)
|
2019-11-02 15:42:24 +00:00
|
|
|
|
2019-12-14 15:23:31 +00:00
|
|
|
image_stores = etcd_client.get_prefix(config['etcd']['image_store_prefix'],
|
|
|
|
value_in_json=True)
|
2019-11-02 15:42:24 +00:00
|
|
|
user_image_store = next(filter(
|
|
|
|
lambda s, store_name=image_store_name: s.value["name"] == store_name,
|
|
|
|
image_stores
|
|
|
|
))
|
|
|
|
|
|
|
|
image_store_pool = user_image_store.value['attributes']['pool']
|
|
|
|
|
|
|
|
except Exception as e:
|
2019-11-18 17:39:57 +00:00
|
|
|
logger.exception(e)
|
2019-11-02 15:42:24 +00:00
|
|
|
else:
|
|
|
|
# At least our basic data is available
|
|
|
|
qemu_img_convert_command = ["qemu-img", "convert", "-f", "qcow2",
|
|
|
|
"-O", "raw", image_full_path, "image.raw"]
|
2019-10-25 06:42:40 +00:00
|
|
|
|
2019-11-02 15:42:24 +00:00
|
|
|
if qemu_img_type(image_full_path) == "qcow2":
|
|
|
|
try:
|
|
|
|
# Convert .qcow2 to .raw
|
|
|
|
subprocess.check_output(qemu_img_convert_command)
|
|
|
|
except Exception as e:
|
2019-11-18 17:39:57 +00:00
|
|
|
logger.exception(e)
|
2019-11-02 15:42:24 +00:00
|
|
|
else:
|
2019-11-25 06:52:36 +00:00
|
|
|
# Import and Protect
|
|
|
|
r_status = image_storage_handler.import_image(src="image.raw",
|
|
|
|
dest=image_uuid,
|
|
|
|
protect=True)
|
|
|
|
if r_status:
|
|
|
|
# Everything is successfully done
|
|
|
|
image.value["status"] = "CREATED"
|
|
|
|
etcd_client.put(image.key, json.dumps(image.value))
|
|
|
|
|
2019-10-25 06:42:40 +00:00
|
|
|
else:
|
2019-11-02 15:42:24 +00:00
|
|
|
# The user provided image is either not found or of invalid format
|
|
|
|
image.value["status"] = "INVALID_IMAGE"
|
2019-11-18 17:39:57 +00:00
|
|
|
etcd_client.put(image.key, json.dumps(image.value))
|
2019-10-25 06:42:40 +00:00
|
|
|
|
2019-11-02 15:42:24 +00:00
|
|
|
try:
|
|
|
|
os.remove("image.raw")
|
|
|
|
except Exception:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2019-11-18 17:39:57 +00:00
|
|
|
main()
|