summaryrefslogtreecommitdiffstats
path: root/onap-dcae-dcaepolicy-lib/onap_dcae_dcaepolicy_lib/utils.py
blob: 7f12347f7d51b7556879d18cfe82308b555136c4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
# org.onap.dcae
# ================================================================================
# Copyright (c) 2018 AT&T Intellectual Property. 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.
# ============LICENSE_END=========================================================
#
# ECOMP is a trademark and service mark of AT&T Intellectual Property.

"""generic utils to be used by dcae_policy decorators for the policy lifecycle in cloudify"""

from copy import deepcopy
from decimal import Decimal, DecimalException

FIELD_NAME_DELIMITER = ":"
FIELD_TYPE_DELIMITER = "::"
FIELD_TYPE_NUMBER = "number"
KEYWORD_DESC = "desc"
KEYWORD_NULLS_LAST = "nulls-last"

class Utils(object):
    """generic static class used for policy operations"""

    @staticmethod
    def remove_empties(any_list):
        """returns the any_list without empty elements"""
        return [element for element in any_list or [] if element]

    @staticmethod
    def get_field_value(parent, field_path, field_type=None):
        """
        Find and return the field :field_path: under :parent:

        Optionally, converts the field value to field_type.

        Parser of the :field_path: is using the delimiter ":" (semicolon)

        Example:
            parent = ctx.node.properties
            field_path = "docker_config:policy:apply_order"

            will return the value of the apply_order field under the ctx.node.properties in

                properties:
                    docker_config:
                        policy:
                            apply_order
        """
        if not parent or not field_path or not isinstance(parent, dict):
            return

        field_path = Utils.remove_empties([
            path.strip() for path in field_path.split(FIELD_NAME_DELIMITER)
        ])

        if not field_path:
            return

        field_value = None
        field_idx = len(field_path) - 1
        for (idx, field_name) in enumerate(field_path):
            parent = parent.get(field_name)
            if idx == field_idx:
                field_value = deepcopy(parent)
                if field_type in [FIELD_TYPE_NUMBER] and isinstance(field_value, (str, unicode)):
                    try:
                        field_value = Decimal(field_value)
                    except DecimalException:
                        pass
            elif not parent or not isinstance(parent, dict):
                return
        return field_value

    @staticmethod
    def parse_clause_item(clause_item):
        """
        Parses: the :clause_item: in policy_apply_order_clause
        and returns (field_path, field_type, reverse, nulls_last)

        delimiters: are whitespaces, "::"

        nulls-first is the default sorting order

        :clause_item: format is <field_path> [:: <field_type>] [desc] [nulls-last]

        Examples: "config:db_client" versus "config:foo desc" versus
            "matchingConditions:priority::number desc nulls-last"
        """
        field_path = field_type = desc = nulls_last = None

        if not clause_item or not isinstance(clause_item, (str, unicode)):
            return field_path, field_type, bool(desc), bool(nulls_last)

        for idx, token in enumerate(clause_item.split()):
            if idx == 0:
                split_for_type = token.split(FIELD_TYPE_DELIMITER)
                field_path = split_for_type[0]
                field_type = split_for_type[1] if len(split_for_type) > 1 else None
            elif token == KEYWORD_DESC:
                desc = True
            elif token == KEYWORD_NULLS_LAST:
                nulls_last = True
        return field_path, field_type, bool(desc), bool(nulls_last)

    @staticmethod
    def key_with_none_in_sort(reverse, nulls_last, value):
        """
        constructs tuple for proper placement of None values (last versus first)
        in the sorted list of values regardless of the :reverse:
        """
        return reverse == nulls_last or value is None, value