200 lines
6.0 KiB
Python
Raw Normal View History

2020-10-02 16:07:50 +09:30
#
# Copyright (C) 2020 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 compute.
2020-10-02 16:07:50 +09:30
"""
import os
import psutil
from gns3server.config import Config
from gns3server.utils.cpu_percent import CpuPercent
from gns3server.version import __version__
from gns3server.utils.path import get_default_project_directory
from gns3server.compute.port_manager import PortManager
from gns3server.compute.project_manager import ProjectManager
from gns3server.utils.interfaces import interfaces
from gns3server.compute.qemu import Qemu
from gns3server.compute.virtualbox import VirtualBox
from gns3server.compute.vmware import VMware
2020-10-31 16:02:21 +10:30
from gns3server import schemas
2020-10-02 16:07:50 +09:30
from fastapi import APIRouter, HTTPException, Body, Response, status
2020-10-02 16:07:50 +09:30
from fastapi.encoders import jsonable_encoder
from uuid import UUID
from typing import Optional, List
router = APIRouter()
2021-04-13 18:46:50 +09:30
@router.post("/projects/{project_id}/ports/udp", status_code=status.HTTP_201_CREATED)
2020-10-02 16:07:50 +09:30
def allocate_udp_port(project_id: UUID) -> dict:
"""
Allocate an UDP port on the compute.
"""
pm = ProjectManager.instance()
project = pm.get_project(str(project_id))
m = PortManager.instance()
udp_port = m.get_free_udp_port(project)
return {"udp_port": udp_port}
@router.get("/network/interfaces")
def network_interfaces() -> dict:
"""
List all the network interfaces available on the compute"
"""
network_interfaces = interfaces()
return network_interfaces
@router.get("/network/ports")
def network_ports() -> dict:
"""
List all the ports used on the compute"
"""
m = PortManager.instance()
2021-04-17 23:34:28 +09:30
return m.asdict()
2020-10-02 16:07:50 +09:30
@router.get("/version")
def compute_version() -> dict:
2020-10-02 16:07:50 +09:30
"""
Retrieve the server version number.
"""
local_server = Config.instance().settings.Server.local
2020-10-02 16:07:50 +09:30
return {"version": __version__, "local": local_server}
@router.get("/statistics")
def compute_statistics() -> dict:
2020-10-02 16:07:50 +09:30
"""
Retrieve the server version number.
"""
try:
memory_total = psutil.virtual_memory().total
memory_free = psutil.virtual_memory().available
memory_used = memory_total - memory_free # actual memory usage in a cross platform fashion
swap_total = psutil.swap_memory().total
swap_free = psutil.swap_memory().free
swap_used = psutil.swap_memory().used
cpu_percent = int(CpuPercent.get())
load_average_percent = [int(x / psutil.cpu_count() * 100) for x in psutil.getloadavg()]
memory_percent = int(psutil.virtual_memory().percent)
swap_percent = int(psutil.swap_memory().percent)
disk_usage_percent = int(psutil.disk_usage(get_default_project_directory()).percent)
except psutil.Error as e:
raise HTTPException(status_code=status.HTTP_500_INTERNAL_SERVER_ERROR)
2021-04-13 18:46:50 +09:30
# raise HTTPConflict(text="Psutil error detected: {}".format(e))
return {
"memory_total": memory_total,
"memory_free": memory_free,
"memory_used": memory_used,
"swap_total": swap_total,
"swap_free": swap_free,
"swap_used": swap_used,
"cpu_usage_percent": cpu_percent,
"memory_usage_percent": memory_percent,
"swap_usage_percent": swap_percent,
"disk_usage_percent": disk_usage_percent,
"load_average_percent": load_average_percent,
}
2020-10-02 16:07:50 +09:30
@router.get("/qemu/binaries")
async def get_qemu_binaries(
archs: Optional[List[str]] = Body(None, embed=True)
) -> List[str]:
2020-10-02 16:07:50 +09:30
return await Qemu.binary_list(archs)
@router.get("/qemu/img-binaries")
async def get_image_binaries() -> List[str]:
2020-10-02 16:07:50 +09:30
return await Qemu.img_binary_list()
@router.get("/qemu/capabilities")
async def get_qemu_capabilities() -> dict:
2020-10-02 16:07:50 +09:30
capabilities = {"kvm": []}
kvms = await Qemu.get_kvm_archs()
if kvms:
capabilities["kvm"] = kvms
return capabilities
2021-04-13 18:46:50 +09:30
@router.post(
"/qemu/img",
status_code=status.HTTP_204_NO_CONTENT,
responses={403: {"model": schemas.ErrorMessage, "description": "Forbidden to create Qemu image"}},
)
async def create_qemu_image(image_data: schemas.QemuImageCreate) -> Response:
2020-10-02 16:07:50 +09:30
"""
Create a Qemu image.
"""
if os.path.isabs(image_data.path):
if Config.instance().settings.Server.local is False:
2020-10-02 16:07:50 +09:30
raise HTTPException(status_code=status.HTTP_403_FORBIDDEN)
2021-04-13 18:46:50 +09:30
await Qemu.instance().create_disk(
image_data.qemu_img, image_data.path, jsonable_encoder(image_data, exclude_unset=True)
)
2020-10-02 16:07:50 +09:30
return Response(status_code=status.HTTP_204_NO_CONTENT)
2020-10-02 16:07:50 +09:30
2021-04-13 18:46:50 +09:30
@router.put(
"/qemu/img",
status_code=status.HTTP_204_NO_CONTENT,
responses={403: {"model": schemas.ErrorMessage, "description": "Forbidden to update Qemu image"}},
)
async def update_qemu_image(image_data: schemas.QemuImageUpdate) -> Response:
2020-10-02 16:07:50 +09:30
"""
Update a Qemu image.
"""
if os.path.isabs(image_data.path):
if Config.instance().settings.Server.local is False:
2020-10-02 16:07:50 +09:30
raise HTTPException(status_code=status.HTTP_403_FORBIDDEN)
if image_data.extend:
await Qemu.instance().resize_disk(image_data.qemu_img, image_data.path, image_data.extend)
return Response(status_code=status.HTTP_204_NO_CONTENT)
2020-10-02 16:07:50 +09:30
2021-04-13 18:46:50 +09:30
@router.get("/virtualbox/vms", response_model=List[dict])
async def get_virtualbox_vms() -> List[dict]:
2020-10-02 16:07:50 +09:30
vbox_manager = VirtualBox.instance()
return await vbox_manager.list_vms()
2021-04-13 18:46:50 +09:30
@router.get("/vmware/vms", response_model=List[dict])
async def get_vms() -> List[dict]:
2020-10-02 16:07:50 +09:30
vmware_manager = VMware.instance()
return await vmware_manager.list_vms()