1#!/usr/bin/env python3 2# -*- coding: utf-8 -*- 3# Copyright (c) 2022 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. 15 16import os 17import stat 18import json 19import argparse 20import re 21import pandas as pd 22from bundle_check.bundle_check_common import BundleCheckTools 23from bundle_check.warning_info import BCWarnInfo 24 25 26class OhosInfo: 27 g_root_path = BundleCheckTools.get_root_path() 28 g_ohos_version = BundleCheckTools.get_ohos_version(g_root_path) 29 30 31def check_all_bundle_json(path:str) -> list: 32 ''' 33 @func: 检查指定目录下所有 bundle.json 的文件规范。 34 ''' 35 36 if os.path.isabs(path): 37 target_path = path 38 else: 39 target_path = os.path.join(OhosInfo.g_root_path, os.path.normpath(path)) 40 41 cur_path = os.getcwd() 42 os.chdir(target_path) 43 44 all_bundle = get_all_bundle_json() 45 all_error = [] 46 47 for bundle_json_path in all_bundle: 48 bundle_path = bundle_json_path.strip() 49 bundle = BundleJson(bundle_path) 50 bundle_error = bundle.check() 51 52 subsystem_name = bundle.subsystem_name 53 component_name = bundle.component_name 54 if len(subsystem_name) == 0: 55 subsystem_name = "Unknow" 56 if len(component_name) == 0: 57 component_name = "Unknow" 58 59 if not bundle_error: 60 continue 61 for item in bundle_error: 62 item['rule'] = BCWarnInfo.CHECK_RULE_2_1 63 item['path'] = bundle_path 64 item['component'] = component_name 65 item['subsystem'] = subsystem_name 66 all_error.extend(bundle_error) 67 count = len(all_error) 68 69 print('-------------------------------') 70 print('Bundle.json check successfully!') 71 print('There are {} issues in total'.format(count)) 72 print('-------------------------------') 73 os.chdir(cur_path) 74 return all_error 75 76 77def get_all_bundle_json(path:str = '.') -> list: 78 ''' 79 @func: 获取所有源码工程中所有 bundle.json 文件。 80 ''' 81 exclude_list = [ 82 r'"./out/*"', 83 r'"./.repo/*"' 84 ] 85 cmd = "find {} -name {}".format(path, "bundle.json") 86 for i in exclude_list: 87 cmd += " ! -path {}".format(i) 88 bundle_josn_list = os.popen(cmd).readlines() 89 return bundle_josn_list 90 91 92class BundlesCheck: 93 '''导出全量检查的结果。''' 94 95 @staticmethod 96 def to_json(all_errors:dict, 97 output_path:str = '.', 98 output_name:str = 'all_bundle_error.json'): 99 '''@func: 导出所有错误到 json 格式文件中。''' 100 all_errors = check_all_bundle_json(OhosInfo.g_root_path) 101 all_error_json = json.dumps(all_errors, 102 indent=4, 103 ensure_ascii=False, 104 separators=(', ', ': ')) 105 out_path = os.path.normpath(output_path) + '/' + output_name 106 107 flags = os.O_WRONLY | os.O_CREAT 108 modes = stat.S_IWUSR | stat.S_IRUSR 109 with os.fdopen(os.open(out_path, flags, modes), 'w') as file: 110 file.write(all_error_json) 111 print("Please check " + out_path) 112 113 @staticmethod 114 def to_df(path:str = None) -> pd.DataFrame: 115 '''将所有错误的 dict 数据类型转为 pd.DataFrame 类型。''' 116 if path is None: 117 path = OhosInfo.g_root_path 118 else: 119 path = os.path.join(OhosInfo.g_root_path, path) 120 all_errors = check_all_bundle_json(path) 121 columns = ['子系统', '部件', '文件', '违反规则', '详细', '说明'] 122 errors_list = [] 123 for item in all_errors: 124 error_temp = [ 125 item['subsystem'], 126 item['component'], 127 item['path'], 128 item['rule'], 129 "line" + str(item['line']) + ": " + item['contents'], 130 item['description'] 131 ] 132 errors_list.append(error_temp) 133 ret = pd.DataFrame(errors_list, columns=columns) 134 return ret 135 136 @staticmethod 137 def to_excel(output_path:str = '.', 138 output_name:str = 'all_bundle_error.xlsx'): 139 ''' 140 @func: 导出所有错误到 excel 格式文件中。 141 ''' 142 err_df = BundlesCheck.to_df() 143 outpath = os.path.normpath(output_path) + '/' + output_name 144 err_df.to_excel(outpath, index=None) 145 print('Please check ' + outpath) 146 147 148class BundleJson(object): 149 '''以 bundle.josn 路径来初始化的对象,包含关于该 bundle.josn 的一些属性和操作。 150 @var: 151 - ``__all_errors`` : 表示该文件的所有错误列表。 152 - ``__json`` : 表示将该 josn 文件转为 dict 类型后的内容。 153 - ``__lines`` : 表示将该 josn 文件转为 list 类型后的内容。 154 155 @method: 156 - ``component_name()`` : 返回该 bundle.json 所在部件名。 157 - ``subsystem_name()`` : 返回该 bundle.json 所在子系统名。 158 - ``readlines()`` : 返回该 bundle.json 以每一行内容为元素的 list。 159 - ``get_line_number(s)`` : 返回 s 字符串在该 bundle.josn 中的行号,未知则返回 0。 160 - ``check()`` : 静态检查该 bundle.json,返回错误告警 list。 161 ''' 162 163 def __init__(self, path:str) -> None: 164 self.__all_errors = [] # 该文件的所有错误列表 165 self.__json = {} # 将该 josn 文件转为字典类型内容 166 self.__lines = [] # 将该 josn 文件转为列表类型内容 167 with open(path, 'r') as file: 168 try: 169 self.__json = json.load(file) 170 except json.decoder.JSONDecodeError as error: 171 raise ValueError("'" + path + "'" + " is not a json file.") 172 with open(path, 'r') as file: 173 self.__lines = file.readlines() 174 175 @property 176 def component_name(self) -> str: 177 return self.__json.get('component').get('name') 178 179 @property 180 def subsystem_name(self) -> str: # 目前存在为空的情况 181 return self.__json.get('component').get('subsystem') 182 183 def readlines(self) -> list: 184 return self.__lines 185 186 def get_line_number(self, string) -> int: 187 ''' 188 @func: 获取指定字符串所在行号。 189 ''' 190 line_num = 0 191 for line in self.__lines: 192 line_num += 1 193 if string in line: 194 return line_num 195 return 0 196 197 def check(self) -> list: 198 ''' 199 @func: 检查该 bundle.json 规范。 200 @note: 去除检查 version 字段。 201 ''' 202 err_name = self.check_name() 203 err_segment = self.check_segment() 204 err_component = self.check_component() 205 if err_name: 206 self.__all_errors.append(err_name) 207 if err_segment: 208 self.__all_errors.extend(err_segment) 209 if err_component: 210 self.__all_errors.extend(err_component) 211 212 return self.__all_errors 213 214 # name 215 def check_name(self) -> dict: 216 bundle_error = dict(line=0, contents='"name"') 217 218 if 'name' not in self.__json: 219 bundle_error["description"] = BCWarnInfo.NAME_NO_FIELD 220 return bundle_error 221 222 name = self.__json['name'] 223 bundle_error["line"] = self.get_line_number('"name"') 224 if not name: # 为空 225 bundle_error["description"] = BCWarnInfo.NAME_EMPTY 226 return bundle_error 227 228 bundle_error["description"] = BCWarnInfo.NAME_FORMAT_ERROR + \ 229 BCWarnInfo.COMPONENT_NAME_FROMAT + \ 230 BCWarnInfo.COMPONENT_NAME_FROMAT_LEN 231 match = BundleCheckTools.match_bundle_full_name(name) 232 if not match: 233 return bundle_error 234 match = BundleCheckTools.match_unix_like_name(name.split('/')[1]) 235 if not match: 236 return bundle_error 237 238 return dict() 239 240 # version 241 def check_version(self) -> dict: 242 bundle_error = dict(line=0, contents='version') 243 244 if 'version' not in self.__json: 245 bundle_error["description"] = BCWarnInfo.VERSION_NO_FIELD 246 return bundle_error 247 248 bundle_error["line"] = self.get_line_number('"version": ') 249 if len(self.__json['version']) < 3: # example 3.1 250 bundle_error["description"] = BCWarnInfo.VERSION_ERROR 251 return bundle_error 252 253 if self.__json['version'] != OhosInfo.g_ohos_version: 254 bundle_error['description'] = BCWarnInfo.VERSION_ERROR + \ 255 ' current ohos version is: ' + OhosInfo.g_ohos_version 256 return bundle_error 257 return dict() 258 259 # segment 260 def check_segment(self) -> list: 261 bundle_error_segment = [] 262 bundle_error = dict(line=0, contents='"segment"') 263 264 if 'segment' not in self.__json: 265 bundle_error["description"] = BCWarnInfo.SEGMENT_NO_FIELD 266 bundle_error_segment.append(bundle_error) 267 return bundle_error_segment 268 269 bundle_error["line"] = self.get_line_number('"segment":') 270 if 'destPath' not in self.__json['segment']: 271 bundle_error["description"] = BCWarnInfo.SEGMENT_DESTPATH_NO_FIELD 272 bundle_error_segment.append(bundle_error) 273 return bundle_error_segment 274 275 path = self.__json['segment']['destPath'] 276 bundle_error["line"] = self.get_line_number('"destPath":') 277 bundle_error["contents"] = '"segment:destPath"' 278 if not path: 279 bundle_error["description"] = BCWarnInfo.SEGMENT_DESTPATH_EMPTY 280 bundle_error_segment.append(bundle_error) 281 return bundle_error_segment 282 283 if type(path) != str: 284 bundle_error["description"] = BCWarnInfo.SEGMENT_DESTPATH_UNIQUE 285 bundle_error_segment.append(bundle_error) 286 return bundle_error_segment 287 288 if os.path.isabs(path): 289 bundle_error["description"] = BCWarnInfo.SEGMENT_DESTPATH_ABS 290 bundle_error_segment.append(bundle_error) 291 return bundle_error_segment 292 293 return bundle_error_segment 294 295 # component 296 def check_component(self) -> list: 297 bundle_error_component = [] 298 299 if 'component' not in self.__json: 300 bundle_error = dict(line=0, contents='"component"', 301 description=BCWarnInfo.COMPONENT_NO_FIELD) 302 bundle_error_component.append(bundle_error) 303 return bundle_error_component 304 305 component = self.__json.get('component') 306 component_line = self.get_line_number('"component":') 307 self._check_component_name(component, component_line, bundle_error_component) 308 self._check_component_subsystem(component, component_line, bundle_error_component) 309 self._check_component_syscap(component, bundle_error_component) 310 self._check_component_ast(component, component_line, bundle_error_component) 311 self._check_component_rom(component, component_line, bundle_error_component) 312 self._check_component_ram(component, component_line, bundle_error_component) 313 self._check_component_deps(component, component_line, bundle_error_component) 314 315 return bundle_error_component 316 317 # component name 318 def _check_component_name(self, component: dict, component_line: int, bundle_error_component: list): 319 if 'name' not in component: 320 bundle_error = dict(line=component_line, 321 contents='"component"', 322 description=BCWarnInfo.COMPONENT_NAME_NO_FIELD) 323 bundle_error_component.append(bundle_error) 324 else: 325 bundle_error = dict(line=component_line + 1, 326 contents='"component:name"') # 同名 "name" 暂用 "component" 行号+1 327 if not component['name']: 328 bundle_error["description"] = BCWarnInfo.COMPONENT_NAME_EMPTY 329 bundle_error_component.append(bundle_error) 330 elif 'name' in self.__json and '/' in self.__json['name']: 331 if component['name'] != self.__json['name'].split('/')[1]: 332 bundle_error["description"] = BCWarnInfo.COMPONENT_NAME_VERACITY 333 bundle_error_component.append(bundle_error) 334 335 # component subsystem 336 def _check_component_subsystem(self, component: dict, component_line: int, 337 bundle_error_component: list): 338 if 'subsystem' not in component: 339 bundle_error = dict(line=component_line, 340 contents="component", 341 description=BCWarnInfo.COMPONENT_SUBSYSTEM_NO_FIELD) 342 bundle_error_component.append(bundle_error) 343 else: 344 bundle_error = dict(line=self.get_line_number('"subsystem":'), 345 contents='"component:subsystem"') 346 if not component['subsystem']: 347 bundle_error["description"] = BCWarnInfo.COMPONENT_SUBSYSTEM_EMPTY 348 bundle_error_component.append(bundle_error) 349 elif not BundleCheckTools.is_all_lower(component['subsystem']): 350 bundle_error["description"] = BCWarnInfo.COMPONENT_SUBSYSTEM_LOWCASE 351 bundle_error_component.append(bundle_error) 352 353 # component syscap 可选且可以为空 354 def _check_component_syscap(self, component: dict, bundle_error_component: list): 355 if 'syscap' not in component: 356 pass 357 elif component['syscap']: 358 bundle_error = dict(line=self.get_line_number('"syscap":'), 359 contents='"component:syscap"') 360 err = [] # 收集所有告警 361 for i in component['syscap']: 362 # syscap string empty 363 if not i: 364 err.append(BCWarnInfo.COMPONENT_SYSCAP_STRING_EMPTY) 365 continue 366 match = re.match(r'^SystemCapability(\.[A-Z][a-zA-Z]{1,63}){2,6}$', i) 367 if not match: 368 err.append(BCWarnInfo.COMPONENT_SYSCAP_STRING_FORMAT_ERROR) 369 errs = list(set(err)) # 去重告警 370 if errs: 371 bundle_error["description"] = str(errs) 372 bundle_error_component.append(bundle_error) 373 374 # component adapted_system_type 375 def _check_component_ast(self, component: dict, component_line: int, bundle_error_component: list): 376 if 'adapted_system_type' not in component: 377 bundle_error = dict(line=component_line, contents='"component"', 378 description=BCWarnInfo.COMPONENT_AST_NO_FIELD) 379 bundle_error_component.append(bundle_error) 380 return 381 382 bundle_error = dict(line=self.get_line_number('"adapted_system_type":'), 383 contents='"component:adapted_system_type"') 384 ast = component["adapted_system_type"] 385 if not ast: 386 bundle_error["description"] = BCWarnInfo.COMPONENT_AST_EMPTY 387 bundle_error_component.append(bundle_error) 388 return 389 390 type_set = tuple(set(ast)) 391 if len(ast) > 3 or len(type_set) != len(ast): 392 bundle_error["description"] = BCWarnInfo.COMPONENT_AST_NO_REP 393 bundle_error_component.append(bundle_error) 394 return 395 396 all_type_list = ["mini", "small", "standard"] 397 # 不符合要求的 type 398 error_type = [i for i in ast if i not in all_type_list] 399 if error_type: 400 bundle_error["description"] = BCWarnInfo.COMPONENT_AST_NO_REP 401 bundle_error_component.append(bundle_error) 402 return 403 404 # component rom 405 def _check_component_rom(self, component: dict, component_line: int, bundle_error_component: list): 406 if 'rom' not in component: 407 bundle_error = dict(line=component_line, contents='"component:rom"', 408 description=BCWarnInfo.COMPONENT_ROM_NO_FIELD) 409 bundle_error_component.append(bundle_error) 410 elif not component["rom"]: 411 bundle_error = dict(line=self.get_line_number('"rom":'), 412 contents='"component:rom"', 413 description=BCWarnInfo.COMPONENT_ROM_EMPTY) 414 bundle_error_component.append(bundle_error) 415 else: 416 bundle_error = dict(line=self.get_line_number('"rom":'), 417 contents='"component:rom"') 418 num, unit = BundleCheckTools.split_by_unit(component["rom"]) 419 if num < 0: 420 bundle_error["description"] = BCWarnInfo.COMPONENT_ROM_SIZE_ERROR # 非数值或小于0 421 bundle_error_component.append(bundle_error) 422 elif unit: 423 unit_list = ["KB", "KByte", "MByte", "MB"] 424 if unit not in unit_list: 425 bundle_error["description"] = BCWarnInfo.COMPONENT_ROM_UNIT_ERROR # 单位有误 426 bundle_error_component.append(bundle_error) 427 428 # component ram 429 def _check_component_ram(self, component: dict, component_line: int, bundle_error_component: list): 430 if 'ram' not in component: 431 bundle_error = dict(line=component_line, contents='"component:ram"', 432 description=BCWarnInfo.COMPONENT_RAM_NO_FIELD) 433 bundle_error_component.append(bundle_error) 434 elif not component["ram"]: 435 bundle_error = dict(line=self.get_line_number('"ram":'), 436 contents='"component:ram"', 437 description=BCWarnInfo.COMPONENT_RAM_EMPTY) 438 bundle_error_component.append(bundle_error) 439 else: 440 bundle_error = dict(line=self.get_line_number('"ram":'), 441 contents='"component:ram"') 442 num, unit = BundleCheckTools.split_by_unit(component["ram"]) 443 if num <= 0: 444 bundle_error["description"] = BCWarnInfo.COMPONENT_RAM_SIZE_ERROR # 非数值或小于0 445 bundle_error_component.append(bundle_error) 446 elif unit: 447 unit_list = ["KB", "KByte", "MByte", "MB"] 448 if unit not in unit_list: 449 bundle_error["description"] = BCWarnInfo.COMPONENT_RAM_UNIT_ERROR # 单位有误 450 bundle_error_component.append(bundle_error) 451 452 # component deps 453 def _check_component_deps(self, component: dict, component_line: int, bundle_error_component: list): 454 if 'deps' not in component: 455 bundle_error = dict(line=component_line, contents='"component:deps"', 456 description=BCWarnInfo.COMPONENT_DEPS_NO_FIELD) 457 bundle_error_component.append(bundle_error) 458 else: 459 pass 460 461 462def parse_args(): 463 parser = argparse.ArgumentParser() 464 # exclusive output format 465 ex_format = parser.add_mutually_exclusive_group() 466 ex_format.add_argument("--xlsx", help="output format: xls(default).", 467 action="store_true") 468 ex_format.add_argument("--json", help="output format: json.", 469 action="store_true") 470 # exclusive input 471 ex_input = parser.add_mutually_exclusive_group() 472 ex_input.add_argument("-P", "--project", help="project root path.", type=str) 473 ex_input.add_argument("-p", "--path", help="bundle.json path list.", nargs='+') 474 # output path 475 parser.add_argument("-o", "--output", help="ouput path.") 476 args = parser.parse_args() 477 478 export_path = '.' 479 if args.output: 480 export_path = args.output 481 482 if args.project: 483 if not BundleCheckTools.is_project(args.project): 484 print("'" + args.project + "' is not a oopeharmony project.") 485 exit(1) 486 487 if args.json: 488 BundlesCheck.to_json(export_path) 489 else: 490 BundlesCheck.to_excel(export_path) 491 elif args.path: 492 bundle_list_error = {} 493 for bundle_json_path in args.path: 494 bundle = BundleJson(bundle_json_path) 495 error_field = bundle.check() 496 if error_field: 497 bundle_list_error[bundle_json_path] = \ 498 dict([(BCWarnInfo.CHECK_RULE_2_1, error_field)]) 499 # temp 500 test_json = json.dumps(bundle_list_error, 501 indent=4, separators=(', ', ': '), 502 ensure_ascii=False) 503 print(test_json) 504 else: 505 print("use '-h' get help.") 506 507 508if __name__ == '__main__': 509 parse_args() 510