2019-10-25 06:42:40 +00:00
|
|
|
import json
|
2019-11-18 17:39:57 +00:00
|
|
|
import os
|
2019-12-23 07:58:04 +00:00
|
|
|
import subprocess as sp
|
2019-10-25 06:42:40 +00:00
|
|
|
|
2019-11-25 06:52:36 +00:00
|
|
|
from os.path import join as join_path
|
2019-12-22 07:26:48 +00:00
|
|
|
from ucloud.settings import settings
|
|
|
|
from ucloud.shared import shared
|
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):
|
2019-12-30 09:35:07 +00:00
|
|
|
qemu_img_info_command = [
|
|
|
|
"qemu-img",
|
|
|
|
"info",
|
|
|
|
"--output",
|
|
|
|
"json",
|
|
|
|
path,
|
|
|
|
]
|
2019-10-25 06:42:40 +00:00
|
|
|
try:
|
2019-12-23 07:58:04 +00:00
|
|
|
qemu_img_info = sp.check_output(qemu_img_info_command)
|
2019-10-25 06:42:40 +00:00
|
|
|
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-11-02 15:42:24 +00:00
|
|
|
def main():
|
|
|
|
# We want to get images entries that requests images to be created
|
2019-12-30 09:35:07 +00:00
|
|
|
images = shared.etcd_client.get_prefix(
|
|
|
|
settings["etcd"]["image_prefix"], value_in_json=True
|
|
|
|
)
|
|
|
|
images_to_be_created = list(
|
|
|
|
filter(lambda im: im.value["status"] == "TO_BE_CREATED", images)
|
|
|
|
)
|
2019-11-02 15:42:24 +00:00
|
|
|
|
|
|
|
for image in images_to_be_created:
|
|
|
|
try:
|
2019-12-30 09:35:07 +00:00
|
|
|
image_uuid = image.key.split("/")[-1]
|
|
|
|
image_owner = image.value["owner"]
|
|
|
|
image_filename = image.value["filename"]
|
|
|
|
image_store_name = image.value["store_name"]
|
|
|
|
image_full_path = join_path(
|
|
|
|
settings["storage"]["file_dir"],
|
|
|
|
image_owner,
|
|
|
|
image_filename,
|
|
|
|
)
|
2019-11-02 15:42:24 +00:00
|
|
|
|
2019-12-30 09:35:07 +00:00
|
|
|
image_stores = shared.etcd_client.get_prefix(
|
|
|
|
settings["etcd"]["image_store_prefix"],
|
|
|
|
value_in_json=True,
|
|
|
|
)
|
|
|
|
user_image_store = next(
|
|
|
|
filter(
|
|
|
|
lambda s, store_name=image_store_name: s.value[
|
|
|
|
"name"
|
|
|
|
]
|
|
|
|
== store_name,
|
|
|
|
image_stores,
|
|
|
|
)
|
|
|
|
)
|
2019-11-02 15:42:24 +00:00
|
|
|
|
2019-12-30 09:35:07 +00:00
|
|
|
image_store_pool = user_image_store.value["attributes"][
|
|
|
|
"pool"
|
|
|
|
]
|
2019-11-02 15:42:24 +00:00
|
|
|
|
|
|
|
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
|
2019-12-30 09:35:07 +00:00
|
|
|
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
|
2019-12-23 07:58:04 +00:00
|
|
|
sp.check_output(qemu_img_convert_command,)
|
|
|
|
|
|
|
|
except sp.CalledProcessError:
|
2019-12-30 09:35:07 +00:00
|
|
|
logger.exception(
|
|
|
|
"Image convertion from .qcow2 to .raw failed."
|
|
|
|
)
|
2019-11-02 15:42:24 +00:00
|
|
|
else:
|
2019-11-25 06:52:36 +00:00
|
|
|
# Import and Protect
|
2019-12-30 09:35:07 +00:00
|
|
|
r_status = shared.storage_handler.import_image(
|
|
|
|
src="image.raw", dest=image_uuid, protect=True
|
|
|
|
)
|
2019-11-25 06:52:36 +00:00
|
|
|
if r_status:
|
|
|
|
# Everything is successfully done
|
|
|
|
image.value["status"] = "CREATED"
|
2019-12-30 09:35:07 +00:00
|
|
|
shared.etcd_client.put(
|
|
|
|
image.key, json.dumps(image.value)
|
|
|
|
)
|
2019-12-23 07:58:04 +00:00
|
|
|
finally:
|
|
|
|
try:
|
|
|
|
os.remove("image.raw")
|
|
|
|
except Exception:
|
|
|
|
pass
|
2019-11-25 06:52:36 +00:00
|
|
|
|
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-12-30 09:35:07 +00:00
|
|
|
shared.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
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2019-11-18 17:39:57 +00:00
|
|
|
main()
|