# Copyright (C) 2012 Midokura Japan K.K. # Copyright (C) 2013 Midokura PTE LTD # All Rights Reserved. # # 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. from midonetclient import exc from webob import exc as w_exc from neutron.common import exceptions as n_exc from neutron.openstack.common import log as logging from neutron.plugins.midonet.common import net_util LOG = logging.getLogger(__name__) def handle_api_error(fn): """Wrapper for methods that throws custom exceptions.""" def wrapped(*args, **kwargs): try: return fn(*args, **kwargs) except (w_exc.HTTPException, exc.MidoApiConnectionError) as ex: raise MidonetApiException(msg=ex) return wrapped class MidonetResourceNotFound(n_exc.NotFound): message = _('MidoNet %(resource_type)s %(id)s could not be found') class MidonetApiException(n_exc.NeutronException): message = _("MidoNet API error: %(msg)s") class MidoClient: def __init__(self, mido_api): self.mido_api = mido_api @classmethod def _fill_dto(cls, dto, fields): for field_name, field_value in fields.iteritems(): # We assume the setters are named the # same way as the attributes themselves. try: getattr(dto, field_name)(field_value) except AttributeError: pass return dto @classmethod def _create_dto(cls, dto, fields): return cls._fill_dto(dto, fields).create() @classmethod def _update_dto(cls, dto, fields): return cls._fill_dto(dto, fields).update() @handle_api_error def create_bridge(self, **kwargs): """Create a new bridge :param \**kwargs: configuration of the new bridge :returns: newly created bridge """ LOG.debug(_("MidoClient.create_bridge called: " "kwargs=%(kwargs)s"), {'kwargs': kwargs}) return self._create_dto(self.mido_api.add_bridge(), kwargs) @handle_api_error def delete_bridge(self, id): """Delete a bridge :param id: id of the bridge """ LOG.debug(_("MidoClient.delete_bridge called: id=%(id)s"), {'id': id}) return self.mido_api.delete_bridge(id) @handle_api_error def get_bridge(self, id): """Get a bridge :param id: id of the bridge :returns: requested bridge. None if bridge does not exist. """ LOG.debug(_("MidoClient.get_bridge called: id=%s"), id) try: return self.mido_api.get_bridge(id) except w_exc.HTTPNotFound: raise MidonetResourceNotFound(resource_type='Bridge', id=id) @handle_api_error def update_bridge(self, id, **kwargs): """Update a bridge of the given id with the new fields :param id: id of the bridge :param \**kwargs: the fields to update and their values :returns: bridge object """ LOG.debug(_("MidoClient.update_bridge called: " "id=%(id)s, kwargs=%(kwargs)s"), {'id': id, 'kwargs': kwargs}) try: return self._update_dto(self.mido_api.get_bridge(id), kwargs) except w_exc.HTTPNotFound: raise MidonetResourceNotFound(resource_type='Bridge', id=id) @handle_api_error def create_dhcp(self, bridge, gateway_ip, cidr, host_rts=None, dns_servers=None): """Create a new DHCP entry :param bridge: bridge object to add dhcp to :param gateway_ip: IP address of gateway :param cidr: subnet represented as x.x.x.x/y :param host_rts: list of routes set in the host :param dns_servers: list of dns servers :returns: newly created dhcp """ LOG.debug(_("MidoClient.create_dhcp called: bridge=%(bridge)s, " "cidr=%(cidr)s, gateway_ip=%(gateway_ip)s, " "host_rts=%(host_rts)s, dns_servers=%(dns_servers)s"), {'bridge': bridge, 'cidr': cidr, 'gateway_ip': gateway_ip, 'host_rts': host_rts, 'dns_servers': dns_servers}) self.mido_api.add_bridge_dhcp(bridge, gateway_ip, cidr, host_rts=host_rts, dns_nservers=dns_servers) @handle_api_error def add_dhcp_host(self, bridge, cidr, ip, mac): """Add DHCP host entry :param bridge: bridge the DHCP is configured for :param cidr: subnet represented as x.x.x.x/y :param ip: IP address :param mac: MAC address """ LOG.debug(_("MidoClient.add_dhcp_host called: bridge=%(bridge)s, " "cidr=%(cidr)s, ip=%(ip)s, mac=%(mac)s"), {'bridge': bridge, 'cidr': cidr, 'ip': ip, 'mac': mac}) subnet = bridge.get_dhcp_subnet(net_util.subnet_str(cidr)) if subnet is None: raise MidonetApiException(msg=_("Tried to add to" "non-existent DHCP")) subnet.add_dhcp_host().ip_addr(ip).mac_addr(mac).create() @handle_api_error def remove_dhcp_host(self, bridge, cidr, ip, mac): """Remove DHCP host entry :param bridge: bridge the DHCP is configured for :param cidr: subnet represented as x.x.x.x/y :param ip: IP address :param mac: MAC address """ LOG.debug(_("MidoClient.remove_dhcp_host called: bridge=%(bridge)s, " "cidr=%(cidr)s, ip=%(ip)s, mac=%(mac)s"), {'bridge': bridge, 'cidr': cidr, 'ip': ip, 'mac': mac}) subnet = bridge.get_dhcp_subnet(net_util.subnet_str(cidr)) if subnet is None: LOG.warn(_("Tried to delete mapping from non-existent subnet")) return for dh in subnet.get_dhcp_hosts(): if dh.get_mac_addr() == mac and dh.get_ip_addr() == ip: LOG.debug(_("MidoClient.remove_dhcp_host: Deleting %(dh)r"), {"dh": dh}) dh.delete() @handle_api_error def delete_dhcp_host(self, bridge_id, cidr, ip, mac): """Delete DHCP host entry :param bridge_id: id of the bridge of the DHCP :param cidr: subnet represented as x.x.x.x/y :param ip: IP address :param mac: MAC address """ LOG.debug(_("MidoClient.delete_dhcp_host called: " "bridge_id=%(bridge_id)s, cidr=%(cidr)s, ip=%(ip)s, " "mac=%(mac)s"), {'bridge_id': bridge_id, 'cidr': cidr, 'ip': ip, 'mac': mac}) bridge = self.get_bridge(bridge_id) self.remove_dhcp_host(bridge, net_util.subnet_str(cidr), ip, mac) @handle_api_error def delete_dhcp(self, bridge, cidr): """Delete a DHCP entry :param bridge: bridge to remove DHCP from :param cidr: subnet represented as x.x.x.x/y """ LOG.debug(_("MidoClient.delete_dhcp called: bridge=%(bridge)s, " "cidr=%(cidr)s"), {'bridge': bridge, 'cidr': cidr}) dhcp_subnets = bridge.get_dhcp_subnets() net_addr, net_len = net_util.net_addr(cidr) if not dhcp_subnets: raise MidonetApiException( msg=_("Tried to delete non-existent DHCP")) for dhcp in dhcp_subnets: if (dhcp.get_subnet_prefix() == net_addr and dhcp.get_subnet_length() == str(net_len)): dhcp.delete() break @handle_api_error def delete_port(self, id, delete_chains=False): """Delete a port :param id: id of the port """ LOG.debug(_("MidoClient.delete_port called: id=%(id)s, " "delete_chains=%(delete_chains)s"), {'id': id, 'delete_chains': delete_chains}) if delete_chains: self.delete_port_chains(id) self.mido_api.delete_port(id) @handle_api_error def get_port(self, id): """Get a port :param id: id of the port :returns: requested port. None if it does not exist """ LOG.debug(_("MidoClient.get_port called: id=%(id)s"), {'id': id}) try: return self.mido_api.get_port(id) except w_exc.HTTPNotFound: raise MidonetResourceNotFound(resource_type='Port', id=id) @handle_api_error def add_bridge_port(self, bridge, **kwargs): """Add a port on a bridge :param bridge: bridge to add a new port to :param \**kwargs: configuration of the new port :returns: newly created port """ LOG.debug(_("MidoClient.add_bridge_port called: " "bridge=%(bridge)s, kwargs=%(kwargs)s"), {'bridge': bridge, 'kwargs': kwargs}) return self._create_dto(self.mido_api.add_bridge_port(bridge), kwargs) @handle_api_error def update_port(self, id, **kwargs): """Update a port of the given id with the new fields :param id: id of the port :param \**kwargs: the fields to update and their values """ LOG.debug(_("MidoClient.update_port called: " "id=%(id)s, kwargs=%(kwargs)s"), {'id': id, 'kwargs': kwargs}) try: return self._update_dto(self.mido_api.get_port(id), kwargs) except w_exc.HTTPNotFound: raise MidonetResourceNotFound(resource_type='Port', id=id) @handle_api_error def add_router_port(self, router, **kwargs): """Add a new port to an existing router. :param router: router to add a new port to :param \**kwargs: configuration of the new port :returns: newly created port """ return self._create_dto(self.mido_api.add_router_port(router), kwargs) @handle_api_error def create_router(self, **kwargs): """Create a new router :param \**kwargs: configuration of the new router :returns: newly created router """ LOG.debug(_("MidoClient.create_router called: " "kwargs=%(kwargs)s"), {'kwargs': kwargs}) return self._create_dto(self.mido_api.add_router(), kwargs) @handle_api_error def delete_router(self, id): """Delete a router :param id: id of the router """ LOG.debug(_("MidoClient.delete_router called: id=%(id)s"), {'id': id}) return self.mido_api.delete_router(id) @handle_api_error def get_router(self, id): """Get a router with the given id :param id: id of the router :returns: requested router object. None if it does not exist. """ LOG.debug(_("MidoClient.get_router called: id=%(id)s"), {'id': id}) try: return self.mido_api.get_router(id) except w_exc.HTTPNotFound: raise MidonetResourceNotFound(resource_type='Router', id=id) @handle_api_error def update_router(self, id, **kwargs): """Update a router of the given id with the new name :param id: id of the router :param \**kwargs: the fields to update and their values :returns: router object """ LOG.debug(_("MidoClient.update_router called: " "id=%(id)s, kwargs=%(kwargs)s"), {'id': id, 'kwargs': kwargs}) try: return self._update_dto(self.mido_api.get_router(id), kwargs) except w_exc.HTTPNotFound: raise MidonetResourceNotFound(resource_type='Router', id=id) @handle_api_error def delete_route(self, id): return self.mido_api.delete_route(id) @handle_api_error def add_dhcp_route_option(self, bridge, cidr, gw_ip, dst_ip): """Add Option121 route to subnet :param bridge: Bridge to add the option route to :param cidr: subnet represented as x.x.x.x/y :param gw_ip: IP address of the next hop :param dst_ip: IP address of the destination, in x.x.x.x/y format """ LOG.debug(_("MidoClient.add_dhcp_route_option called: " "bridge=%(bridge)s, cidr=%(cidr)s, gw_ip=%(gw_ip)s" "dst_ip=%(dst_ip)s"), {"bridge": bridge, "cidr": cidr, "gw_ip": gw_ip, "dst_ip": dst_ip}) subnet = bridge.get_dhcp_subnet(net_util.subnet_str(cidr)) if subnet is None: raise MidonetApiException( msg=_("Tried to access non-existent DHCP")) prefix, length = dst_ip.split("/") routes = [{'destinationPrefix': prefix, 'destinationLength': length, 'gatewayAddr': gw_ip}] cur_routes = subnet.get_opt121_routes() if cur_routes: routes = routes + cur_routes subnet.opt121_routes(routes).update() @handle_api_error def link(self, port, peer_id): """Link a port to a given peerId.""" self.mido_api.link(port, peer_id) @handle_api_error def delete_port_routes(self, routes, port_id): """Remove routes whose next hop port is the given port ID.""" for route in routes: if route.get_next_hop_port() == port_id: self.mido_api.delete_route(route.get_id()) @handle_api_error def get_router_routes(self, router_id): """Get all routes for the given router.""" return self.mido_api.get_router_routes(router_id) @handle_api_error def unlink(self, port): """Unlink a port :param port: port object """ LOG.debug(_("MidoClient.unlink called: port=%(port)s"), {'port': port}) if port.get_peer_id(): self.mido_api.unlink(port) else: LOG.warn(_("Attempted to unlink a port that was not linked. %s"), port.get_id()) @handle_api_error def remove_rules_by_property(self, tenant_id, chain_name, key, value): """Remove all the rules that match the provided key and value.""" LOG.debug(_("MidoClient.remove_rules_by_property called: " "tenant_id=%(tenant_id)s, chain_name=%(chain_name)s" "key=%(key)s, value=%(value)s"), {'tenant_id': tenant_id, 'chain_name': chain_name, 'key': key, 'value': value}) chain = self.get_chain_by_name(tenant_id, chain_name) if chain is None: raise MidonetResourceNotFound(resource_type='Chain', id=chain_name) for r in chain.get_rules(): if key in r.get_properties(): if r.get_properties()[key] == value: self.mido_api.delete_rule(r.get_id()) @handle_api_error def add_router_chains(self, router, inbound_chain_name, outbound_chain_name): """Create chains for a new router. Creates inbound and outbound chains for the router with the given names, and the new chains are set on the router. :param router: router to set chains for :param inbound_chain_name: Name of the inbound chain :param outbound_chain_name: Name of the outbound chain """ LOG.debug(_("MidoClient.create_router_chains called: " "router=%(router)s, inbound_chain_name=%(in_chain)s, " "outbound_chain_name=%(out_chain)s"), {"router": router, "in_chain": inbound_chain_name, "out_chain": outbound_chain_name}) tenant_id = router.get_tenant_id() inbound_chain = self.mido_api.add_chain().tenant_id(tenant_id).name( inbound_chain_name,).create() outbound_chain = self.mido_api.add_chain().tenant_id(tenant_id).name( outbound_chain_name).create() # set chains to in/out filters router.inbound_filter_id(inbound_chain.get_id()).outbound_filter_id( outbound_chain.get_id()).update() return inbound_chain, outbound_chain @handle_api_error def delete_router_chains(self, id): """Deletes chains of a router. :param id: router ID to delete chains of """ LOG.debug(_("MidoClient.delete_router_chains called: " "id=%(id)s"), {'id': id}) router = self.get_router(id) if (router.get_inbound_filter_id()): self.mido_api.delete_chain(router.get_inbound_filter_id()) if (router.get_outbound_filter_id()): self.mido_api.delete_chain(router.get_outbound_filter_id()) @handle_api_error def delete_port_chains(self, id): """Deletes chains of a port. :param id: port ID to delete chains of """ LOG.debug(_("MidoClient.delete_port_chains called: " "id=%(id)s"), {'id': id}) port = self.get_port(id) if (port.get_inbound_filter_id()): self.mido_api.delete_chain(port.get_inbound_filter_id()) if (port.get_outbound_filter_id()): self.mido_api.delete_chain(port.get_outbound_filter_id()) @handle_api_error def get_link_port(self, router, peer_router_id): """Setup a route on the router to the next hop router.""" LOG.debug(_("MidoClient.get_link_port called: " "router=%(router)s, peer_router_id=%(peer_router_id)s"), {'router': router, 'peer_router_id': peer_router_id}) # Find the port linked between the two routers link_port = None for p in router.get_peer_ports(): if p.get_device_id() == peer_router_id: link_port = p break return link_port @handle_api_error def add_router_route(self, router, type='Normal', src_network_addr=None, src_network_length=None, dst_network_addr=None, dst_network_length=None, next_hop_port=None, next_hop_gateway=None, weight=100): """Setup a route on the router.""" return self.mido_api.add_router_route( router, type=type, src_network_addr=src_network_addr, src_network_length=src_network_length, dst_network_addr=dst_network_addr, dst_network_length=dst_network_length, next_hop_port=next_hop_port, next_hop_gateway=next_hop_gateway, weight=weight) @handle_api_error def add_static_nat(self, tenant_id, chain_name, from_ip, to_ip, port_id, nat_type='dnat', **kwargs): """Add a static NAT entry :param tenant_id: owner fo the chain to add a NAT to :param chain_name: name of the chain to add a NAT to :param from_ip: IP to translate from :param from_ip: IP to translate from :param to_ip: IP to translate to :param port_id: port to match on :param nat_type: 'dnat' or 'snat' """ LOG.debug(_("MidoClient.add_static_nat called: " "tenant_id=%(tenant_id)s, chain_name=%(chain_name)s, " "from_ip=%(from_ip)s, to_ip=%(to_ip)s, " "port_id=%(port_id)s, nat_type=%(nat_type)s"), {'tenant_id': tenant_id, 'chain_name': chain_name, 'from_ip': from_ip, 'to_ip': to_ip, 'portid': port_id, 'nat_type': nat_type}) if nat_type not in ['dnat', 'snat']: raise ValueError(_("Invalid NAT type passed in %s") % nat_type) chain = self.get_chain_by_name(tenant_id, chain_name) nat_targets = [] nat_targets.append( {'addressFrom': to_ip, 'addressTo': to_ip, 'portFrom': 0, 'portTo': 0}) rule = chain.add_rule().type(nat_type).flow_action('accept').position( 1).nat_targets(nat_targets).properties(kwargs) if nat_type == 'dnat': rule = rule.nw_dst_address(from_ip).nw_dst_length(32).in_ports( [port_id]) else: rule = rule.nw_src_address(from_ip).nw_src_length(32).out_ports( [port_id]) return rule.create() @handle_api_error def add_dynamic_snat(self, tenant_id, pre_chain_name, post_chain_name, snat_ip, port_id, **kwargs): """Add SNAT masquerading rule MidoNet requires two rules on the router, one to do NAT to a range of ports, and another to retrieve back the original IP in the return flow. """ pre_chain = self.get_chain_by_name(tenant_id, pre_chain_name) post_chain = self.get_chain_by_name(tenant_id, post_chain_name) pre_chain.add_rule().nw_dst_address(snat_ip).nw_dst_length( 32).type('rev_snat').flow_action('accept').in_ports( [port_id]).properties(kwargs).position(1).create() nat_targets = [] nat_targets.append( {'addressFrom': snat_ip, 'addressTo': snat_ip, 'portFrom': 1, 'portTo': 65535}) post_chain.add_rule().type('snat').flow_action( 'accept').nat_targets(nat_targets).out_ports( [port_id]).properties(kwargs).position(1).create() @handle_api_error def remove_static_route(self, router, ip): """Remove static route for the IP :param router: next hop router to remove the routes to :param ip: IP address of the route to remove """ LOG.debug(_("MidoClient.remote_static_route called: " "router=%(router)s, ip=%(ip)s"), {'router': router, 'ip': ip}) for r in router.get_routes(): if (r.get_dst_network_addr() == ip and r.get_dst_network_length() == 32): self.mido_api.delete_route(r.get_id()) @handle_api_error def update_port_chains(self, port, inbound_chain_id, outbound_chain_id): """Bind inbound and outbound chains to the port.""" LOG.debug(_("MidoClient.update_port_chains called: port=%(port)s" "inbound_chain_id=%(inbound_chain_id)s, " "outbound_chain_id=%(outbound_chain_id)s"), {"port": port, "inbound_chain_id": inbound_chain_id, "outbound_chain_id": outbound_chain_id}) port.inbound_filter_id(inbound_chain_id).outbound_filter_id( outbound_chain_id).update() @handle_api_error def create_chain(self, tenant_id, name): """Create a new chain.""" LOG.debug(_("MidoClient.create_chain called: tenant_id=%(tenant_id)s " " name=%(name)s"), {"tenant_id": tenant_id, "name": name}) return self.mido_api.add_chain().tenant_id(tenant_id).name( name).create() @handle_api_error def delete_chain(self, id): """Delete chain matching the ID.""" LOG.debug(_("MidoClient.delete_chain called: id=%(id)s"), {"id": id}) self.mido_api.delete_chain(id) @handle_api_error def delete_chains_by_names(self, tenant_id, names): """Delete chains matching the names given for a tenant.""" LOG.debug(_("MidoClient.delete_chains_by_names called: " "tenant_id=%(tenant_id)s names=%(names)s "), {"tenant_id": tenant_id, "names": names}) chains = self.mido_api.get_chains({'tenant_id': tenant_id}) for c in chains: if c.get_name() in names: self.mido_api.delete_chain(c.get_id()) @handle_api_error def get_chain_by_name(self, tenant_id, name): """Get the chain by its name.""" LOG.debug(_("MidoClient.get_chain_by_name called: " "tenant_id=%(tenant_id)s name=%(name)s "), {"tenant_id": tenant_id, "name": name}) for c in self.mido_api.get_chains({'tenant_id': tenant_id}): if c.get_name() == name: return c return None @handle_api_error def get_port_group_by_name(self, tenant_id, name): """Get the port group by name.""" LOG.debug(_("MidoClient.get_port_group_by_name called: " "tenant_id=%(tenant_id)s name=%(name)s "), {"tenant_id": tenant_id, "name": name}) for p in self.mido_api.get_port_groups({'tenant_id': tenant_id}): if p.get_name() == name: return p return None @handle_api_error def create_port_group(self, tenant_id, name): """Create a port group Create a new port group for a given name and ID. """ LOG.debug(_("MidoClient.create_port_group called: " "tenant_id=%(tenant_id)s name=%(name)s"), {"tenant_id": tenant_id, "name": name}) return self.mido_api.add_port_group().tenant_id(tenant_id).name( name).create() @handle_api_error def delete_port_group_by_name(self, tenant_id, name): """Delete port group matching the name given for a tenant.""" LOG.debug(_("MidoClient.delete_port_group_by_name called: " "tenant_id=%(tenant_id)s name=%(name)s "), {"tenant_id": tenant_id, "name": name}) pgs = self.mido_api.get_port_groups({'tenant_id': tenant_id}) for pg in pgs: if pg.get_name() == name: LOG.debug(_("Deleting pg %(id)s"), {"id": pg.get_id()}) self.mido_api.delete_port_group(pg.get_id()) @handle_api_error def add_port_to_port_group_by_name(self, tenant_id, name, port_id): """Add a port to a port group with the given name.""" LOG.debug(_("MidoClient.add_port_to_port_group_by_name called: " "tenant_id=%(tenant_id)s name=%(name)s " "port_id=%(port_id)s"), {"tenant_id": tenant_id, "name": name, "port_id": port_id}) pg = self.get_port_group_by_name(tenant_id, name) if pg is None: raise MidonetResourceNotFound(resource_type='PortGroup', id=name) pg = pg.add_port_group_port().port_id(port_id).create() return pg @handle_api_error def remove_port_from_port_groups(self, port_id): """Remove a port binding from all the port groups.""" LOG.debug(_("MidoClient.remove_port_from_port_groups called: " "port_id=%(port_id)s"), {"port_id": port_id}) port = self.get_port(port_id) for pg in port.get_port_groups(): pg.delete() @handle_api_error def add_chain_rule(self, chain, action='accept', **kwargs): """Create a new accept chain rule.""" self.mido_api.add_chain_rule(chain, action, **kwargs)