Statistics
| Branch: | Tag: | Revision:

root / snf-cyclades-app / synnefo / logic / subnets.py @ 7b438672

History | View | Annotate | Download (10.6 kB)

1 9eb0710b Dionysis Grigoropoulos
# Copyright 2013 GRNET S.A. All rights reserved.
2 9eb0710b Dionysis Grigoropoulos
#
3 9eb0710b Dionysis Grigoropoulos
# Redistribution and use in source and binary forms, with or
4 9eb0710b Dionysis Grigoropoulos
# without modification, are permitted provided that the following
5 9eb0710b Dionysis Grigoropoulos
# conditions are met:
6 9eb0710b Dionysis Grigoropoulos
#
7 9eb0710b Dionysis Grigoropoulos
#   1. Redistributions of source code must retain the above
8 9eb0710b Dionysis Grigoropoulos
#      copyright notice, this list of conditions and the following
9 9eb0710b Dionysis Grigoropoulos
#      disclaimer.
10 9eb0710b Dionysis Grigoropoulos
#
11 9eb0710b Dionysis Grigoropoulos
#   2. Redistributions in binary form must reproduce the above
12 9eb0710b Dionysis Grigoropoulos
#      copyright notice, this list of conditions and the following
13 9eb0710b Dionysis Grigoropoulos
#      disclaimer in the documentation and/or other materials
14 9eb0710b Dionysis Grigoropoulos
#      provided with the distribution.
15 9eb0710b Dionysis Grigoropoulos
#
16 9eb0710b Dionysis Grigoropoulos
# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
17 9eb0710b Dionysis Grigoropoulos
# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18 9eb0710b Dionysis Grigoropoulos
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19 9eb0710b Dionysis Grigoropoulos
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
20 9eb0710b Dionysis Grigoropoulos
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21 9eb0710b Dionysis Grigoropoulos
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 9eb0710b Dionysis Grigoropoulos
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
23 9eb0710b Dionysis Grigoropoulos
# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
24 9eb0710b Dionysis Grigoropoulos
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 9eb0710b Dionysis Grigoropoulos
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
26 9eb0710b Dionysis Grigoropoulos
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 9eb0710b Dionysis Grigoropoulos
# POSSIBILITY OF SUCH DAMAGE.
28 9eb0710b Dionysis Grigoropoulos
#
29 9eb0710b Dionysis Grigoropoulos
# The views and conclusions contained in the software and
30 9eb0710b Dionysis Grigoropoulos
# documentation are those of the authors and should not be
31 9eb0710b Dionysis Grigoropoulos
# interpreted as representing official policies, either expressed
32 9eb0710b Dionysis Grigoropoulos
# or implied, of GRNET S.A.
33 9eb0710b Dionysis Grigoropoulos
34 f82dfec6 Christos Stavrakakis
import ipaddr
35 9eb0710b Dionysis Grigoropoulos
from logging import getLogger
36 f82dfec6 Christos Stavrakakis
from functools import wraps
37 f82dfec6 Christos Stavrakakis
38 f82dfec6 Christos Stavrakakis
from django.conf import settings
39 f82dfec6 Christos Stavrakakis
from django.db import transaction
40 3549cb2f Dionysis Grigoropoulos
from django.db.models import Q
41 3549cb2f Dionysis Grigoropoulos
42 9eb0710b Dionysis Grigoropoulos
from snf_django.lib import api
43 9eb0710b Dionysis Grigoropoulos
from snf_django.lib.api import faults
44 9eb0710b Dionysis Grigoropoulos
45 9eb0710b Dionysis Grigoropoulos
from synnefo.db.models import Subnet, Network, IPPoolTable
46 9eb0710b Dionysis Grigoropoulos
47 9eb0710b Dionysis Grigoropoulos
log = getLogger(__name__)
48 9eb0710b Dionysis Grigoropoulos
49 9eb0710b Dionysis Grigoropoulos
50 9eb0710b Dionysis Grigoropoulos
def subnet_command(action):
51 9eb0710b Dionysis Grigoropoulos
    def decorator(func):
52 9eb0710b Dionysis Grigoropoulos
        @wraps(func)
53 9eb0710b Dionysis Grigoropoulos
        @transaction.commit_on_success()
54 9eb0710b Dionysis Grigoropoulos
        def wrapper(subnet, *args, **kwargs):
55 9eb0710b Dionysis Grigoropoulos
            return func(subnet, *args, **kwargs)
56 9eb0710b Dionysis Grigoropoulos
        return wrapper
57 9eb0710b Dionysis Grigoropoulos
    return decorator
58 9eb0710b Dionysis Grigoropoulos
59 9eb0710b Dionysis Grigoropoulos
60 9eb0710b Dionysis Grigoropoulos
def list_subnets(user_id):
61 9eb0710b Dionysis Grigoropoulos
    """List all subnets of a user"""
62 3549cb2f Dionysis Grigoropoulos
    log.debug('list_subnets %s', user_id)
63 9eb0710b Dionysis Grigoropoulos
64 91430ce0 Dionysis Grigoropoulos
    query = (((Q(network__userid=user_id) & Q(network__public=False)) |
65 91430ce0 Dionysis Grigoropoulos
              Q(network__public=True)) & Q(deleted=False))
66 91430ce0 Dionysis Grigoropoulos
    user_subnets = Subnet.objects.filter(query)
67 5d83d2ff Dionysis Grigoropoulos
    return user_subnets
68 9eb0710b Dionysis Grigoropoulos
69 9eb0710b Dionysis Grigoropoulos
70 9eb0710b Dionysis Grigoropoulos
@transaction.commit_on_success
71 f82dfec6 Christos Stavrakakis
def create_subnet(*args, **kwargs):
72 f82dfec6 Christos Stavrakakis
    return _create_subnet(*args, **kwargs)
73 f82dfec6 Christos Stavrakakis
74 f82dfec6 Christos Stavrakakis
75 f82dfec6 Christos Stavrakakis
def _create_subnet(network_id, user_id, cidr, name, ipversion=4, gateway=None,
76 b7311f3d Dionysis Grigoropoulos
                   dhcp=True, slaac=True, dns_nameservers=None,
77 f82dfec6 Christos Stavrakakis
                   allocation_pools=None, host_routes=None):
78 316787ab Dionysis Grigoropoulos
    """Create a subnet
79 f82dfec6 Christos Stavrakakis

80 9eb0710b Dionysis Grigoropoulos
    network_id and the desired cidr are mandatory, everything else is optional
81 316787ab Dionysis Grigoropoulos

82 9eb0710b Dionysis Grigoropoulos
    """
83 9eb0710b Dionysis Grigoropoulos
    try:
84 9eb0710b Dionysis Grigoropoulos
        network = Network.objects.get(id=network_id)
85 9eb0710b Dionysis Grigoropoulos
    except Network.DoesNotExist:
86 b7311f3d Dionysis Grigoropoulos
        raise api.faults.ItemNotFound("No network found with that id")
87 9eb0710b Dionysis Grigoropoulos
88 5d83d2ff Dionysis Grigoropoulos
    if user_id != network.userid:
89 5d83d2ff Dionysis Grigoropoulos
        raise api.faults.Unauthorized("Unauthorized operation")
90 5d83d2ff Dionysis Grigoropoulos
91 9eb0710b Dionysis Grigoropoulos
    if ipversion not in [4, 6]:
92 9eb0710b Dionysis Grigoropoulos
        raise api.faults.BadRequest("Malformed IP version type")
93 9eb0710b Dionysis Grigoropoulos
94 99718617 Dionysis Grigoropoulos
    check_number_of_subnets(network, ipversion)
95 99718617 Dionysis Grigoropoulos
96 1709a768 Dionysis Grigoropoulos
    if network.backend_networks.exists():
97 1709a768 Dionysis Grigoropoulos
        raise api.faults.BadRequest("Cannot create subnet in network %s, VMs"
98 1709a768 Dionysis Grigoropoulos
                                    " are already connected to this network" %
99 1709a768 Dionysis Grigoropoulos
                                    network_id)
100 1709a768 Dionysis Grigoropoulos
101 99718617 Dionysis Grigoropoulos
    try:
102 f82dfec6 Christos Stavrakakis
        cidr_ip = ipaddr.IPNetwork(cidr)
103 99718617 Dionysis Grigoropoulos
    except ValueError:
104 99718617 Dionysis Grigoropoulos
        raise api.faults.BadRequest("Malformed CIDR")
105 5d83d2ff Dionysis Grigoropoulos
106 9eb0710b Dionysis Grigoropoulos
    if ipversion == 6:
107 f23fbacf Christos Stavrakakis
        validate_subnet_params(subnet6=cidr, gateway6=gateway)
108 9eb0710b Dionysis Grigoropoulos
    else:
109 f23fbacf Christos Stavrakakis
        validate_subnet_params(subnet=cidr, gateway=gateway)
110 9eb0710b Dionysis Grigoropoulos
111 9eb0710b Dionysis Grigoropoulos
    name = check_name_length(name)
112 9eb0710b Dionysis Grigoropoulos
    sub = Subnet.objects.create(name=name, network=network, cidr=cidr,
113 9eb0710b Dionysis Grigoropoulos
                                ipversion=ipversion, gateway=gateway,
114 5d83d2ff Dionysis Grigoropoulos
                                dhcp=dhcp, host_routes=host_routes,
115 5d83d2ff Dionysis Grigoropoulos
                                dns_nameservers=dns_nameservers)
116 9eb0710b Dionysis Grigoropoulos
117 b7311f3d Dionysis Grigoropoulos
    gateway_ip = ipaddr.IPAddress(gateway) if gateway else None
118 b7311f3d Dionysis Grigoropoulos
119 9eb0710b Dionysis Grigoropoulos
    if allocation_pools is not None:
120 9eb0710b Dionysis Grigoropoulos
        if ipversion == 6:
121 9eb0710b Dionysis Grigoropoulos
            raise api.faults.Conflict("Can't allocate an IP Pool in IPv6")
122 f23fbacf Christos Stavrakakis
    elif ipversion == 4:
123 eb5f7eb0 Dionysis Grigoropoulos
        # Check if the gateway is the first IP of the subnet, or the last. In
124 eb5f7eb0 Dionysis Grigoropoulos
        # that case create a single ip pool.
125 b7311f3d Dionysis Grigoropoulos
        if gateway_ip:
126 b7311f3d Dionysis Grigoropoulos
            if int(gateway_ip) - int(cidr_ip) == 1:
127 f23fbacf Christos Stavrakakis
                allocation_pools = [(gateway_ip + 1, cidr_ip.broadcast - 1)]
128 eb5f7eb0 Dionysis Grigoropoulos
            elif int(cidr_ip.broadcast) - int(gateway_ip) == 1:
129 eb5f7eb0 Dionysis Grigoropoulos
                allocation_pools = [(cidr_ip.network + 1, gateway_ip - 1)]
130 b7311f3d Dionysis Grigoropoulos
            else:
131 b7311f3d Dionysis Grigoropoulos
                # If the gateway isn't the first available ip, create two
132 b7311f3d Dionysis Grigoropoulos
                # different ip pools adjacent to said ip
133 f23fbacf Christos Stavrakakis
                allocation_pools = [(cidr_ip.network + 1, gateway_ip - 1),
134 f23fbacf Christos Stavrakakis
                                    (gateway_ip + 1, cidr_ip.broadcast - 1)]
135 9eb0710b Dionysis Grigoropoulos
        else:
136 f23fbacf Christos Stavrakakis
            allocation_pools = [(cidr_ip.network + 1, cidr_ip.broadcast - 1)]
137 9eb0710b Dionysis Grigoropoulos
138 5d83d2ff Dionysis Grigoropoulos
    if allocation_pools:
139 f23fbacf Christos Stavrakakis
        # Validate the allocation pools
140 f23fbacf Christos Stavrakakis
        validate_pools(allocation_pools, cidr_ip, gateway_ip)
141 5d83d2ff Dionysis Grigoropoulos
        create_ip_pools(allocation_pools, cidr_ip, sub)
142 9eb0710b Dionysis Grigoropoulos
143 9eb0710b Dionysis Grigoropoulos
    return sub
144 9eb0710b Dionysis Grigoropoulos
145 9eb0710b Dionysis Grigoropoulos
146 9eb0710b Dionysis Grigoropoulos
def get_subnet(sub_id):
147 9eb0710b Dionysis Grigoropoulos
    """Show info of a specific subnet"""
148 9eb0710b Dionysis Grigoropoulos
    log.debug('get_subnet %s', sub_id)
149 9eb0710b Dionysis Grigoropoulos
    try:
150 9eb0710b Dionysis Grigoropoulos
        subnet = Subnet.objects.get(id=sub_id)
151 9eb0710b Dionysis Grigoropoulos
    except Subnet.DoesNotExist:
152 9eb0710b Dionysis Grigoropoulos
        raise api.faults.ItemNotFound("Subnet not found")
153 9eb0710b Dionysis Grigoropoulos
154 5d83d2ff Dionysis Grigoropoulos
    return subnet
155 9eb0710b Dionysis Grigoropoulos
156 9eb0710b Dionysis Grigoropoulos
157 9eb0710b Dionysis Grigoropoulos
def delete_subnet():
158 316787ab Dionysis Grigoropoulos
    """Delete a subnet, raises BadRequest
159 9eb0710b Dionysis Grigoropoulos
    A subnet is deleted ONLY when the network that it belongs to is deleted
160 316787ab Dionysis Grigoropoulos

161 9eb0710b Dionysis Grigoropoulos
    """
162 9eb0710b Dionysis Grigoropoulos
    raise api.faults.BadRequest("Deletion of a subnet is not supported")
163 9eb0710b Dionysis Grigoropoulos
164 9eb0710b Dionysis Grigoropoulos
165 9eb0710b Dionysis Grigoropoulos
@transaction.commit_on_success
166 8646e606 Dionysis Grigoropoulos
def update_subnet(sub_id, name, user_id):
167 316787ab Dionysis Grigoropoulos
    """Update the fields of a subnet
168 9eb0710b Dionysis Grigoropoulos
    Only the name can be updated
169 316787ab Dionysis Grigoropoulos

170 9eb0710b Dionysis Grigoropoulos
    """
171 5d83d2ff Dionysis Grigoropoulos
    log.info('Update subnet %s, name %s' % (sub_id, name))
172 9eb0710b Dionysis Grigoropoulos
173 9eb0710b Dionysis Grigoropoulos
    try:
174 9eb0710b Dionysis Grigoropoulos
        subnet = Subnet.objects.get(id=sub_id)
175 9eb0710b Dionysis Grigoropoulos
    except:
176 9eb0710b Dionysis Grigoropoulos
        raise api.faults.ItemNotFound("Subnet not found")
177 9eb0710b Dionysis Grigoropoulos
178 8646e606 Dionysis Grigoropoulos
    if user_id != subnet.network.userid:
179 8646e606 Dionysis Grigoropoulos
        raise api.faults.Unauthorized("Unauthorized operation")
180 8646e606 Dionysis Grigoropoulos
181 9eb0710b Dionysis Grigoropoulos
    check_name_length(name)
182 9eb0710b Dionysis Grigoropoulos
183 9eb0710b Dionysis Grigoropoulos
    subnet.name = name
184 9eb0710b Dionysis Grigoropoulos
    subnet.save()
185 9eb0710b Dionysis Grigoropoulos
186 9eb0710b Dionysis Grigoropoulos
    return subnet
187 9eb0710b Dionysis Grigoropoulos
188 9eb0710b Dionysis Grigoropoulos
189 9eb0710b Dionysis Grigoropoulos
#Utility functions
190 9eb0710b Dionysis Grigoropoulos
def create_ip_pools(pools, cidr, subnet):
191 9eb0710b Dionysis Grigoropoulos
    """Create IP Pools in the database"""
192 f23fbacf Christos Stavrakakis
    return [_create_ip_pool(pool, cidr, subnet) for pool in pools]
193 f23fbacf Christos Stavrakakis
194 f23fbacf Christos Stavrakakis
195 f23fbacf Christos Stavrakakis
def _create_ip_pool(pool, cidr, subnet):
196 f23fbacf Christos Stavrakakis
    size = int(pool[1]) - int(pool[0]) + 1
197 f23fbacf Christos Stavrakakis
    base = str(cidr)
198 f23fbacf Christos Stavrakakis
    offset = int(pool[0]) - int(cidr.network)
199 f23fbacf Christos Stavrakakis
    return IPPoolTable.objects.create(size=size, offset=offset,
200 f23fbacf Christos Stavrakakis
                                      base=base, subnet=subnet)
201 9eb0710b Dionysis Grigoropoulos
202 9eb0710b Dionysis Grigoropoulos
203 9eb0710b Dionysis Grigoropoulos
def check_number_of_subnets(network, version):
204 9eb0710b Dionysis Grigoropoulos
    """Check if a user can add a subnet in a network"""
205 9eb0710b Dionysis Grigoropoulos
    if network.subnets.filter(ipversion=version):
206 9eb0710b Dionysis Grigoropoulos
        raise api.faults.BadRequest("Only one subnet of IPv4/IPv6 per "
207 9eb0710b Dionysis Grigoropoulos
                                    "network is allowed")
208 9eb0710b Dionysis Grigoropoulos
209 9eb0710b Dionysis Grigoropoulos
210 9eb0710b Dionysis Grigoropoulos
def check_name_length(name):
211 9eb0710b Dionysis Grigoropoulos
    """Check if the length of a name is within acceptable value"""
212 9eb0710b Dionysis Grigoropoulos
    if len(str(name)) > Subnet.SUBNET_NAME_LENGTH:
213 9eb0710b Dionysis Grigoropoulos
        raise api.faults.BadRequest("Subnet name too long")
214 9eb0710b Dionysis Grigoropoulos
    return name
215 9eb0710b Dionysis Grigoropoulos
216 9eb0710b Dionysis Grigoropoulos
217 f23fbacf Christos Stavrakakis
def validate_pools(pool_list, cidr, gateway):
218 316787ab Dionysis Grigoropoulos
    """Validate IP Pools
219 316787ab Dionysis Grigoropoulos

220 9eb0710b Dionysis Grigoropoulos
    Validate the given IP pools are inside the cidr range
221 9eb0710b Dionysis Grigoropoulos
    Validate there are no overlaps in the given pools
222 9eb0710b Dionysis Grigoropoulos
    Finally, validate the gateway isn't in the given ip pools
223 9eb0710b Dionysis Grigoropoulos
    Input must be a list containing a sublist with start/end ranges as
224 9eb0710b Dionysis Grigoropoulos
    ipaddr.IPAddress items eg.,
225 9eb0710b Dionysis Grigoropoulos
    [[IPv4Address('192.168.42.11'), IPv4Address('192.168.42.15')],
226 9eb0710b Dionysis Grigoropoulos
     [IPv4Address('192.168.42.30'), IPv4Address('192.168.42.60')]]
227 316787ab Dionysis Grigoropoulos

228 9eb0710b Dionysis Grigoropoulos
    """
229 9eb0710b Dionysis Grigoropoulos
    if pool_list[0][0] <= cidr.network:
230 9eb0710b Dionysis Grigoropoulos
        raise api.faults.Conflict("IP Pool out of bounds")
231 9eb0710b Dionysis Grigoropoulos
    elif pool_list[-1][1] >= cidr.broadcast:
232 9eb0710b Dionysis Grigoropoulos
        raise api.faults.Conflict("IP Pool out of bounds")
233 9eb0710b Dionysis Grigoropoulos
234 9eb0710b Dionysis Grigoropoulos
    for start, end in pool_list:
235 9eb0710b Dionysis Grigoropoulos
        if start > end:
236 9eb0710b Dionysis Grigoropoulos
            raise api.faults.Conflict("Invalid IP pool range")
237 9eb0710b Dionysis Grigoropoulos
        # Raise BadRequest if gateway is inside the pool range
238 b7311f3d Dionysis Grigoropoulos
        if gateway:
239 b7311f3d Dionysis Grigoropoulos
            if not (gateway < start or gateway > end):
240 b7311f3d Dionysis Grigoropoulos
                raise api.faults.Conflict("Gateway cannot be in pool range")
241 9eb0710b Dionysis Grigoropoulos
242 eb5f7eb0 Dionysis Grigoropoulos
    # Check if there is a conflict between the IP Pool ranges
243 9eb0710b Dionysis Grigoropoulos
    end = cidr.network
244 9eb0710b Dionysis Grigoropoulos
    for pool in pool_list:
245 9eb0710b Dionysis Grigoropoulos
        if end >= pool[0]:
246 9eb0710b Dionysis Grigoropoulos
            raise api.faults.Conflict("IP Pool range conflict")
247 9eb0710b Dionysis Grigoropoulos
        end = pool[1]
248 f82dfec6 Christos Stavrakakis
249 f82dfec6 Christos Stavrakakis
250 f82dfec6 Christos Stavrakakis
def validate_subnet_params(subnet=None, gateway=None, subnet6=None,
251 f82dfec6 Christos Stavrakakis
                           gateway6=None):
252 f82dfec6 Christos Stavrakakis
    if subnet:
253 f82dfec6 Christos Stavrakakis
        try:
254 f82dfec6 Christos Stavrakakis
            # Use strict option to not all subnets with host bits set
255 f82dfec6 Christos Stavrakakis
            network = ipaddr.IPv4Network(subnet, strict=True)
256 f82dfec6 Christos Stavrakakis
        except ValueError:
257 f82dfec6 Christos Stavrakakis
            raise faults.BadRequest("Invalid network IPv4 subnet")
258 f82dfec6 Christos Stavrakakis
259 f82dfec6 Christos Stavrakakis
        # Check that network size is allowed!
260 f82dfec6 Christos Stavrakakis
        prefixlen = network.prefixlen
261 f82dfec6 Christos Stavrakakis
        if prefixlen > 29 or prefixlen <= settings.MAX_CIDR_BLOCK:
262 f82dfec6 Christos Stavrakakis
            raise faults.OverLimit(
263 f82dfec6 Christos Stavrakakis
                message="Unsupported network size",
264 f82dfec6 Christos Stavrakakis
                details="Netmask must be in range: (%s, 29]" %
265 f82dfec6 Christos Stavrakakis
                settings.MAX_CIDR_BLOCK)
266 f82dfec6 Christos Stavrakakis
        if gateway:  # Check that gateway belongs to network
267 f82dfec6 Christos Stavrakakis
            try:
268 f82dfec6 Christos Stavrakakis
                gateway = ipaddr.IPv4Address(gateway)
269 f82dfec6 Christos Stavrakakis
            except ValueError:
270 f82dfec6 Christos Stavrakakis
                raise faults.BadRequest("Invalid network IPv4 gateway")
271 f82dfec6 Christos Stavrakakis
            if not gateway in network:
272 f82dfec6 Christos Stavrakakis
                raise faults.BadRequest("Invalid network IPv4 gateway")
273 f82dfec6 Christos Stavrakakis
274 f82dfec6 Christos Stavrakakis
    if subnet6:
275 f82dfec6 Christos Stavrakakis
        try:
276 f82dfec6 Christos Stavrakakis
            # Use strict option to not all subnets with host bits set
277 f82dfec6 Christos Stavrakakis
            network6 = ipaddr.IPv6Network(subnet6, strict=True)
278 f82dfec6 Christos Stavrakakis
        except ValueError:
279 f82dfec6 Christos Stavrakakis
            raise faults.BadRequest("Invalid network IPv6 subnet")
280 f82dfec6 Christos Stavrakakis
        # Check that network6 is an /64 subnet, because this is imposed by
281 f82dfec6 Christos Stavrakakis
        # 'mac2eui64' utiity.
282 f82dfec6 Christos Stavrakakis
        if network6.prefixlen != 64:
283 f82dfec6 Christos Stavrakakis
            msg = ("Unsupported IPv6 subnet size. Network netmask must be"
284 f82dfec6 Christos Stavrakakis
                   " /64")
285 f82dfec6 Christos Stavrakakis
            raise faults.BadRequest(msg)
286 f82dfec6 Christos Stavrakakis
        if gateway6:
287 f82dfec6 Christos Stavrakakis
            try:
288 f82dfec6 Christos Stavrakakis
                gateway6 = ipaddr.IPv6Address(gateway6)
289 f82dfec6 Christos Stavrakakis
            except ValueError:
290 f82dfec6 Christos Stavrakakis
                raise faults.BadRequest("Invalid network IPv6 gateway")
291 f82dfec6 Christos Stavrakakis
            if not gateway6 in network6:
292 f82dfec6 Christos Stavrakakis
                raise faults.BadRequest("Invalid network IPv6 gateway")
293 0d1f9117 Dionysis Grigoropoulos
294 0d1f9117 Dionysis Grigoropoulos
295 0d1f9117 Dionysis Grigoropoulos
def parse_allocation_pools(allocation_pools):
296 0d1f9117 Dionysis Grigoropoulos
    alloc = list()
297 0d1f9117 Dionysis Grigoropoulos
    for pool in allocation_pools:
298 0d1f9117 Dionysis Grigoropoulos
        try:
299 0d1f9117 Dionysis Grigoropoulos
            start, end = pool.split(',')
300 0d1f9117 Dionysis Grigoropoulos
            alloc.append([ipaddr.IPv4Address(start),
301 0d1f9117 Dionysis Grigoropoulos
                          ipaddr.IPv4Address(end)])
302 0d1f9117 Dionysis Grigoropoulos
        except ValueError:
303 44bd008a Dionysis Grigoropoulos
            raise faults.BadRequest("Malformed IPv4 address")
304 0d1f9117 Dionysis Grigoropoulos
305 0d1f9117 Dionysis Grigoropoulos
    return alloc