96064deb创建于 2025年11月27日历史提交
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# Copyright (c) 2025 Huawei Device Co., Ltd.
# 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.


import subprocess
import argparse
import os
import sys
import shlex


def parse_args(args):
    parser = argparse.ArgumentParser()
    parser.add_argument('--sdk-out-dir')
    parser.add_argument('--cert-path', type=str, required=True,
                        help='Path to the certificate file')
    options = parser.parse_args(args)
    return options


def get_script_directory():
    return os.path.dirname(os.path.abspath(__file__))


def convert_to_absolute_path(relative_or_absolute_path, base_directory):
    if os.path.isabs(relative_or_absolute_path):
        return relative_or_absolute_path
    else:
        return os.path.join(base_directory, relative_or_absolute_path)


def find_file(cert_path, key_path):
    if not os.path.isfile(cert_path):
        raise FileNotFoundError(f"The certificate file does not exist: {cert_path}")
        
    if not os.path.isfile(key_path):
        raise FileNotFoundError(f"The certificate file does not exist: {key_path}")


def sign_sdk(zipfile, sign_list, sign_results, ohos_sdk_dir, cert_path):
    out_path = os.path.join('out', 'sdk')
    packing_tool_dir_re = os.path.join(out_path, 'ohos-sdk', 'linux', 'toolchains', 'lib')
    packing_tool_dir = os.path.join(os.path.dirname(os.path.dirname(os.path.dirname(__file__))), packing_tool_dir_re)
    packing_tool = os.path.abspath(os.path.join(packing_tool_dir, 'binary-sign-tool'))
    
    sign = f"ohos sdk key store"
    cert_file = os.path.abspath(os.path.join(cert_path, 'OHOS_SDK.cer'))
    key_store_file = os.path.abspath(os.path.join(cert_path, 'OHOS_SDK.p12'))

    try:
        find_file(cert_file, key_store_file)
    except FileNotFoundError as e:
        print(f"Error: {e}")

    key_pwd = os.getenv('KEY_PWD')
    key_store_pwd = os.getenv('KEY_STORE_PWD')
    
    if zipfile.endswith('.zip'):
        dir_name = zipfile.split('-')[0]
        cmd1 = ['unzip', "-q", zipfile]
        subprocess.call(cmd1)
        need_sign_files = []
        
        for root, dirs, files in os.walk(dir_name):
            for file in files:
                file = os.path.join(root, file)
                need_sign_files.append(file)
        for file in need_sign_files:
            if file.split('/')[-1] in sign_list or file.endswith('.so') or file.endswith('.dylib') \
                    or file.split('/')[-2] == 'bin' and not file.endswith('.js'):
                cmd2 = [packing_tool, 
                    'sign', 
                    '-keyAlias', sign,  
                    '-signAlg', 'SHA256withECDSA', 
                    '-keystoreFile', key_store_file,
                    '-inFile', file, 
                    '-outFile', file,
                    '-keyPwd', key_pwd,
                    '-appCertFile', cert_file, 
                    '-keystorePwd',key_store_pwd]
                subprocess.call(cmd2)
        cmd3 = ['rm', zipfile]
        subprocess.call(cmd3)
        cmd4 = ['zip', '-rq', zipfile, dir_name]
        subprocess.call(cmd4)
        cmd5 = ['rm', '-rf', dir_name]
        subprocess.call(cmd5)


def main(args):
    options = parse_args(args)
    ohos_sdk_dir = os.path.join(options.sdk_out_dir, 'ohos')
    script_directory = get_script_directory()
    cert_path = convert_to_absolute_path(options.cert_path, script_directory)
    os.chdir(ohos_sdk_dir)
    sign_list = ['lldb-argdumper', 'fsevents.node', 'idl', 'restool', 'diff', 'ark_asm',
                 'ark_disasm', 'hdc', 'syscap_tool', 'hnpcli', 'rawheap_translator']
    sign_results = []
    for file in os.listdir('.'):
        sign_sdk(file, sign_list, sign_results, ohos_sdk_dir, cert_path)
    for cmd, process in sign_results:
        try:
            stdout, stderr = process.communicate(timeout=3600)
            if process.returncode:
                print(f"cmd:{' '.join(cmd)}, result is {stdout}")       
                raise Exception(f"run command {' '.join(cmd)} fail, error is {stderr}")
        except Exception as e:
            raise TimeoutError(r"sign timeout")


if __name__ == '__main__':
    sys.exit(main(sys.argv[1:]))