# Licensed under the Apache License, Version 2.0 (the "License"); you may # not use this file except in compliance with the License. You may obtain # a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the # License for the specific language governing permissions and limitations # under the License. import os import random import socket import sys import traceback import flask import jsonschema from neutronclient.common import exceptions as n_exceptions from neutronclient.neutron import client from neutronclient.v2_0 import client as client_v2 from oslo_concurrency import processutils from oslo_config import cfg from werkzeug import exceptions as w_exceptions from kuryr._i18n import _LE from kuryr.common import constants as const from kuryr.common import exceptions DOCKER_NETNS_BASE = '/var/run/docker/netns' PORT_POSTFIX = 'port' def get_neutron_client_simple(url, auth_url, token): auths = auth_url.rsplit('/', 1) version = auths[1][1:] return client.Client(version, endpoint_url=url, token=token) def get_neutron_client(url, username, tenant_name, password, auth_url, ca_cert, insecure, timeout=30): return client_v2.Client(endpoint_url=url, timeout=timeout, username=username, tenant_name=tenant_name, password=password, auth_url=auth_url, ca_cert=ca_cert, insecure=insecure) # Return all errors as JSON. From http://flask.pocoo.org/snippets/83/ def make_json_app(import_name, **kwargs): """Creates a JSON-oriented Flask app. All error responses that you don't specifically manage yourself will have application/json content type, and will contain JSON that follows the libnetwork remote driver protocol. { "Err": "405: Method Not Allowed" } See: - https://github.com/docker/libnetwork/blob/3c8e06bc0580a2a1b2440fe0792fbfcd43a9feca/docs/remote.md#errors # noqa """ app = flask.Flask(import_name, **kwargs) @app.errorhandler(exceptions.KuryrException) @app.errorhandler(n_exceptions.NeutronClientException) @app.errorhandler(jsonschema.ValidationError) @app.errorhandler(processutils.ProcessExecutionError) def make_json_error(ex): app.logger.error(_LE("Unexpected error happened: {0}").format(ex)) traceback.print_exc(file=sys.stderr) response = flask.jsonify({"Err": str(ex)}) response.status_code = w_exceptions.InternalServerError.code if isinstance(ex, w_exceptions.HTTPException): response.status_code = ex.code elif isinstance(ex, n_exceptions.NeutronClientException): response.status_code = ex.status_code elif isinstance(ex, jsonschema.ValidationError): response.status_code = w_exceptions.BadRequest.code content_type = 'application/vnd.docker.plugins.v1+json; charset=utf-8' response.headers['Content-Type'] = content_type return response for code in w_exceptions.default_exceptions: app.error_handler_spec[None][code] = make_json_error return app def get_sandbox_key(container_id): """Returns a sandbox key constructed with the given container ID. :param container_id: the ID of the Docker container as string :returns: the constructed sandbox key as string """ return os.path.join(DOCKER_NETNS_BASE, container_id[:12]) def get_neutron_port_name(docker_endpoint_id): """Returns a Neutron port name. :param docker_endpoint_id: the EndpointID :returns: the Neutron port name formatted appropriately """ return '-'.join([docker_endpoint_id, PORT_POSTFIX]) def get_hostname(): """Returns the host name.""" return socket.gethostname() def get_neutron_subnetpool_name(subnet_cidr): """Returns a Neutron subnetpool name. :param subnet_cidr: The subnetpool allocation cidr :returns: the Neutron subnetpool_name name formatted appropriately """ name_prefix = cfg.CONF.subnetpool_name_prefix return '-'.join([name_prefix, subnet_cidr]) def get_dict_format_fixed_ips_from_kv_format(fixed_ips): """Returns fixed_ips in dict format. :param fixed_ips: Format that neutron client expects for list_ports ex, ['subnet_id=5083bda8-1b7c-4625-97f3-1d4c33bfeea8', 'ip_address=192.168.1.2'] :returns: normal dict form, [{'subnet_id': '5083bda8-1b7c-4625-97f3-1d4c33bfeea8', 'ip_address': '192.168.1.2'}] """ new_fixed_ips = [] for fixed_ip in fixed_ips: if 'subnet_id' == fixed_ip.split('=')[0]: subnet_id = fixed_ip.split('=')[1] else: ip = fixed_ip.split('=')[1] new_fixed_ips.append({'subnet_id': subnet_id, 'ip_address': ip}) return new_fixed_ips def getrandbits(bit_size=256): return str(random.getrandbits(bit_size)).encode('utf-8') def create_net_tags(tag): tags = [] tags.append(const.NEUTRON_ID_LH_OPTION + ':' + tag[:32]) if len(tag) > 32: tags.append(const.NEUTRON_ID_UH_OPTION + ':' + tag[32:64]) return tags def make_net_tags(tag): tags = create_net_tags(tag) return ','.join(map(str, tags)) def make_net_name(netid, tags=True): if tags: return const.NET_NAME_PREFIX + netid[:8] return netid