-
Notifications
You must be signed in to change notification settings - Fork 31
/
Copy pathsupported_versions.py
176 lines (144 loc) · 6.37 KB
/
supported_versions.py
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
#!/usr/bin/env python
"""
Common data model supports several versions of Elasticsearch index mappings.
The following are versions of Elasticsearch for which the data model is released at the same time or were released
in the past.
Notice that in some cases the actually used ES version may slightly differ from the version of used model.
However, this still means that the data model is fully compatible with used Elasticsearch version. For instance,
as of writing we are using Elasticsearch 5.6.13 in OCP 4.1 but it is using model 5.5.2. Given there were no breaking
changes in index mappings between ES 5.6.13 and 5.5.2 we do not have to release different version of the model.
"""
_es2x = "2.4.4"
_es5x = "5.5.2"
_es6x = "6.8.3"
elasticsearch = [_es2x, _es5x, _es6x]
index_type_placeholder = "_index_type_"
_default_ = "_default_"
_doc = "_doc"
_index_types = {
_es2x: _default_,
_es5x: _default_,
_es6x: _doc
}
_index_templates = {
_es2x: "index_pattern_old_model",
_es5x: "index_pattern_old_model",
_es6x: "index_pattern"
}
def index_patterns(es_version):
"""
We change the index_patterns depending on ES version. Starting with ES6 we are switching to the
new data model. See https://github.com/openshift/origin-aggregated-logging/pull/1807#issuecomment-573721937
:param es_version:
:return: Field name containing index_patterns
"""
return _index_templates[es_version]
def index_type_name(es_version):
"""
:param es_version:
:return: Index type for given Elasticsearch version
"""
if es_version in elasticsearch:
return _index_types[es_version]
else:
raise Exception("No index type found for ES version", es_version)
def bw_mapping_compatibility(es_version, skeleton):
"""
Convert mapping skeleton to earlier Elasticsearch version.
See <https://github.com/ViaQ/elasticsearch-templates/issues/71> for follow up.
"""
if es_version == _es6x:
# es6x is the latest and all templates should be already
# formatted correctly
return
elif es_version == _es5x:
# BW conversion from es6x to es5x
_transform_skeleton_6x_to_5x(skeleton)
elif es_version == _es2x:
# First, BW conversion from es6x to es5x
_transform_skeleton_6x_to_5x(skeleton)
# Then, BW conversion from es5x to es2x
# Convert mappings of dynamic templates
_idx_type = index_type_name(es_version)
for template in skeleton['mappings'][_idx_type]['dynamic_templates']:
# json.dump(template, sys.stdout, indent=2, separators=(',', ': '), sort_keys=True)
first_key = list(template.keys())[0]
dynamic_mapping = template[first_key]['mapping']
_transform_mapping_5x_to_2x(dynamic_mapping)
# Convert mappings of properties
# for properties in skeleton['mappings'][_idx_type]['properties']:
# p_mapping = skeleton['mappings'][_idx_type]['properties'][properties]
# _transform_mapping_5x_to_2x(p_mapping)
default_mapping = skeleton['mappings'][_idx_type]
_transform_mapping_5x_to_2x(default_mapping)
else:
print("Unsupported Elasticsearch version: {} for index skeleton".format(es_version))
def _transform_skeleton_6x_to_5x(skeleton):
# See <https://github.com/ViaQ/elasticsearch-templates/issues/98>
skeleton['template'] = skeleton.pop('index_patterns')
def _transform_mapping_5x_to_2x(mapping):
keys = list(mapping.keys())
for key in keys:
# Transform 'norms'.
# See <https://github.com/ViaQ/elasticsearch-templates/issues/73>
if key == "norms":
# val = not mapping[key]
val = mapping[key]
mapping.pop(key, None)
mapping[key] = { 'enabled': val }
# TODO: Transform 'string' type and 'index'
# See <https://github.com/ViaQ/elasticsearch-templates/issues/67>
# See <https://github.com/ViaQ/elasticsearch-templates/issues/68>
if key == "type":
if mapping[key] == 'keyword':
mapping[key] = 'string'
mapping['index'] = 'not_analyzed'
elif mapping[key] == 'text':
mapping[key] = 'string'
mapping['index'] = 'analyzed'
elif mapping[key] in ['date', 'boolean', 'ip', 'long', 'integer', 'short', 'byte', 'double', 'float']:
mapping['index'] = 'not_analyzed'
if 'fields' in mapping:
fields_keys = mapping['fields'].keys()
for fields_key in fields_keys:
_transform_mapping_5x_to_2x(mapping['fields'][fields_key])
if 'properties' in mapping:
properties_keys = mapping['properties'].keys()
for property_key in properties_keys:
# ES2.x did not support ipv6 type. ES5.x support both ipv4 and ipv6 as "ip" type.
# Because of that we need to override ipv6 type to keyword type, which will be transformed to
# the string type later.
# The only way how we detect the ipv6 field type is by its name "ipaddr6"
if property_key == 'ipaddr6':
mapping['properties'][property_key]['type'] = 'keyword'
# ... and follows recursive call to mapping transformation.
_transform_mapping_5x_to_2x(mapping['properties'][property_key])
def bw_index_pattern_compatibility(es_version, res, field):
"""
Convert index-pattern field to earlier Elasticsearch version.
:param str es_version:
:param dict res: Output
:param dict field: Original field
"""
if es_version == _es6x:
return
elif es_version == _es5x:
return
elif es_version == _es2x:
_transform_field_5x_to_2x(res, field)
else:
print("Unsupported Elasticsearch version: {} for index pattern".format(es_version))
def _transform_field_5x_to_2x(res, field):
res["indexed"] = res.pop("searchable")
res["doc_values"] = res.pop("readFromDocValues")
res.pop("aggregatable")
res["analyzed"] = field.get("index", "") == True
if res["type"] == "text":
res["type"] = "string"
elif res["type"] == "keyword":
res["type"] = "string"
# assuming convention ipv6 address is called (or ends with) "ipaddr6"
# ES 2x did not support ipv6 types, we assume the same apply to Kibana 4.6.4 as well...
elif res["type"] == "ip" and res["name"].endswith("ipaddr6"):
res["type"] = "string"
return field