1 # -*- coding: utf-8 -*-
3 from jsonschema
import validate
as js_v
, exceptions
as js_e
4 from http
import HTTPStatus
6 __author__
= "Alfonso Tierno <alfonso.tiernosepulveda@telefonica.com>"
8 version_date
= "Mar 2018"
11 Validator of input data using JSON schemas for those items that not contains an OSM yang information model
15 patern_name
= "^[ -~]+$"
16 nameshort_schema
= {"type": "string", "minLength": 1, "maxLength": 60, "pattern": "^[^,;()\\.\\$'\"]+$"}
17 passwd_schema
= {"type": "string", "minLength": 1, "maxLength": 60}
18 name_schema
= {"type": "string", "minLength": 1, "maxLength": 255, "pattern": "^[^,;()'\"]+$"}
19 string_schema
= {"type": "string", "minLength": 1, "maxLength": 255}
20 xml_text_schema
= {"type": "string", "minLength": 1, "maxLength": 1000, "pattern": "^[^']+$"}
21 description_schema
= {"type": ["string", "null"], "maxLength": 255, "pattern": "^[^'\"]+$"}
22 id_schema_fake
= {"type": "string", "minLength": 2, "maxLength": 36}
23 bool_schema
= {"type": "boolean"}
24 null_schema
= {"type": "null"}
25 # "pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$"
26 id_schema
= {"type": "string", "pattern": "^[a-fA-F0-9]{8}(-[a-fA-F0-9]{4}){3}-[a-fA-F0-9]{12}$"}
27 time_schema
= {"type": "string", "pattern": "^[0-9]{4}-[0-1][0-9]-[0-3][0-9]T[0-2][0-9]([0-5]:){2}"}
28 pci_schema
= {"type": "string", "pattern": "^[0-9a-fA-F]{4}(:[0-9a-fA-F]{2}){2}\\.[0-9a-fA-F]$"}
29 # allows [] for wildcards. For that reason huge length limit is set
30 pci_extended_schema
= {"type": "string", "pattern": "^[0-9a-fA-F.:-\\[\\]]{12,40}$"}
31 http_schema
= {"type": "string", "pattern": "^https?://[^'\"=]+$"}
32 bandwidth_schema
= {"type": "string", "pattern": "^[0-9]+ *([MG]bps)?$"}
33 memory_schema
= {"type": "string", "pattern": "^[0-9]+ *([MG]i?[Bb])?$"}
34 integer0_schema
= {"type": "integer", "minimum": 0}
35 integer1_schema
= {"type": "integer", "minimum": 1}
36 path_schema
= {"type": "string", "pattern": "^(\\.){0,2}(/[^/\"':{}\\(\\)]+)+$"}
37 vlan_schema
= {"type": "integer", "minimum": 1, "maximum": 4095}
38 vlan1000_schema
= {"type": "integer", "minimum": 1000, "maximum": 4095}
39 mac_schema
= {"type": "string",
40 "pattern": "^[0-9a-fA-F][02468aceACE](:[0-9a-fA-F]{2}){5}$"} # must be unicast: LSB bit of MSB byte ==0
41 dpid_Schema
= {"type": "string", "pattern": "^[0-9a-fA-F]{2}(:[0-9a-fA-F]{2}){7}$"}
42 # mac_schema={"type":"string", "pattern":"^([0-9a-fA-F]{2}:){5}[0-9a-fA-F]{2}$"}
43 ip_schema
= {"type": "string",
44 "pattern": "^((25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$"}
45 ip_prefix_schema
= {"type": "string",
46 "pattern": "^((25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}"
47 "(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)/(30|[12]?[0-9])$"}
48 port_schema
= {"type": "integer", "minimum": 1, "maximum": 65534}
49 object_schema
= {"type": "object"}
50 schema_version_2
= {"type": "integer", "minimum": 2, "maximum": 2}
51 # schema_version_string={"type":"string","enum": ["0.1", "2", "0.2", "3", "0.3"]}
52 log_level_schema
= {"type": "string", "enum": ["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"]}
53 checksum_schema
= {"type": "string", "pattern": "^[0-9a-fA-F]{32}$"}
54 size_schema
= {"type": "integer", "minimum": 1, "maximum": 100}
55 array_edition_schema
= {
57 "patternProperties": {
60 "additionalProperties": False,
63 nameshort_list_schema
= {
66 "items": nameshort_schema
,
70 ns_instantiate_vdu
= {
71 "title": "ns action instantiate input schema for vdu",
72 "$schema": "http://json-schema.org/draft-04/schema#",
83 "vim-volume-id": name_schema
,
85 "required": ["name", "vim-volume-id"],
86 "additionalProperties": False
96 "ip-address": ip_schema
,
97 "mac-address": mac_schema
,
98 "floating-ip-required": bool_schema
,
100 "required": ["name"],
101 "additionalProperties": False
106 "additionalProperties": False
109 ip_profile_dns_schema
= {
115 "address": ip_schema
,
117 "required": ["address"],
118 "additionalProperties": False
122 ip_profile_dhcp_schema
= {
125 "enabled": {"type": "boolean"},
126 "count": integer1_schema
,
127 "start-address": ip_schema
129 "additionalProperties": False,
132 ip_profile_schema
= {
133 "title": "ip profile validation schame",
134 "$schema": "http://json-schema.org/draft-04/schema#",
137 "ip-version": {"enum": ["ipv4", "ipv6"]},
138 "subnet-address": ip_prefix_schema
,
139 "gateway-address": ip_schema
,
140 "dns-server": ip_profile_dns_schema
,
141 "dhcp-params": ip_profile_dhcp_schema
,
145 ip_profile_update_schema
= {
146 "title": "ip profile validation schame",
147 "$schema": "http://json-schema.org/draft-04/schema#",
150 "ip-version": {"enum": ["ipv4", "ipv6"]},
151 "subnet-address": {"oneOf": [null_schema
, ip_prefix_schema
]},
152 "gateway-address": {"oneOf": [null_schema
, ip_schema
]},
153 "dns-server": {"oneOf": [null_schema
, ip_profile_dns_schema
]},
155 "dhcp-params": {"oneOf": [null_schema
, ip_profile_dhcp_schema
]},
157 "additionalProperties": False
160 ns_instantiate_internal_vld
= {
161 "title": "ns action instantiate input schema for vdu",
162 "$schema": "http://json-schema.org/draft-04/schema#",
166 "vim-network-name": name_schema
,
167 "ip-profile": ip_profile_update_schema
,
168 "internal-connection-point": {
174 "id-ref": name_schema
,
175 "ip-address": ip_schema
,
176 # "mac-address": mac_schema,
178 "required": ["id-ref"],
180 "additionalProperties": False
184 "required": ["name"],
186 "additionalProperties": False
190 "title": "ns action instantiate input schema",
191 "$schema": "http://json-schema.org/draft-04/schema#",
194 "lcmOperationType": string_schema
,
195 "nsInstanceId": id_schema
,
196 "nsName": name_schema
,
197 "nsDescription": {"oneOf": [description_schema
, {"type": "null"}]},
199 "vimAccountId": id_schema
,
200 "ssh_keys": {"type": "array", "items": {"type": "string"}},
202 "vduImage": name_schema
,
209 "member-vnf-index": name_schema
,
210 "vimAccountId": id_schema
,
214 "items": ns_instantiate_vdu
,
219 "items": ns_instantiate_internal_vld
222 "required": ["member-vnf-index"],
224 "additionalProperties": False
233 "name": string_schema
,
234 "vim-network-name": {"OneOf": [string_schema
, object_schema
]},
235 "ip-profile": object_schema
,
236 "vnfd-connection-point-ref": {
242 "member-vnf-index-ref": name_schema
,
243 "vnfd-connection-point-ref": name_schema
,
244 "ip-address": ip_schema
,
245 # "mac-address": mac_schema,
247 "required": ["member-vnf-index-ref", "vnfd-connection-point-ref"],
249 "additionalProperties": False
253 "required": ["name"],
254 "additionalProperties": False
258 "required": ["nsName", "nsdId", "vimAccountId"],
259 "additionalProperties": False
262 ns_action
= { # TODO for the moment it is only contemplated the vnfd primitive execution
263 "title": "ns action input schema",
264 "$schema": "http://json-schema.org/draft-04/schema#",
267 "lcmOperationType": string_schema
,
268 "nsInstanceId": id_schema
,
269 "member_vnf_index": name_schema
,
270 "vnf_member_index": name_schema
, # TODO for backward compatibility. To remove in future
271 "vdu_id": name_schema
,
272 "primitive": name_schema
,
273 "primitive_params": {"type": "object"},
275 "required": ["primitive", "primitive_params"], # TODO add member_vnf_index
276 "additionalProperties": False
278 ns_scale
= { # TODO for the moment it is only VDU-scaling
279 "title": "ns scale input schema",
280 "$schema": "http://json-schema.org/draft-04/schema#",
283 "lcmOperationType": string_schema
,
284 "nsInstanceId": id_schema
,
285 "scaleType": {"enum": ["SCALE_VNF"]},
289 "vnfInstanceId": name_schema
,
290 "scaleVnfType": {"enum": ["SCALE_OUT", 'SCALE_IN']},
294 "scaling-group-descriptor": name_schema
,
295 "member-vnf-index": name_schema
,
296 "scaling-policy": name_schema
,
298 "required": ["scaling-group-descriptor", "member-vnf-index"],
299 "additionalProperties": False
302 "required": ["scaleVnfType", "scaleByStepData"], # vnfInstanceId
303 "additionalProperties": False
305 "scaleTime": time_schema
,
307 "required": ["scaleType", "scaleVnfData"],
308 "additionalProperties": False
312 schema_version
= {"type": "string", "enum": ["1.0"]}
313 vim_account_edit_schema
= {
314 "title": "vim_account edit input schema",
315 "$schema": "http://json-schema.org/draft-04/schema#",
319 "description": description_schema
,
320 "type": nameshort_schema
, # currently "openvim" or "openstack", can be enlarged with plugins
322 "datacenter": name_schema
,
323 "vim_url": description_schema
,
324 "vim_url_admin": description_schema
,
325 "vim_tenant": name_schema
,
326 "vim_tenant_name": name_schema
,
327 "vim_username": nameshort_schema
,
328 "vim_password": passwd_schema
,
329 "config": {"type": "object"}
331 "additionalProperties": False
333 schema_type
= {"type": "string"}
335 vim_account_new_schema
= {
336 "title": "vim_account creation input schema",
337 "$schema": "http://json-schema.org/draft-04/schema#",
340 "schema_version": schema_version
,
341 "schema_type": schema_type
,
343 "description": description_schema
,
345 "datacenter": name_schema
,
346 "vim_type": {"enum": ["openstack", "openvim", "vmware", "opennebula", "aws"]},
347 "vim_url": description_schema
,
348 # "vim_url_admin": description_schema,
349 # "vim_tenant": name_schema,
350 "vim_tenant_name": name_schema
,
351 "vim_user": nameshort_schema
,
352 "vim_password": passwd_schema
,
353 "config": {"type": "object"}
355 "required": ["name", "vim_url", "vim_type", "vim_user", "vim_password", "vim_tenant_name"],
356 "additionalProperties": False
362 "description": description_schema
,
366 "type": {"type": "string", "enum": ["opendaylight", "floodlight", "onos"]},
367 "version": {"type": "string", "minLength": 1, "maxLength": 12},
368 "user": nameshort_schema
,
369 "password": passwd_schema
372 "title": "sdn controller information schema",
373 "$schema": "http://json-schema.org/draft-04/schema#",
375 "properties": sdn_properties
,
376 "required": ["name", "port", 'ip', 'dpid', 'type'],
377 "additionalProperties": False
380 "title": "sdn controller update information schema",
381 "$schema": "http://json-schema.org/draft-04/schema#",
383 "properties": sdn_properties
,
384 # "required": ["name", "port", 'ip', 'dpid', 'type'],
385 "additionalProperties": False
387 sdn_port_mapping_schema
= {
388 "$schema": "http://json-schema.org/draft-04/schema#",
389 "title": "sdn port mapping information schema",
394 "compute_node": nameshort_schema
,
400 "pci": pci_extended_schema
,
401 "switch_port": nameshort_schema
,
402 "switch_mac": mac_schema
408 "required": ["compute_node", "ports"]
411 sdn_external_port_schema
= {
412 "$schema": "http://json-schema.org/draft-04/schema#",
413 "title": "External port information",
416 "port": {"type": "string", "minLength": 1, "maxLength": 60},
427 "name": nameshort_schema
,
429 "type": {"enum": ["overlay", 'underlay']},
430 "ip-address": ip_schema
,
431 # TODO, add user, password, ssh-key
432 "mac-address": mac_schema
,
433 "vim-network-name": nameshort_schema
, # interface is connected to one vim network, or switch port
434 # TODO "vim-network-id": nameshort_schema,
435 # # provide this in case SDN assist must deal with this interface
436 # "switch-dpid": dpid_Schema,
437 # "switch-port": nameshort_schema,
438 # "switch-mac": nameshort_schema,
439 # "switch-vlan": vlan_schema,
441 "required": ["name", "mgmt", "ip-address"],
442 "additionalProperties": False
445 "title": "pdu creation input schema",
446 "$schema": "http://json-schema.org/draft-04/schema#",
449 "name": nameshort_schema
,
450 "type": nameshort_schema
,
451 "description": description_schema
,
452 "shared": bool_schema
,
453 "vims": nameshort_list_schema
,
454 "vim_accounts": nameshort_list_schema
,
457 "items": pdu_interface
,
461 "required": ["name", "type", "interfaces"],
462 "additionalProperties": False
466 "title": "pdu edit input schema",
467 "$schema": "http://json-schema.org/draft-04/schema#",
470 "name": nameshort_schema
,
471 "type": nameshort_schema
,
472 "description": description_schema
,
473 "shared": bool_schema
,
474 "vims": {"oneOff": [array_edition_schema
, nameshort_list_schema
]},
475 "vim_accounts": {"oneOff": [array_edition_schema
, nameshort_list_schema
]},
476 "interfaces": {"oneOff": [
477 array_edition_schema
,
480 "items": pdu_interface
,
485 "additionalProperties": False,
491 "$schema": "http://json-schema.org/draft-04/schema#",
492 "title": "New user schema",
495 "username": nameshort_schema
,
496 "password": passwd_schema
,
497 "projects": nameshort_list_schema
,
499 "required": ["username", "password", "projects"],
500 "additionalProperties": False
503 "$schema": "http://json-schema.org/draft-04/schema#",
504 "title": "User edit schema for administrators",
507 "password": passwd_schema
,
510 nameshort_list_schema
,
516 "additionalProperties": False
520 project_new_schema
= {
521 "$schema": "http://json-schema.org/draft-04/schema#",
522 "title": "New project schema for administrators",
525 "name": nameshort_schema
,
526 "admin": bool_schema
,
528 "required": ["name"],
529 "additionalProperties": False
531 project_edit_schema
= {
532 "$schema": "http://json-schema.org/draft-04/schema#",
533 "title": "Project edit schema for administrators",
536 "admin": bool_schema
,
538 "additionalProperties": False,
544 nbi_new_input_schemas
= {
545 "users": user_new_schema
,
546 "projects": project_new_schema
,
547 "vim_accounts": vim_account_new_schema
,
548 "sdns": sdn_new_schema
,
549 "ns_instantiate": ns_instantiate
,
550 "ns_action": ns_action
,
551 "ns_scale": ns_scale
,
552 "pdus": pdu_new_schema
,
555 nbi_edit_input_schemas
= {
556 "users": user_edit_schema
,
557 "projects": project_edit_schema
,
558 "vim_accounts": vim_account_edit_schema
,
559 "sdns": sdn_edit_schema
,
560 "pdus": pdu_edit_schema
,
564 class ValidationError(Exception):
565 def __init__(self
, message
, http_code
=HTTPStatus
.UNPROCESSABLE_ENTITY
):
566 self
.http_code
= http_code
567 Exception.__init
__(self
, message
)
570 def validate_input(indata
, schema_to_use
):
572 Validates input data against json schema
573 :param indata: user input data. Should be a dictionary
574 :param schema_to_use: jsonschema to test
575 :return: None if ok, raises ValidationError exception on error
579 js_v(indata
, schema_to_use
)
581 except js_e
.ValidationError
as e
:
583 error_pos
= "at '" + ":".join(map(str, e
.path
)) + "'"
586 raise ValidationError("Format error {} '{}' ".format(error_pos
, e
.message
))
587 except js_e
.SchemaError
:
588 raise ValidationError("Bad json schema {}".format(schema_to_use
), http_code
=HTTPStatus
.INTERNAL_SERVER_ERROR
)