2016-03-03 16:02:27 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
|
|
|
# Copyright (C) 2016 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/>.
|
|
|
|
|
2016-08-25 19:14:29 +02:00
|
|
|
import ipaddress
|
2016-03-10 10:32:07 +01:00
|
|
|
import aiohttp
|
|
|
|
import asyncio
|
2018-10-16 15:56:06 +07:00
|
|
|
import async_timeout
|
2016-08-22 18:49:25 +02:00
|
|
|
import socket
|
2016-03-10 10:32:07 +01:00
|
|
|
import json
|
2016-05-25 14:10:03 +02:00
|
|
|
import uuid
|
2016-08-30 09:58:37 +02:00
|
|
|
import sys
|
2016-06-07 19:38:01 +02:00
|
|
|
import io
|
2017-05-05 20:09:51 +02:00
|
|
|
from operator import itemgetter
|
2020-10-22 20:37:34 +10:30
|
|
|
from aiohttp import web
|
2016-03-03 16:02:27 +01:00
|
|
|
|
2016-05-13 18:48:10 -06:00
|
|
|
from ..utils import parse_version
|
2018-10-15 17:05:49 +07:00
|
|
|
from ..utils.asyncio import locking
|
2020-10-02 16:07:50 +09:30
|
|
|
from ..controller.controller_error import (
|
|
|
|
ControllerError,
|
|
|
|
ControllerNotFoundError,
|
|
|
|
ControllerForbiddenError,
|
|
|
|
ControllerTimeoutError,
|
2021-04-13 18:46:50 +09:30
|
|
|
ControllerUnauthorizedError,
|
|
|
|
)
|
2018-08-22 16:54:43 +07:00
|
|
|
from ..version import __version__, __version_info__
|
2016-03-04 16:58:53 +01:00
|
|
|
|
2016-03-04 16:11:31 +01:00
|
|
|
|
2016-03-04 16:55:59 +01:00
|
|
|
import logging
|
2021-04-13 18:46:50 +09:30
|
|
|
|
2016-03-04 16:55:59 +01:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2016-03-04 16:11:31 +01:00
|
|
|
|
2016-04-15 17:57:06 +02:00
|
|
|
class ComputeError(ControllerError):
|
2016-03-04 16:11:31 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
2020-10-02 16:07:50 +09:30
|
|
|
# FIXME: broken
|
|
|
|
class ComputeConflict(ComputeError):
|
2016-06-07 19:38:01 +02:00
|
|
|
"""
|
|
|
|
Raise when the compute send a 409 that we can handle
|
|
|
|
|
|
|
|
:param response: The response of the compute
|
|
|
|
"""
|
2016-06-08 14:14:03 +02:00
|
|
|
|
2016-06-07 19:38:01 +02:00
|
|
|
def __init__(self, response):
|
2020-10-02 16:07:50 +09:30
|
|
|
super().__init__(response["message"])
|
2016-06-07 19:38:01 +02:00
|
|
|
self.response = response
|
|
|
|
|
|
|
|
|
2016-04-15 17:57:06 +02:00
|
|
|
class Compute:
|
2016-03-03 16:02:27 +01:00
|
|
|
"""
|
2016-04-15 17:57:06 +02:00
|
|
|
A GNS3 compute.
|
2016-03-03 16:02:27 +01:00
|
|
|
"""
|
|
|
|
|
2021-04-13 18:46:50 +09:30
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
compute_id,
|
|
|
|
controller=None,
|
|
|
|
protocol="http",
|
|
|
|
host="localhost",
|
|
|
|
port=3080,
|
|
|
|
user=None,
|
|
|
|
password=None,
|
|
|
|
name=None,
|
|
|
|
console_host=None,
|
|
|
|
ssl_context=None,
|
|
|
|
):
|
2016-07-22 11:43:14 +02:00
|
|
|
self._http_session = None
|
2016-03-18 16:55:54 +01:00
|
|
|
assert controller is not None
|
2016-04-15 17:57:06 +02:00
|
|
|
log.info("Create compute %s", compute_id)
|
2016-05-25 14:10:03 +02:00
|
|
|
|
2021-04-05 14:21:41 +09:30
|
|
|
# if compute_id is None:
|
|
|
|
# self._id = str(uuid.uuid4())
|
|
|
|
# else:
|
|
|
|
self._id = compute_id
|
2016-05-25 14:10:03 +02:00
|
|
|
|
2016-05-25 11:27:41 +02:00
|
|
|
self.protocol = protocol
|
2016-10-26 18:32:01 +02:00
|
|
|
self._console_host = console_host
|
2016-05-25 11:27:41 +02:00
|
|
|
self.host = host
|
|
|
|
self.port = port
|
2016-03-16 15:55:07 +01:00
|
|
|
self._user = None
|
|
|
|
self._password = None
|
2016-03-03 16:02:27 +01:00
|
|
|
self._connected = False
|
2019-04-14 17:42:20 +07:00
|
|
|
self._notifications = None
|
2016-08-29 17:36:24 +02:00
|
|
|
self._closed = False # Close mean we are destroying the compute node
|
2016-03-18 16:55:54 +01:00
|
|
|
self._controller = controller
|
2016-05-11 15:19:00 -06:00
|
|
|
self._set_auth(user, password)
|
2020-10-02 16:07:50 +09:30
|
|
|
self._cpu_usage_percent = 0
|
|
|
|
self._memory_usage_percent = 0
|
|
|
|
self._disk_usage_percent = 0
|
2018-08-22 16:54:43 +07:00
|
|
|
self._last_error = None
|
2020-10-27 23:25:19 +10:30
|
|
|
self._ssl_context = ssl_context
|
2021-04-13 18:46:50 +09:30
|
|
|
self._capabilities = {"version": "", "platform": "", "cpus": 0, "memory": 0, "disk_size": 0, "node_types": []}
|
2016-05-23 11:20:52 +02:00
|
|
|
self.name = name
|
2016-08-25 19:14:29 +02:00
|
|
|
# Cache of interfaces on remote host
|
|
|
|
self._interfaces_cache = None
|
2016-12-15 21:57:59 +01:00
|
|
|
self._connection_failure = 0
|
|
|
|
|
2016-07-20 12:43:23 +02:00
|
|
|
def _session(self):
|
|
|
|
if self._http_session is None or self._http_session.closed is True:
|
2020-11-05 17:15:25 +10:30
|
|
|
connector = aiohttp.TCPConnector(force_close=True, ssl_context=self._ssl_context)
|
2020-10-23 19:42:21 +10:30
|
|
|
self._http_session = aiohttp.ClientSession(connector=connector)
|
2016-07-20 12:43:23 +02:00
|
|
|
return self._http_session
|
|
|
|
|
2016-05-11 15:19:00 -06:00
|
|
|
def _set_auth(self, user, password):
|
2016-03-16 15:55:07 +01:00
|
|
|
"""
|
|
|
|
Set authentication parameters
|
|
|
|
"""
|
2016-05-26 10:11:11 +02:00
|
|
|
if user is None or len(user.strip()) == 0:
|
|
|
|
self._user = None
|
|
|
|
self._password = None
|
2016-03-16 15:55:07 +01:00
|
|
|
self._auth = None
|
2016-05-26 10:11:11 +02:00
|
|
|
else:
|
|
|
|
self._user = user.strip()
|
2016-05-26 13:32:52 +02:00
|
|
|
if password:
|
2021-04-12 17:02:23 +09:30
|
|
|
self._password = password
|
2017-07-20 15:42:07 +02:00
|
|
|
try:
|
2021-04-12 17:02:23 +09:30
|
|
|
self._auth = aiohttp.BasicAuth(self._user, self._password.get_secret_value(), "utf-8")
|
2017-07-20 15:42:07 +02:00
|
|
|
except ValueError as e:
|
|
|
|
log.error(str(e))
|
2016-05-26 13:32:52 +02:00
|
|
|
else:
|
|
|
|
self._password = None
|
|
|
|
self._auth = aiohttp.BasicAuth(self._user, "")
|
2016-03-16 15:55:07 +01:00
|
|
|
|
2018-08-28 15:42:06 +07:00
|
|
|
def set_last_error(self, msg):
|
|
|
|
"""
|
|
|
|
Set the last error message for this compute.
|
|
|
|
|
|
|
|
:param msg: message
|
|
|
|
"""
|
|
|
|
self._last_error = msg
|
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def interfaces(self):
|
2016-08-25 19:14:29 +02:00
|
|
|
"""
|
|
|
|
Get the list of network on compute
|
|
|
|
"""
|
|
|
|
if not self._interfaces_cache:
|
2018-10-15 17:05:49 +07:00
|
|
|
response = await self.get("/network/interfaces")
|
2016-08-25 19:14:29 +02:00
|
|
|
self._interfaces_cache = response.json
|
|
|
|
return self._interfaces_cache
|
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def update(self, **kwargs):
|
2016-05-25 11:27:41 +02:00
|
|
|
for kw in kwargs:
|
2016-10-27 16:54:05 +02:00
|
|
|
if kw not in ("user", "password"):
|
|
|
|
setattr(self, kw, kwargs[kw])
|
|
|
|
# It's important to set user and password at the same time
|
|
|
|
if "user" in kwargs or "password" in kwargs:
|
|
|
|
self._set_auth(kwargs.get("user", self._user), kwargs.get("password", self._password))
|
2018-10-16 15:56:06 +07:00
|
|
|
if self._http_session and not self._http_session.closed:
|
|
|
|
await self._http_session.close()
|
2016-05-25 11:27:41 +02:00
|
|
|
self._connected = False
|
2018-08-16 21:31:56 +07:00
|
|
|
self._controller.notification.controller_emit("compute.updated", self.__json__())
|
2016-06-08 14:25:11 +02:00
|
|
|
self._controller.save()
|
2016-05-25 11:27:41 +02:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def close(self):
|
2019-04-14 16:48:12 +07:00
|
|
|
|
2016-06-02 13:44:12 +02:00
|
|
|
self._connected = False
|
2018-10-16 15:56:06 +07:00
|
|
|
if self._http_session and not self._http_session.closed:
|
|
|
|
await self._http_session.close()
|
2019-04-14 16:48:12 +07:00
|
|
|
try:
|
2019-04-14 17:42:20 +07:00
|
|
|
if self._notifications:
|
|
|
|
await self._notifications
|
2019-04-14 16:48:12 +07:00
|
|
|
except asyncio.CancelledError:
|
|
|
|
pass
|
2016-08-29 17:36:24 +02:00
|
|
|
self._closed = True
|
2016-06-02 13:44:12 +02:00
|
|
|
|
2016-05-23 11:20:52 +02:00
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""
|
|
|
|
:returns: Compute name
|
|
|
|
"""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@name.setter
|
|
|
|
def name(self, name):
|
2021-04-05 14:21:41 +09:30
|
|
|
|
2021-04-05 14:39:50 +09:30
|
|
|
if name is not None:
|
|
|
|
self._name = name
|
|
|
|
else:
|
|
|
|
if self._user:
|
|
|
|
user = self._user
|
|
|
|
# Due to random user generated by 1.4 it's common to have a very long user
|
|
|
|
if len(user) > 14:
|
|
|
|
user = user[:11] + "..."
|
2021-04-13 18:37:58 +09:30
|
|
|
self._name = f"{self._protocol}://{user}@{self._host}:{self._port}"
|
2021-04-05 14:39:50 +09:30
|
|
|
else:
|
2021-04-13 18:37:58 +09:30
|
|
|
self._name = f"{self._protocol}://{self._host}:{self._port}"
|
2016-05-23 11:20:52 +02:00
|
|
|
|
2016-05-11 16:31:16 +02:00
|
|
|
@property
|
|
|
|
def connected(self):
|
|
|
|
"""
|
|
|
|
:returns: True if compute node is connected
|
|
|
|
"""
|
|
|
|
return self._connected
|
|
|
|
|
2016-03-03 16:02:27 +01:00
|
|
|
@property
|
|
|
|
def id(self):
|
|
|
|
"""
|
2016-04-15 17:57:06 +02:00
|
|
|
:returns: Compute identifier (string)
|
2016-03-03 16:02:27 +01:00
|
|
|
"""
|
|
|
|
return self._id
|
|
|
|
|
|
|
|
@property
|
|
|
|
def host(self):
|
|
|
|
"""
|
2016-04-15 17:57:06 +02:00
|
|
|
:returns: Compute host (string)
|
2016-03-03 16:02:27 +01:00
|
|
|
"""
|
|
|
|
return self._host
|
|
|
|
|
2016-08-25 19:14:29 +02:00
|
|
|
@property
|
|
|
|
def host_ip(self):
|
|
|
|
"""
|
|
|
|
Return the IP associated to the host
|
|
|
|
"""
|
2017-02-06 16:47:40 +01:00
|
|
|
try:
|
|
|
|
return socket.gethostbyname(self._host)
|
|
|
|
except socket.gaierror:
|
2021-04-13 18:46:50 +09:30
|
|
|
return "0.0.0.0"
|
2016-08-25 19:14:29 +02:00
|
|
|
|
2016-05-25 11:27:41 +02:00
|
|
|
@host.setter
|
|
|
|
def host(self, host):
|
|
|
|
self._host = host
|
2016-10-26 18:32:01 +02:00
|
|
|
if self._console_host is None:
|
|
|
|
self._console_host = host
|
2016-05-25 11:27:41 +02:00
|
|
|
|
2016-10-26 14:43:47 +02:00
|
|
|
@property
|
|
|
|
def console_host(self):
|
|
|
|
return self._console_host
|
|
|
|
|
2016-04-19 15:35:50 +02:00
|
|
|
@property
|
|
|
|
def port(self):
|
|
|
|
"""
|
|
|
|
:returns: Compute port (integer)
|
|
|
|
"""
|
|
|
|
return self._port
|
|
|
|
|
2016-05-25 11:27:41 +02:00
|
|
|
@port.setter
|
|
|
|
def port(self, port):
|
|
|
|
self._port = port
|
|
|
|
|
2016-04-19 15:35:50 +02:00
|
|
|
@property
|
|
|
|
def protocol(self):
|
|
|
|
"""
|
|
|
|
:returns: Compute protocol (string)
|
|
|
|
"""
|
|
|
|
return self._protocol
|
|
|
|
|
2016-05-25 11:27:41 +02:00
|
|
|
@protocol.setter
|
|
|
|
def protocol(self, protocol):
|
|
|
|
self._protocol = protocol
|
|
|
|
|
2016-03-16 15:55:07 +01:00
|
|
|
@property
|
|
|
|
def user(self):
|
|
|
|
return self._user
|
|
|
|
|
|
|
|
@user.setter
|
|
|
|
def user(self, value):
|
2016-05-11 15:19:00 -06:00
|
|
|
self._set_auth(value, self._password)
|
2016-03-16 15:55:07 +01:00
|
|
|
|
|
|
|
@property
|
|
|
|
def password(self):
|
|
|
|
return self._password
|
|
|
|
|
2016-06-01 17:50:31 -06:00
|
|
|
@password.setter
|
2016-03-16 15:55:07 +01:00
|
|
|
def password(self, value):
|
2016-05-11 15:19:00 -06:00
|
|
|
self._set_auth(self._user, value)
|
2016-03-16 15:55:07 +01:00
|
|
|
|
2016-06-30 09:45:11 +02:00
|
|
|
@property
|
|
|
|
def cpu_usage_percent(self):
|
|
|
|
return self._cpu_usage_percent
|
|
|
|
|
|
|
|
@property
|
|
|
|
def memory_usage_percent(self):
|
|
|
|
return self._memory_usage_percent
|
|
|
|
|
2020-07-19 14:16:07 +09:30
|
|
|
@property
|
|
|
|
def disk_usage_percent(self):
|
|
|
|
return self._disk_usage_percent
|
|
|
|
|
2016-06-15 15:12:38 +02:00
|
|
|
def __json__(self, topology_dump=False):
|
|
|
|
"""
|
|
|
|
:param topology_dump: Filter to keep only properties require for saving on disk
|
|
|
|
"""
|
|
|
|
if topology_dump:
|
|
|
|
return {
|
|
|
|
"compute_id": self._id,
|
|
|
|
"name": self._name,
|
|
|
|
"protocol": self._protocol,
|
|
|
|
"host": self._host,
|
2021-04-13 18:46:50 +09:30
|
|
|
"port": self._port,
|
2016-06-15 15:12:38 +02:00
|
|
|
}
|
2016-03-03 16:02:27 +01:00
|
|
|
return {
|
2016-04-15 17:57:06 +02:00
|
|
|
"compute_id": self._id,
|
2016-05-23 11:20:52 +02:00
|
|
|
"name": self._name,
|
2016-03-03 16:02:27 +01:00
|
|
|
"protocol": self._protocol,
|
|
|
|
"host": self._host,
|
|
|
|
"port": self._port,
|
|
|
|
"user": self._user,
|
2016-06-30 09:45:11 +02:00
|
|
|
"connected": self._connected,
|
|
|
|
"cpu_usage_percent": self._cpu_usage_percent,
|
2016-08-29 15:53:10 +02:00
|
|
|
"memory_usage_percent": self._memory_usage_percent,
|
2020-07-19 14:16:07 +09:30
|
|
|
"disk_usage_percent": self._disk_usage_percent,
|
2018-08-22 16:54:43 +07:00
|
|
|
"capabilities": self._capabilities,
|
2021-04-13 18:46:50 +09:30
|
|
|
"last_error": self._last_error,
|
2016-03-03 16:02:27 +01:00
|
|
|
}
|
2016-03-10 10:32:07 +01:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def download_file(self, project, path):
|
2016-07-21 20:17:36 +02:00
|
|
|
"""
|
|
|
|
Read file of a project and download it
|
|
|
|
|
|
|
|
:param project: A project object
|
|
|
|
:param path: The path of the file in the project
|
|
|
|
:returns: A file stream
|
|
|
|
"""
|
|
|
|
|
2021-04-13 18:37:58 +09:30
|
|
|
url = self._getUrl(f"/projects/{project.id}/files/{path}")
|
2018-10-15 17:05:49 +07:00
|
|
|
response = await self._session().request("GET", url, auth=self._auth)
|
2016-07-21 20:17:36 +02:00
|
|
|
if response.status == 404:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerNotFoundError(f"{path} not found on compute")
|
2016-09-19 16:51:15 +02:00
|
|
|
return response
|
2016-07-21 20:17:36 +02:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def download_image(self, image_type, image):
|
2017-11-13 22:12:39 +01:00
|
|
|
"""
|
|
|
|
Read file of a project and download it
|
|
|
|
|
|
|
|
:param image_type: Image type
|
|
|
|
:param image: The path of the image
|
|
|
|
:returns: A file stream
|
|
|
|
"""
|
|
|
|
|
2021-04-13 18:37:58 +09:30
|
|
|
url = self._getUrl(f"/{image_type}/images/{image}")
|
2018-10-15 17:05:49 +07:00
|
|
|
response = await self._session().request("GET", url, auth=self._auth)
|
2017-11-13 22:12:39 +01:00
|
|
|
if response.status == 404:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerNotFoundError(f"{image} not found on compute")
|
2017-11-13 22:12:39 +01:00
|
|
|
return response
|
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def http_query(self, method, path, data=None, dont_connect=False, **kwargs):
|
2016-12-15 21:57:59 +01:00
|
|
|
"""
|
|
|
|
:param dont_connect: If true do not reconnect if not connected
|
|
|
|
"""
|
2018-01-10 16:22:55 +07:00
|
|
|
|
2016-12-15 21:57:59 +01:00
|
|
|
if not self._connected and not dont_connect:
|
2016-09-01 15:36:41 +02:00
|
|
|
if self._id == "vm" and not self._controller.gns3vm.running:
|
2018-10-15 17:05:49 +07:00
|
|
|
await self._controller.gns3vm.start()
|
|
|
|
await self.connect()
|
2016-12-15 21:57:59 +01:00
|
|
|
if not self._connected and not dont_connect:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ComputeError(f"Cannot connect to compute '{self._name}' with request {method} {path}")
|
2018-10-15 17:05:49 +07:00
|
|
|
response = await self._run_http_query(method, path, data=data, **kwargs)
|
2016-04-14 12:22:10 +02:00
|
|
|
return response
|
2016-03-18 16:55:54 +01:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def _try_reconnect(self):
|
2017-02-06 17:56:08 +01:00
|
|
|
"""
|
|
|
|
We catch error during reconnect
|
|
|
|
"""
|
|
|
|
try:
|
2018-10-15 17:05:49 +07:00
|
|
|
await self.connect()
|
2020-10-02 16:07:50 +09:30
|
|
|
except ControllerError:
|
2017-02-06 17:56:08 +01:00
|
|
|
pass
|
|
|
|
|
2018-08-25 14:10:47 +07:00
|
|
|
@locking
|
2018-10-15 17:05:49 +07:00
|
|
|
async def connect(self):
|
2016-03-18 16:55:54 +01:00
|
|
|
"""
|
|
|
|
Check if remote server is accessible
|
|
|
|
"""
|
2018-01-10 16:22:55 +07:00
|
|
|
|
2018-08-12 01:49:48 -07:00
|
|
|
if not self._connected and not self._closed and self.host:
|
2016-08-29 15:53:10 +02:00
|
|
|
try:
|
2021-04-13 18:37:58 +09:30
|
|
|
log.info(f"Connecting to compute '{self._id}'")
|
2018-10-15 17:05:49 +07:00
|
|
|
response = await self._run_http_query("GET", "/capabilities")
|
2018-01-10 16:22:55 +07:00
|
|
|
except ComputeError as e:
|
2021-04-13 18:37:58 +09:30
|
|
|
log.warning(f"Cannot connect to compute '{self._id}': {e}")
|
2020-06-01 19:40:53 +09:30
|
|
|
# Try to reconnect after 5 seconds if server unavailable only if not during tests (otherwise we create a ressource usage bomb)
|
2016-08-30 09:58:37 +02:00
|
|
|
if not hasattr(sys, "_called_from_test") or not sys._called_from_test:
|
2020-06-01 19:40:53 +09:30
|
|
|
if self.id != "local" and self.id != "vm" and not self._controller.compute_has_open_project(self):
|
2021-04-13 18:46:50 +09:30
|
|
|
log.warning(
|
|
|
|
f"Not reconnecting to compute '{self._id}' because there is no project opened on it"
|
|
|
|
)
|
2018-08-09 16:59:10 +07:00
|
|
|
return
|
2016-12-15 21:57:59 +01:00
|
|
|
self._connection_failure += 1
|
|
|
|
# After 5 failure we close the project using the compute to avoid sync issues
|
2018-08-09 16:59:10 +07:00
|
|
|
if self._connection_failure == 10:
|
2021-04-13 18:37:58 +09:30
|
|
|
log.error(f"Could not connect to compute '{self._id}' after multiple attempts: {e}")
|
2018-10-15 17:05:49 +07:00
|
|
|
await self._controller.close_compute_projects(self)
|
2020-06-01 19:40:53 +09:30
|
|
|
asyncio.get_event_loop().call_later(5, lambda: asyncio.ensure_future(self._try_reconnect()))
|
2016-08-29 15:53:10 +02:00
|
|
|
return
|
2020-10-22 20:37:34 +10:30
|
|
|
except web.HTTPNotFound:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerNotFoundError(f"The server {self._id} is not a GNS3 server or it's a 1.X server")
|
2020-10-22 20:37:34 +10:30
|
|
|
except web.HTTPUnauthorized:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerUnauthorizedError(f"Invalid auth for server {self._id}")
|
2020-10-22 20:37:34 +10:30
|
|
|
except web.HTTPServiceUnavailable:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerNotFoundError(f"The server {self._id} is unavailable")
|
2017-03-13 16:59:42 +01:00
|
|
|
except ValueError:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ComputeError(f"Invalid server url for server {self._id}")
|
2016-04-20 16:24:30 +02:00
|
|
|
|
2016-03-16 15:55:07 +01:00
|
|
|
if "version" not in response.json:
|
2021-04-13 18:37:58 +09:30
|
|
|
msg = f"The server {self._id} is not a GNS3 server"
|
2018-08-22 16:54:43 +07:00
|
|
|
log.error(msg)
|
2018-10-17 17:32:10 +07:00
|
|
|
await self._http_session.close()
|
2020-10-02 16:07:50 +09:30
|
|
|
raise ControllerNotFoundError(msg)
|
2016-08-29 15:53:10 +02:00
|
|
|
self._capabilities = response.json
|
2018-08-22 16:54:43 +07:00
|
|
|
|
|
|
|
if response.json["version"].split("-")[0] != __version__.split("-")[0]:
|
2018-09-28 15:04:38 +02:00
|
|
|
if self._name.startswith("GNS3 VM"):
|
2021-04-13 18:46:50 +09:30
|
|
|
msg = (
|
|
|
|
"GNS3 version {} is not the same as the GNS3 VM version {}. Please upgrade the GNS3 VM.".format(
|
|
|
|
__version__, response.json["version"]
|
|
|
|
)
|
|
|
|
)
|
2018-09-28 15:04:38 +02:00
|
|
|
else:
|
2021-04-13 18:46:50 +09:30
|
|
|
msg = "GNS3 controller version {} is not the same as compute {} version {}".format(
|
|
|
|
__version__, self._name, response.json["version"]
|
|
|
|
)
|
2018-08-22 16:54:43 +07:00
|
|
|
if __version_info__[3] == 0:
|
|
|
|
# Stable release
|
|
|
|
log.error(msg)
|
2018-10-17 17:32:10 +07:00
|
|
|
await self._http_session.close()
|
2018-08-22 16:54:43 +07:00
|
|
|
self._last_error = msg
|
2020-10-02 16:07:50 +09:30
|
|
|
raise ControllerError(msg)
|
2018-08-22 16:54:43 +07:00
|
|
|
elif parse_version(__version__)[:2] != parse_version(response.json["version"])[:2]:
|
|
|
|
# We don't allow different major version to interact even with dev build
|
|
|
|
log.error(msg)
|
2018-10-17 17:32:10 +07:00
|
|
|
await self._http_session.close()
|
2018-08-22 16:54:43 +07:00
|
|
|
self._last_error = msg
|
2020-10-02 16:07:50 +09:30
|
|
|
raise ControllerError(msg)
|
2018-08-22 16:54:43 +07:00
|
|
|
else:
|
2021-04-13 18:37:58 +09:30
|
|
|
msg = f"{msg}\nUsing different versions may result in unexpected problems. Please use at your own risk."
|
2019-03-15 13:14:55 +07:00
|
|
|
self._controller.notification.controller_emit("log.warning", {"message": msg})
|
2016-03-16 15:55:07 +01:00
|
|
|
|
2016-07-21 15:10:11 +02:00
|
|
|
self._notifications = asyncio.gather(self._connect_notification())
|
2016-03-18 16:55:54 +01:00
|
|
|
self._connected = True
|
2016-12-15 21:57:59 +01:00
|
|
|
self._connection_failure = 0
|
2018-08-22 16:54:43 +07:00
|
|
|
self._last_error = None
|
2018-08-16 21:31:56 +07:00
|
|
|
self._controller.notification.controller_emit("compute.updated", self.__json__())
|
2016-03-18 16:55:54 +01:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def _connect_notification(self):
|
2016-03-18 16:55:54 +01:00
|
|
|
"""
|
|
|
|
Connect to the notification stream
|
|
|
|
"""
|
2018-10-16 15:56:06 +07:00
|
|
|
|
2019-04-14 16:48:12 +07:00
|
|
|
ws_url = self._getUrl("/notifications/ws")
|
|
|
|
try:
|
|
|
|
async with self._session().ws_connect(ws_url, auth=self._auth, heartbeat=10) as ws:
|
2021-04-13 18:37:58 +09:30
|
|
|
log.info(f"Connected to compute '{self._id}' WebSocket '{ws_url}'")
|
2019-04-14 16:48:12 +07:00
|
|
|
async for response in ws:
|
|
|
|
if response.type == aiohttp.WSMsgType.TEXT:
|
|
|
|
msg = json.loads(response.data)
|
|
|
|
action = msg.pop("action")
|
|
|
|
event = msg.pop("event")
|
|
|
|
project_id = msg.pop("project_id", None)
|
|
|
|
if action == "ping":
|
|
|
|
self._cpu_usage_percent = event["cpu_usage_percent"]
|
|
|
|
self._memory_usage_percent = event["memory_usage_percent"]
|
2020-07-19 14:16:07 +09:30
|
|
|
self._disk_usage_percent = event["disk_usage_percent"]
|
2021-04-13 18:46:50 +09:30
|
|
|
# FIXME: slow down number of compute events
|
2019-04-14 16:48:12 +07:00
|
|
|
self._controller.notification.controller_emit("compute.updated", self.__json__())
|
|
|
|
else:
|
2021-04-13 18:46:50 +09:30
|
|
|
await self._controller.notification.dispatch(
|
|
|
|
action, event, project_id=project_id, compute_id=self.id
|
|
|
|
)
|
2018-10-16 15:56:06 +07:00
|
|
|
else:
|
2019-04-14 16:48:12 +07:00
|
|
|
if response.type == aiohttp.WSMsgType.CLOSE:
|
|
|
|
await ws.close()
|
|
|
|
elif response.type == aiohttp.WSMsgType.ERROR:
|
2021-04-13 18:37:58 +09:30
|
|
|
log.error(f"Error received on compute '{self._id}' WebSocket '{ws_url}': {ws.exception()}")
|
2019-04-14 16:48:12 +07:00
|
|
|
elif response.type == aiohttp.WSMsgType.CLOSED:
|
|
|
|
pass
|
|
|
|
break
|
2020-10-23 19:42:21 +10:30
|
|
|
except aiohttp.ClientError as e:
|
2021-04-13 18:37:58 +09:30
|
|
|
log.error(f"Client response error received on compute '{self._id}' WebSocket '{ws_url}': {e}")
|
2019-04-14 16:48:12 +07:00
|
|
|
finally:
|
2020-03-19 18:02:01 +10:30
|
|
|
self._connected = False
|
2021-04-13 18:37:58 +09:30
|
|
|
log.info(f"Connection closed to compute '{self._id}' WebSocket '{ws_url}'")
|
2019-04-14 16:48:12 +07:00
|
|
|
|
|
|
|
# Try to reconnect after 1 second if server unavailable only if not during tests (otherwise we create a ressources usage bomb)
|
2021-04-13 15:41:59 +09:30
|
|
|
if self.id != "local" and not hasattr(sys, "_called_from_test") or not sys._called_from_test:
|
2021-04-13 18:37:58 +09:30
|
|
|
log.info(f"Reconnecting to to compute '{self._id}' WebSocket '{ws_url}'")
|
2018-10-15 17:05:49 +07:00
|
|
|
asyncio.get_event_loop().call_later(1, lambda: asyncio.ensure_future(self.connect()))
|
2018-10-16 15:56:06 +07:00
|
|
|
|
2016-08-29 17:36:24 +02:00
|
|
|
self._cpu_usage_percent = None
|
|
|
|
self._memory_usage_percent = None
|
2020-07-19 14:16:07 +09:30
|
|
|
self._disk_usage_percent = None
|
2018-08-16 21:31:56 +07:00
|
|
|
self._controller.notification.controller_emit("compute.updated", self.__json__())
|
2016-03-18 16:55:54 +01:00
|
|
|
|
|
|
|
def _getUrl(self, path):
|
2016-10-17 14:10:25 +02:00
|
|
|
host = self._host
|
|
|
|
# IPV6
|
2016-10-17 18:20:29 +02:00
|
|
|
if host:
|
|
|
|
# IPV6
|
|
|
|
if ":" in host:
|
|
|
|
# Reduce IPV6 to his simple form
|
|
|
|
host = str(ipaddress.IPv6Address(host))
|
|
|
|
if host == "::":
|
|
|
|
host = "::1"
|
2021-04-13 18:37:58 +09:30
|
|
|
host = f"[{host}]"
|
2016-10-17 18:20:29 +02:00
|
|
|
elif host == "0.0.0.0":
|
|
|
|
host = "127.0.0.1"
|
2021-04-13 18:37:58 +09:30
|
|
|
return f"{self._protocol}://{host}:{self._port}/v3/compute{path}"
|
2016-03-16 15:55:07 +01:00
|
|
|
|
2017-10-13 11:03:56 +02:00
|
|
|
def get_url(self, path):
|
|
|
|
""" Returns URL for specific path at Compute"""
|
|
|
|
return self._getUrl(path)
|
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def _run_http_query(self, method, path, data=None, timeout=20, raw=False):
|
2018-10-16 15:56:06 +07:00
|
|
|
with async_timeout.timeout(timeout):
|
2016-03-18 16:55:54 +01:00
|
|
|
url = self._getUrl(path)
|
2021-04-13 18:46:50 +09:30
|
|
|
headers = {"content-type": "application/json"}
|
2017-05-16 19:28:47 +02:00
|
|
|
chunked = None
|
2016-04-14 12:22:10 +02:00
|
|
|
if data == {}:
|
|
|
|
data = None
|
|
|
|
elif data is not None:
|
2021-04-13 18:46:50 +09:30
|
|
|
if hasattr(data, "__json__"):
|
2016-06-06 19:51:35 +02:00
|
|
|
data = json.dumps(data.__json__())
|
2016-12-15 17:12:54 +01:00
|
|
|
elif isinstance(data, aiohttp.streams.EmptyStreamReader):
|
|
|
|
data = None
|
2016-06-06 19:51:35 +02:00
|
|
|
# Stream the request
|
2016-10-04 10:56:38 +02:00
|
|
|
elif isinstance(data, aiohttp.streams.StreamReader) or isinstance(data, bytes):
|
2016-06-06 19:51:35 +02:00
|
|
|
chunked = True
|
2021-04-13 18:46:50 +09:30
|
|
|
headers["content-type"] = "application/octet-stream"
|
2016-10-04 10:56:38 +02:00
|
|
|
# If the data is an open file we will iterate on it
|
|
|
|
elif isinstance(data, io.BufferedIOBase):
|
|
|
|
chunked = True
|
2021-04-13 18:46:50 +09:30
|
|
|
headers["content-type"] = "application/octet-stream"
|
2016-06-06 19:51:35 +02:00
|
|
|
else:
|
2017-05-16 19:28:47 +02:00
|
|
|
data = json.dumps(data).encode("utf-8")
|
2016-11-11 10:38:59 +01:00
|
|
|
try:
|
2021-04-13 18:37:58 +09:30
|
|
|
log.debug(f"Attempting request to compute: {method} {url} {headers}")
|
2021-04-13 18:46:50 +09:30
|
|
|
response = await self._session().request(
|
|
|
|
method, url, headers=headers, data=data, auth=self._auth, chunked=chunked, timeout=timeout
|
|
|
|
)
|
2018-06-07 22:26:23 +07:00
|
|
|
except asyncio.TimeoutError:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ComputeError(f"Timeout error for {method} call to {url} after {timeout}s")
|
2021-04-13 18:46:50 +09:30
|
|
|
except (
|
|
|
|
aiohttp.ClientError,
|
|
|
|
aiohttp.ServerDisconnectedError,
|
|
|
|
aiohttp.ClientResponseError,
|
|
|
|
ValueError,
|
|
|
|
KeyError,
|
|
|
|
socket.gaierror,
|
|
|
|
) as e:
|
2017-10-26 16:29:01 +02:00
|
|
|
# aiohttp 2.3.1 raises socket.gaierror when cannot find host
|
2016-11-11 10:38:59 +01:00
|
|
|
raise ComputeError(str(e))
|
2018-10-15 17:05:49 +07:00
|
|
|
body = await response.read()
|
2016-07-27 18:31:02 +02:00
|
|
|
if body and not raw:
|
2016-07-11 09:33:55 +02:00
|
|
|
body = body.decode()
|
|
|
|
|
|
|
|
if response.status >= 300:
|
|
|
|
# Try to decode the GNS3 error
|
2016-07-27 18:31:02 +02:00
|
|
|
if body and not raw:
|
2016-03-18 16:55:54 +01:00
|
|
|
try:
|
2016-07-11 09:33:55 +02:00
|
|
|
msg = json.loads(body)["message"]
|
|
|
|
except (KeyError, ValueError):
|
|
|
|
msg = body
|
|
|
|
else:
|
|
|
|
msg = ""
|
|
|
|
|
2020-10-02 16:07:50 +09:30
|
|
|
if response.status == 401:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerUnauthorizedError(f"Invalid authentication for compute {self.id}")
|
2016-07-11 09:33:55 +02:00
|
|
|
elif response.status == 403:
|
2020-10-02 16:07:50 +09:30
|
|
|
raise ControllerForbiddenError(msg)
|
2016-07-11 09:33:55 +02:00
|
|
|
elif response.status == 404:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerNotFoundError(f"{method} {path} not found")
|
2017-05-16 08:36:54 +02:00
|
|
|
elif response.status == 408 or response.status == 504:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerTimeoutError(f"{method} {path} request timeout")
|
2016-07-11 09:33:55 +02:00
|
|
|
elif response.status == 409:
|
|
|
|
try:
|
|
|
|
raise ComputeConflict(json.loads(body))
|
|
|
|
# If the 409 doesn't come from a GNS3 server
|
2016-05-11 15:19:00 -06:00
|
|
|
except ValueError:
|
2020-10-02 16:07:50 +09:30
|
|
|
raise ControllerError(msg)
|
2016-07-11 09:33:55 +02:00
|
|
|
elif response.status == 500:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise aiohttp.web.HTTPInternalServerError(text=f"Internal server error {url}")
|
2016-07-11 09:33:55 +02:00
|
|
|
elif response.status == 503:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise aiohttp.web.HTTPServiceUnavailable(text=f"Service unavailable {url} {body}")
|
2016-06-06 19:51:35 +02:00
|
|
|
else:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise NotImplementedError(f"{response.status} status code is not supported for {method} '{url}'")
|
2016-07-11 09:33:55 +02:00
|
|
|
if body and len(body):
|
2016-07-27 18:31:02 +02:00
|
|
|
if raw:
|
|
|
|
response.body = body
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
response.json = json.loads(body)
|
|
|
|
except ValueError:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ControllerError(f"The server {self._id} is not a GNS3 server")
|
2016-07-11 09:33:55 +02:00
|
|
|
else:
|
|
|
|
response.json = {}
|
2016-07-27 18:31:02 +02:00
|
|
|
response.body = b""
|
2016-07-11 09:33:55 +02:00
|
|
|
return response
|
2016-03-10 10:32:07 +01:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def get(self, path, **kwargs):
|
2020-10-23 19:42:21 +10:30
|
|
|
return await self.http_query("GET", path, **kwargs)
|
2016-04-14 12:22:10 +02:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def post(self, path, data={}, **kwargs):
|
|
|
|
response = await self.http_query("POST", path, data, **kwargs)
|
2016-04-14 12:22:10 +02:00
|
|
|
return response
|
2016-03-14 20:54:05 +01:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def put(self, path, data={}, **kwargs):
|
|
|
|
response = await self.http_query("PUT", path, data, **kwargs)
|
2016-04-18 17:36:38 +02:00
|
|
|
return response
|
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def delete(self, path, **kwargs):
|
2020-10-02 16:07:50 +09:30
|
|
|
return await self.http_query("DELETE", path, **kwargs)
|
2016-06-02 16:44:38 +02:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def forward(self, method, type, path, data=None):
|
2016-06-02 16:44:38 +02:00
|
|
|
"""
|
|
|
|
Forward a call to the emulator on compute
|
|
|
|
"""
|
2016-11-02 11:06:45 +01:00
|
|
|
try:
|
2021-04-13 18:37:58 +09:30
|
|
|
action = f"/{type}/{path}"
|
2018-10-15 17:05:49 +07:00
|
|
|
res = await self.http_query(method, action, data=data, timeout=None)
|
2017-05-16 19:28:47 +02:00
|
|
|
except aiohttp.ServerDisconnectedError:
|
2020-10-23 19:42:21 +10:30
|
|
|
raise ControllerError(f"Connection lost to {self._id} during {method} {action}")
|
2016-06-02 18:38:47 +02:00
|
|
|
return res.json
|
2016-06-08 14:14:03 +02:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def images(self, type):
|
2016-06-08 14:14:03 +02:00
|
|
|
"""
|
2019-03-18 15:33:37 +07:00
|
|
|
Return the list of images available for this type on the compute node.
|
2016-06-08 14:14:03 +02:00
|
|
|
"""
|
2020-10-22 20:05:37 +10:30
|
|
|
images = []
|
2016-06-08 14:14:03 +02:00
|
|
|
|
2021-04-13 18:37:58 +09:30
|
|
|
res = await self.http_query("GET", f"/{type}/images", timeout=None)
|
2016-06-08 14:14:03 +02:00
|
|
|
images = res.json
|
|
|
|
|
2016-12-12 10:13:29 +01:00
|
|
|
try:
|
|
|
|
if type in ["qemu", "dynamips", "iou"]:
|
2021-04-13 18:46:50 +09:30
|
|
|
# for local_image in list_images(type):
|
2019-03-18 15:33:37 +07:00
|
|
|
# if local_image['filename'] not in [i['filename'] for i in images]:
|
|
|
|
# images.append(local_image)
|
2021-04-13 18:46:50 +09:30
|
|
|
images = sorted(images, key=itemgetter("filename"))
|
2017-05-09 12:25:15 +02:00
|
|
|
else:
|
2021-04-13 18:46:50 +09:30
|
|
|
images = sorted(images, key=itemgetter("image"))
|
2016-12-12 10:13:29 +01:00
|
|
|
except OSError as e:
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ComputeError(f"Cannot list images: {str(e)}")
|
2016-06-08 14:14:03 +02:00
|
|
|
return images
|
2016-07-21 20:17:36 +02:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def list_files(self, project):
|
2016-07-21 20:17:36 +02:00
|
|
|
"""
|
|
|
|
List files in the project on computes
|
|
|
|
"""
|
2021-04-13 18:37:58 +09:30
|
|
|
path = f"/projects/{project.id}/files"
|
2018-10-15 17:05:49 +07:00
|
|
|
res = await self.http_query("GET", path, timeout=None)
|
2016-07-21 20:17:36 +02:00
|
|
|
return res.json
|
2016-08-25 19:14:29 +02:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
async def get_ip_on_same_subnet(self, other_compute):
|
2016-08-25 19:14:29 +02:00
|
|
|
"""
|
2018-03-15 14:17:39 +07:00
|
|
|
Try to find the best ip for communication from one compute
|
2016-08-25 19:14:29 +02:00
|
|
|
to another
|
|
|
|
|
|
|
|
:returns: Tuple (ip_for_this_compute, ip_for_other_compute)
|
|
|
|
"""
|
|
|
|
if other_compute == self:
|
2020-10-23 19:42:21 +10:30
|
|
|
return self.host_ip, self.host_ip
|
2016-08-25 19:14:29 +02:00
|
|
|
|
2016-12-13 16:46:09 +01:00
|
|
|
# Perhaps the user has correct network gateway, we trust him
|
2021-04-13 18:46:50 +09:30
|
|
|
if self.host_ip not in ("0.0.0.0", "127.0.0.1") and other_compute.host_ip not in ("0.0.0.0", "127.0.0.1"):
|
2020-10-23 19:42:21 +10:30
|
|
|
return self.host_ip, other_compute.host_ip
|
2016-12-13 16:46:09 +01:00
|
|
|
|
2018-10-15 17:05:49 +07:00
|
|
|
this_compute_interfaces = await self.interfaces()
|
|
|
|
other_compute_interfaces = await other_compute.interfaces()
|
2016-08-25 19:14:29 +02:00
|
|
|
|
|
|
|
# Sort interface to put the compute host in first position
|
|
|
|
# we guess that if user specified this host it could have a reason (VMware Nat / Host only interface)
|
|
|
|
this_compute_interfaces = sorted(this_compute_interfaces, key=lambda i: i["ip_address"] != self.host_ip)
|
2021-04-13 18:46:50 +09:30
|
|
|
other_compute_interfaces = sorted(
|
|
|
|
other_compute_interfaces, key=lambda i: i["ip_address"] != other_compute.host_ip
|
|
|
|
)
|
2016-08-25 19:14:29 +02:00
|
|
|
|
|
|
|
for this_interface in this_compute_interfaces:
|
2016-09-05 18:02:49 +02:00
|
|
|
# Skip if no ip or no netmask (vbox when stopped set a null netmask)
|
|
|
|
if len(this_interface["ip_address"]) == 0 or this_interface["netmask"] is None:
|
2016-08-25 19:14:29 +02:00
|
|
|
continue
|
2016-11-10 14:46:25 +01:00
|
|
|
# Ignore 169.254 network because it's for Windows special purpose
|
|
|
|
if this_interface["ip_address"].startswith("169.254."):
|
|
|
|
continue
|
2016-08-25 19:14:29 +02:00
|
|
|
|
2021-04-13 18:46:50 +09:30
|
|
|
this_network = ipaddress.ip_network(
|
|
|
|
"{}/{}".format(this_interface["ip_address"], this_interface["netmask"]), strict=False
|
|
|
|
)
|
2016-08-25 19:14:29 +02:00
|
|
|
|
|
|
|
for other_interface in other_compute_interfaces:
|
2016-11-11 16:08:52 +01:00
|
|
|
if len(other_interface["ip_address"]) == 0 or other_interface["netmask"] is None:
|
2016-08-25 19:14:29 +02:00
|
|
|
continue
|
|
|
|
|
|
|
|
# Avoid stuff like 127.0.0.1
|
|
|
|
if other_interface["ip_address"] == this_interface["ip_address"]:
|
|
|
|
continue
|
|
|
|
|
2021-04-13 18:46:50 +09:30
|
|
|
other_network = ipaddress.ip_network(
|
|
|
|
"{}/{}".format(other_interface["ip_address"], other_interface["netmask"]), strict=False
|
|
|
|
)
|
2016-08-25 19:14:29 +02:00
|
|
|
if this_network.overlaps(other_network):
|
2020-10-23 19:42:21 +10:30
|
|
|
return this_interface["ip_address"], other_interface["ip_address"]
|
2016-11-10 22:39:16 +01:00
|
|
|
|
2021-04-13 18:37:58 +09:30
|
|
|
raise ValueError(f"No common subnet for compute {self.name} and {other_compute.name}")
|