Test Failed
Pull Request — master (#257)
by
unknown
03:57
created

build.utils.map_evc_event_content()   A

Complexity

Conditions 1

Size

Total Lines 9
Code Lines 8

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 5
CRAP Score 1

Importance

Changes 0
Metric Value
eloc 8
dl 0
loc 9
ccs 5
cts 5
cp 1
rs 10
c 0
b 0
f 0
cc 1
nop 2
crap 1
1
"""Utility functions."""
2 1
import functools
3 1
from pathlib import Path
4
5 1
from flask import request
6 1
from openapi_core import create_spec
7 1
from openapi_core.contrib.flask import FlaskOpenAPIRequest
8 1
from openapi_core.validation.request.validators import RequestValidator
9 1
from openapi_spec_validator import validate_spec
10 1
from openapi_spec_validator.readers import read_from_filename
11 1
from werkzeug.exceptions import BadRequest, UnsupportedMediaType
12
13 1
from kytos.core import log
14 1
from kytos.core.events import KytosEvent
15
16
17 1
def map_evc_event_content(evc, **kwargs):
18
    """Returns a set of values from evc to be used for content"""
19 1
    return kwargs | {"evc_id": evc.id,
20 1
            "name": evc.name,
21 1
            "metadata": evc.metadata,
22
            "active": evc._active,
23
            "enabled": evc._enabled,
24 1
            "uni_a": evc.uni_a.as_dict(),
25
            "uni_z": evc.uni_z.as_dict()}
26 1
27
28
def emit_event(controller, name, context="kytos/mef_eline", content=None):
29
    """Send an event when something happens with an EVC."""
30
    event_name = f"{context}.{name}"
31
    event = KytosEvent(name=event_name, content=content)
32 1
    controller.buffers.app.put(event)
33
34 1
35 1
def notify_link_available_tags(controller, link, src_func=None):
36
    """Notify link available tags."""
37
    emit_event(controller, "link_available_tags", content={
38
        "link": link,
39
        "src_func": src_func
40 1
    })
41
42
43
def compare_endpoint_trace(endpoint, vlan, trace):
44
    """Compare and endpoint with a trace step."""
45
    if vlan and "vlan" in trace:
46 1
        return (
47
            endpoint.switch.dpid == trace["dpid"]
48
            and endpoint.port_number == trace["port"]
49 1
            and vlan == trace["vlan"]
50 1
        )
51 1
    return (
52 1
        endpoint.switch.dpid == trace["dpid"]
53 1
        and endpoint.port_number == trace["port"]
54 1
    )
55
56
57
def compare_uni_out_trace(uni, trace):
58
    """Check if the trace last step (output) matches the UNI attributes."""
59
    # keep compatibility for old versions of sdntrace-cp
60 1
    if "out" not in trace:
61
        return True
62 1
    if not isinstance(trace["out"], dict):
63 1
        return False
64 1
    uni_vlan = uni.user_tag.value if uni.user_tag else None
65 1
    return (
66 1
        uni.interface.port_number == trace["out"].get("port")
67 1
        and uni_vlan == trace["out"].get("vlan")
68
    )
69
70 1
71
def uni_to_str(uni):
72 1
    """Create a string representation of the uni: intf_id:portno[:vlan]."""
73 1
    dpid = uni.interface.switch.dpid
74 1
    port = uni.interface.port_number
75
    uni_str = str(dpid) + ':' + str(port)
76 1
    if uni.user_tag:
77
        uni_str += ':' + str(uni.user_tag.value)
78 1
    return uni_str
79
80
81 1
def load_spec():
82
    """Validate openapi spec."""
83
    napp_dir = Path(__file__).parent
84
    yml_file = napp_dir / "openapi.yml"
85
    spec_dict, _ = read_from_filename(yml_file)
86
87
    validate_spec(spec_dict)
88 1
89 1
    return create_spec(spec_dict)
90 1
91 1
92 1
def validate(spec):
93 1
    """Decorator to validate a REST endpoint input.
94 1
95 1
    Uses the schema defined in the openapi.yml file
96 1
    to validate.
97 1
    """
98 1
99 1
    def validate_decorator(func):
100 1
        @functools.wraps(func)
101
        def wrapper_validate(*args, **kwargs):
102 1
            try:
103 1
                data = request.get_json()
104 1
            except BadRequest:
105 1
                result = "The request body is not a well-formed JSON."
106 1
                log.debug("create_circuit result %s %s", result, 400)
107
                raise BadRequest(result) from BadRequest
108
            if data is None:
109 1
                result = "The request body mimetype is not application/json."
110 1
                log.debug("update result %s %s", result, 415)
111 1
                raise UnsupportedMediaType(result)
112 1
113
            validator = RequestValidator(spec)
114
            openapi_request = FlaskOpenAPIRequest(request)
115
            result = validator.validate(openapi_request)
116
            if result.errors:
117
                error_response = (
118
                    "The request body contains invalid API data."
119
                )
120 1
                errors = result.errors[0]
121 1
                if hasattr(errors, "schema_errors"):
122 1
                    schema_errors = errors.schema_errors[0]
123
                    error_log = {
124
                        "error_message": schema_errors.message,
125 1
                        "error_validator": schema_errors.validator,
126 1
                        "error_validator_value": schema_errors.validator_value,
127
                        "error_path": list(schema_errors.path),
128 1
                        "error_schema": schema_errors.schema,
129
                        "error_schema_path": list(schema_errors.schema_path),
130 1
                    }
131
                    log.debug("Invalid request (API schema): %s", error_log)
132
                    error_response += f" {schema_errors.message} for field"
133
                    error_response += (
134
                        f" {'/'.join(map(str,schema_errors.path))}."
135
                    )
136
                raise BadRequest(error_response) from BadRequest
137
            return func(*args, data=data, **kwargs)
138
139
        return wrapper_validate
140
141
    return validate_decorator
142