• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1#!/usr/bin/env python3
2# -*- coding: utf-8 -*-
3# Copyright (c) 2022-2023 Huawei Device Co., Ltd.
4# Licensed under the Apache License, Version 2.0 (the "License");
5# you may not use this file except in compliance with the License.
6# You may obtain a copy of the License at
7#
8#     http://www.apache.org/licenses/LICENSE-2.0
9#
10# Unless required by applicable law or agreed to in writing, software
11# distributed under the License is distributed on an "AS IS" BASIS,
12# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13# See the License for the specific language governing permissions and
14# limitations under the License.
15import json
16import os
17import getopt
18import sys
19import contextlib
20
21# Store the hash table of the services that need to validate
22PRIVILEGE_HASH = {}
23CRITICAL_HASH = {}
24
25
26class CfgValidateError(Exception):
27    """
28    When the process list verification fails, throw this exception
29    """
30    def __init__(self, name, reason):
31        super().__init__()
32        self.name = name
33        self.reason = reason
34
35
36class CfgItem:
37    """
38    CfgItem is the value of HASH, representing the interesetd field of a service read from a cfg file
39    """
40
41    def __init__(self):
42        self.uid = ""
43        self.gid = []
44        self.need_verified = False
45        self.enabled_critical = False
46        self.loc = ""
47        self.critical = []
48        self.related_item = ProcessItem()
49
50    def __init__(self, loc):
51        self.uid = ""
52        self.gid = []
53        self.need_verified = False
54        self.enabled_critical = False
55        self.loc = loc
56        self.critical = []
57        self.related_item = ProcessItem()
58
59    @classmethod
60    def _is_need_verified_uid(self, uid: str):
61        return uid == "root" or uid == "system"
62
63    @classmethod
64    def _is_need_verified_gid(self, gid):
65        # To enable gid-root validate, change it to "return gid == "root""
66        return False
67
68    @classmethod
69    def _is_need_verified_critical(self, critical: list):
70        return critical[0] == 1
71
72    def set_uid(self, uid):
73        """
74        Set uid and check it at the same time.
75        The uid needs to be validated only if _is_need_verified_uid return True
76        """
77        if CfgItem._is_need_verified_uid(uid):
78            self.uid = uid
79            self.need_verified = True
80
81    def append_gid(self, gid):
82        """
83        Append gid and check it at the same time.
84        The gid needs to be validated only if _is_need_verified_gid return True
85        """
86        if CfgItem._is_need_verified_gid(gid) and gid not in self.gid:
87            self.gid.append(gid)
88            self.need_verified = True
89
90    def set_critical(self, critical: bool):
91        if CfgItem._is_need_verified_critical(critical):
92            self.critical = critical
93            self.enabled_critical = True
94
95    def handle_socket(self, socket: dict):
96        """
97        Validate possible field "socket" in the field "services"
98        """
99        for i in socket:
100            if ("uid" in i) and CfgItem._is_need_verified_uid(i["uid"]):
101                self.need_verified = True
102                if self.uid != "" and self.uid != i["uid"]:
103                    print("Error: uid and uid in socket is not same!")
104                    print("Cfg location: {}".format(self.loc))
105                    raise CfgValidateError("Customization Error", "cfgs check not pass")
106                self.uid = i["uid"]
107            if "gid" in i :
108                if isinstance(i["gid"], str) and i["gid"] not in self.gid:
109                    self.append_gid(i["gid"])
110                    continue
111                for item in i["gid"]:
112                    self.append_gid(item)
113
114
115    def record_related_item(self, related_item):
116        """
117        When its permissions does not match those in process list,
118        records the permissions given in process list
119        """
120        self.related_item = related_item
121
122
123class ProcessItem:
124    """
125    Processitem is the data structure of an item read from the process list
126    """
127    def __init__(self):
128        self.name = ""
129        self.uid = ""
130        self.gid = []
131        self.critical = []
132
133    def __init__(self, process_item=None):
134        """
135        Use the JSON item in the process list to initialize the class
136        """
137        if process_item is None:
138            self.name = ""
139            self.uid = ""
140            self.gid = []
141            self.critical = []
142            return
143
144        self.name = process_item["name"]
145
146        if "uid" in process_item:
147            self.uid = process_item["uid"]
148        else:
149            self.uid = ""
150        if "gid" in process_item:
151            if isinstance(process_item["gid"], str):
152                self.gid = []
153                self.gid.append(process_item["gid"])
154            else:
155                self.gid = process_item["gid"]
156        else:
157            self.gid = []
158        if "critical" in process_item:
159            self.critical = process_item["critical"]
160        else:
161            self.critical = []
162
163    def verify(self, cfg_item):
164        """
165        Returns whether the corresponding CFG (cfg_item) has passed the verification
166        """
167        if self.uid or self.gid:
168            return self._verify_uid(cfg_item.uid) and self._verify_gid(cfg_item.gid)
169        if self.critical:
170            return self.critical == cfg_item.critical
171        return False
172
173    def _verify_uid(self, uid):
174        return not ((uid == "root" or uid == "system") and (uid != self.uid))
175
176    def _verify_gid(self, gid):
177        return not ("root" in gid and "root" not in self.gid)
178
179
180def print_privilege_hash():
181    global PRIVILEGE_HASH
182    for i in PRIVILEGE_HASH.items():
183        print("Name: {}\nuid: {}\ngiven uid: {}\ngid: ".format(i[0], i[1].uid, i[1].related_item.uid), end="")
184
185        for gid in i[1].gid:
186            print(gid, end=" ")
187        print("")
188
189        print("given gid: ", end=" ")
190        for gid in i[1].related_item.gid:
191            print(gid, end=" ")
192        print("")
193        print("Cfg location: {}".format(i[1].loc))
194        print("")
195
196
197def print_critical_hash():
198    global CRITICAL_HASH
199    for i in CRITICAL_HASH.items():
200        print("Cfg location: {}\n".format(i[1].loc), end="")
201        print("Name: {}\ncritical: {}\n".format(i[0], i[1].critical), end="")
202        print("Whitelist-allowed critical: {}\n".format(i[1].related_item.critical))
203        print("")
204
205
206def container_validate(process_path: str, list_name: str, item_container: list):
207    with open(process_path) as fp:
208        data = json.load(fp)
209        if list_name not in data:
210            print("Error: {}is not a valid whilelist, it has not a wanted field name".format(process_path))
211            raise CfgValidateError("Customization Error", "cfgs check not pass")
212
213        for i in data[list_name]:
214            if i["name"] not in item_container :
215                # no CfgItem in HASH meet the item in process list
216                continue
217
218            temp_item = ProcessItem(i)
219            if temp_item.name not in item_container:
220                continue
221
222            if temp_item.verify(item_container.get(temp_item.name)):
223                # Process field check passed, remove the corresponding service from HASH
224                item_container.pop(temp_item.name)
225            else:
226                item_container.get(temp_item.name).record_related_item(temp_item)
227
228
229def check_container(critical_process_path: str):
230    global PRIVILEGE_HASH
231    global CRITICAL_HASH
232    if PRIVILEGE_HASH:
233        # The remaining services in HASH do not pass the high-privilege validation
234        print("Error: some services are not authenticated. Listed as follow:")
235        print_privilege_hash()
236
237        raise CfgValidateError("Customization Error", "cfgs check not pass")
238
239    if CRITICAL_HASH:
240        # The remaining services in HASH do not pass the critical validation
241        print("Error: some services do not match with critical whitelist({}).".format(critical_process_path), end="")
242        print(" Directly enable critical or modify enabled critical services are prohibited!", end="")
243        print(" Misconfigured services listed as follow:")
244        print_critical_hash()
245
246        raise CfgValidateError("Customization Error", "cfgs check not pass")
247    return
248
249
250
251def validate_cfg_file(privilege_process_path: str, critical_process_path: str, result_path: str):
252    """
253    Load the process list file
254    For each item in the list, find out whether there is a CfgItem needs validation in HASH
255    """
256    global PRIVILEGE_HASH
257    global CRITICAL_HASH
258    if not os.path.exists(privilege_process_path):
259        print("High-privilege process check skipped: file [{}] not exist".format(privilege_process_path))
260        PRIVILEGE_HASH.clear()
261    else:
262        container_validate(privilege_process_path, "high_privilege_process_list", PRIVILEGE_HASH)
263    if not os.path.exists(critical_process_path):
264        print("Critical-reboot process check skipped: file [{}] not exist".format(critical_process_path))
265        CRITICAL_HASH.clear()
266    else:
267        container_validate(critical_process_path, "critical_reboot_process_list", CRITICAL_HASH)
268
269    check_container(critical_process_path)
270
271
272def handle_services(filename: str, field: str):
273    global PRIVILEGE_HASH
274    global CRITICAL_HASH
275    cfg_item = CfgItem(filename)
276    key = field['name']
277    if "uid" in field:
278        cfg_item.set_uid(field["uid"])
279    if "gid" in field:
280        if isinstance(field["gid"], str):
281            cfg_item.append_gid(field["gid"])
282        else:
283            for item in field["gid"]:
284                cfg_item.append_gid(item)
285    if "socket" in field:
286        cfg_item.handle_socket(field["socket"])
287    if "critical" in field:
288        cfg_item.set_critical(field["critical"])
289    if cfg_item.need_verified:
290        # Services that need to check permissions are added to HASH
291        PRIVILEGE_HASH[key] = cfg_item
292    if cfg_item.enabled_critical:
293        # Services that need to check critical are added to HASH
294        CRITICAL_HASH[key] = cfg_item
295
296
297def parse_cfg_file(filename: str):
298    """
299    Load the cfg file in JSON format
300    """
301    with open(filename) as fp:
302        try:
303            data = json.load(fp)
304        except json.decoder.JSONDecodeError:
305            print("\nError: loading cfg file {} failed. Cfg file not in JSON format!\n".format(filename))
306        if "services" not in data:
307            return
308        for field in data['services']:
309            handle_services(filename, field)
310    return
311
312
313def iterate_cfg_folder(cfg_dir: str):
314    for file in os.listdir(cfg_dir):
315        if file.endswith(".cfg"):
316            parse_cfg_file("{}/{}".format(cfg_dir, file))
317    return
318
319
320def main():
321    opts, args = getopt.getopt(sys.argv[1:], '', ['sys-cfg-folder=', 'vendor-cfg-folder=', \
322        'high-privilege-process-list-path=', 'critical-reboot-process-list-path=', 'result-path='])
323
324    sys_cfg_folder = opts[0][1]
325    if not os.path.exists(sys_cfg_folder):
326        print("Process field check skipped: file [{}] not exist".format(sys_cfg_folder))
327        return
328
329    vendor_cfg_folder = opts[1][1]
330    if not os.path.exists(vendor_cfg_folder):
331        print("Process field check skipped: file [{}] not exist".format(vendor_cfg_folder))
332        return
333
334    privilege_process_path = opts[2][1]
335
336    critical_process_path = opts[3][1]
337
338    iterate_cfg_folder(sys_cfg_folder)
339    iterate_cfg_folder(vendor_cfg_folder)
340    validate_cfg_file(privilege_process_path, critical_process_path, None)
341
342    return
343
344if __name__ == "__main__":
345
346    main()
347
348