2021-06-06 10:22:47 +03:00
|
|
|
#
|
|
|
|
# Copyright (C) 2021 GNS3 Technologies Inc.
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
"""
|
|
|
|
API routes for images.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import os
|
|
|
|
import logging
|
|
|
|
import urllib.parse
|
|
|
|
|
2021-08-20 09:28:41 +03:00
|
|
|
from fastapi import APIRouter, Request, Response, Depends, status
|
2021-08-30 10:23:41 +03:00
|
|
|
from sqlalchemy.orm.exc import MultipleResultsFound
|
2021-10-18 10:34:30 +03:00
|
|
|
from typing import List, Optional
|
2021-06-06 10:22:47 +03:00
|
|
|
from gns3server import schemas
|
|
|
|
|
|
|
|
from gns3server.utils.images import InvalidImageError, default_images_directory, write_image
|
|
|
|
from gns3server.db.repositories.images import ImagesRepository
|
2021-10-10 10:05:11 +03:00
|
|
|
from gns3server.db.repositories.templates import TemplatesRepository
|
|
|
|
from gns3server.db.repositories.rbac import RbacRepository
|
|
|
|
from gns3server.controller import Controller
|
2021-06-06 10:22:47 +03:00
|
|
|
from gns3server.controller.controller_error import (
|
|
|
|
ControllerError,
|
|
|
|
ControllerNotFoundError,
|
|
|
|
ControllerForbiddenError,
|
|
|
|
ControllerBadRequestError
|
|
|
|
)
|
|
|
|
|
2021-10-10 10:05:11 +03:00
|
|
|
from .dependencies.authentication import get_current_active_user
|
2021-06-06 10:22:47 +03:00
|
|
|
from .dependencies.database import get_repository
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
router = APIRouter()
|
|
|
|
|
|
|
|
|
2021-10-10 10:05:11 +03:00
|
|
|
@router.get("", response_model=List[schemas.Image])
|
2021-06-06 10:22:47 +03:00
|
|
|
async def get_images(
|
|
|
|
images_repo: ImagesRepository = Depends(get_repository(ImagesRepository)),
|
|
|
|
) -> List[schemas.Image]:
|
|
|
|
"""
|
|
|
|
Return all images.
|
|
|
|
"""
|
|
|
|
|
|
|
|
return await images_repo.get_images()
|
|
|
|
|
|
|
|
|
2021-08-30 10:23:41 +03:00
|
|
|
@router.post("/upload/{image_path:path}", response_model=schemas.Image, status_code=status.HTTP_201_CREATED)
|
2021-06-06 10:22:47 +03:00
|
|
|
async def upload_image(
|
2021-08-30 10:23:41 +03:00
|
|
|
image_path: str,
|
2021-06-06 10:22:47 +03:00
|
|
|
request: Request,
|
2021-08-11 10:28:23 +03:00
|
|
|
image_type: schemas.ImageType = schemas.ImageType.qemu,
|
2021-06-06 10:22:47 +03:00
|
|
|
images_repo: ImagesRepository = Depends(get_repository(ImagesRepository)),
|
2021-10-10 10:05:11 +03:00
|
|
|
templates_repo: TemplatesRepository = Depends(get_repository(TemplatesRepository)),
|
|
|
|
current_user: schemas.User = Depends(get_current_active_user),
|
2021-10-18 10:34:30 +03:00
|
|
|
rbac_repo: RbacRepository = Depends(get_repository(RbacRepository)),
|
|
|
|
install_appliances: Optional[bool] = True
|
2021-06-06 10:22:47 +03:00
|
|
|
) -> schemas.Image:
|
|
|
|
"""
|
|
|
|
Upload an image.
|
2021-10-10 10:05:11 +03:00
|
|
|
|
|
|
|
Example: curl -X POST http://host:port/v3/images/upload/my_image_name.qcow2?image_type=qemu \
|
|
|
|
-H 'Authorization: Bearer <token>' --data-binary @"/path/to/image.qcow2"
|
2021-06-06 10:22:47 +03:00
|
|
|
"""
|
|
|
|
|
2021-08-30 10:23:41 +03:00
|
|
|
image_path = urllib.parse.unquote(image_path)
|
|
|
|
image_dir, image_name = os.path.split(image_path)
|
2021-06-06 10:22:47 +03:00
|
|
|
directory = default_images_directory(image_type)
|
2021-08-30 10:23:41 +03:00
|
|
|
full_path = os.path.abspath(os.path.join(directory, image_dir, image_name))
|
|
|
|
if os.path.commonprefix([directory, full_path]) != directory:
|
2021-10-10 10:05:11 +03:00
|
|
|
raise ControllerForbiddenError(f"Cannot write image, '{image_path}' is forbidden")
|
2021-06-06 10:22:47 +03:00
|
|
|
|
2021-08-30 10:23:41 +03:00
|
|
|
if await images_repo.get_image(image_path):
|
|
|
|
raise ControllerBadRequestError(f"Image '{image_path}' already exists")
|
2021-06-06 10:22:47 +03:00
|
|
|
|
|
|
|
try:
|
2021-08-30 10:23:41 +03:00
|
|
|
image = await write_image(image_name, image_type, full_path, request.stream(), images_repo)
|
2021-06-06 10:22:47 +03:00
|
|
|
except (OSError, InvalidImageError) as e:
|
2021-08-30 10:23:41 +03:00
|
|
|
raise ControllerError(f"Could not save {image_type} image '{image_path}': {e}")
|
2021-06-06 10:22:47 +03:00
|
|
|
|
2021-10-18 10:34:30 +03:00
|
|
|
if install_appliances:
|
|
|
|
# attempt to automatically create templates based on image checksum
|
|
|
|
await Controller.instance().appliance_manager.install_appliances_from_image(
|
|
|
|
image_path,
|
2021-10-10 10:05:11 +03:00
|
|
|
image.checksum,
|
|
|
|
images_repo,
|
2021-10-18 10:34:30 +03:00
|
|
|
templates_repo,
|
|
|
|
rbac_repo,
|
|
|
|
current_user,
|
2021-10-10 10:05:11 +03:00
|
|
|
directory
|
|
|
|
)
|
|
|
|
|
2021-06-06 10:22:47 +03:00
|
|
|
return image
|
|
|
|
|
|
|
|
|
2021-08-30 10:23:41 +03:00
|
|
|
@router.get("/{image_path:path}", response_model=schemas.Image)
|
2021-06-06 10:22:47 +03:00
|
|
|
async def get_image(
|
2021-08-30 10:23:41 +03:00
|
|
|
image_path: str,
|
2021-06-06 10:22:47 +03:00
|
|
|
images_repo: ImagesRepository = Depends(get_repository(ImagesRepository)),
|
|
|
|
) -> schemas.Image:
|
|
|
|
"""
|
|
|
|
Return an image.
|
|
|
|
"""
|
|
|
|
|
2021-08-30 10:23:41 +03:00
|
|
|
image_path = urllib.parse.unquote(image_path)
|
|
|
|
image = await images_repo.get_image(image_path)
|
2021-06-06 10:22:47 +03:00
|
|
|
if not image:
|
2021-08-30 10:23:41 +03:00
|
|
|
raise ControllerNotFoundError(f"Image '{image_path}' not found")
|
2021-06-06 10:22:47 +03:00
|
|
|
return image
|
|
|
|
|
|
|
|
|
2021-08-30 10:23:41 +03:00
|
|
|
@router.delete("/{image_path:path}", status_code=status.HTTP_204_NO_CONTENT)
|
2021-06-06 10:22:47 +03:00
|
|
|
async def delete_image(
|
2021-08-30 10:23:41 +03:00
|
|
|
image_path: str,
|
2021-06-06 10:22:47 +03:00
|
|
|
images_repo: ImagesRepository = Depends(get_repository(ImagesRepository)),
|
|
|
|
) -> None:
|
|
|
|
"""
|
|
|
|
Delete an image.
|
|
|
|
"""
|
|
|
|
|
2021-08-30 10:23:41 +03:00
|
|
|
image_path = urllib.parse.unquote(image_path)
|
|
|
|
|
|
|
|
try:
|
|
|
|
image = await images_repo.get_image(image_path)
|
|
|
|
except MultipleResultsFound:
|
|
|
|
raise ControllerBadRequestError(f"Image '{image_path}' matches multiple images. "
|
|
|
|
f"Please include the relative path of the image")
|
|
|
|
|
2021-06-06 10:22:47 +03:00
|
|
|
if not image:
|
2021-08-30 10:23:41 +03:00
|
|
|
raise ControllerNotFoundError(f"Image '{image_path}' not found")
|
2021-06-06 10:22:47 +03:00
|
|
|
|
2021-10-18 10:34:30 +03:00
|
|
|
templates = await images_repo.get_image_templates(image.image_id)
|
|
|
|
if templates:
|
|
|
|
template_names = ", ".join([template.name for template in templates])
|
|
|
|
raise ControllerError(f"Image '{image_path}' is used by one or more templates: {template_names}")
|
2021-06-06 10:22:47 +03:00
|
|
|
|
|
|
|
try:
|
|
|
|
os.remove(image.path)
|
|
|
|
except OSError:
|
|
|
|
log.warning(f"Could not delete image file {image.path}")
|
|
|
|
|
2021-08-30 10:23:41 +03:00
|
|
|
success = await images_repo.delete_image(image_path)
|
2021-06-06 10:22:47 +03:00
|
|
|
if not success:
|
2021-08-30 10:23:41 +03:00
|
|
|
raise ControllerError(f"Image '{image_path}' could not be deleted")
|
2021-08-20 09:28:41 +03:00
|
|
|
|
|
|
|
|
|
|
|
@router.post("/prune", status_code=status.HTTP_204_NO_CONTENT)
|
|
|
|
async def prune_images(
|
|
|
|
images_repo: ImagesRepository = Depends(get_repository(ImagesRepository)),
|
|
|
|
) -> Response:
|
|
|
|
"""
|
|
|
|
Prune images not attached to any template.
|
|
|
|
"""
|
|
|
|
|
|
|
|
await images_repo.prune_images()
|
|
|
|
return Response(status_code=status.HTTP_204_NO_CONTENT)
|