xref: /OK3568_Linux_fs/yocto/poky/scripts/oe-publish-sdk (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun#!/usr/bin/env python3
2*4882a593Smuzhiyun#
3*4882a593Smuzhiyun# OpenEmbedded SDK publishing tool
4*4882a593Smuzhiyun#
5*4882a593Smuzhiyun# Copyright (C) 2015-2016 Intel Corporation
6*4882a593Smuzhiyun#
7*4882a593Smuzhiyun# SPDX-License-Identifier: GPL-2.0-only
8*4882a593Smuzhiyun#
9*4882a593Smuzhiyun
10*4882a593Smuzhiyunimport sys
11*4882a593Smuzhiyunimport os
12*4882a593Smuzhiyunimport argparse
13*4882a593Smuzhiyunimport glob
14*4882a593Smuzhiyunimport re
15*4882a593Smuzhiyunimport subprocess
16*4882a593Smuzhiyunimport logging
17*4882a593Smuzhiyunimport shutil
18*4882a593Smuzhiyunimport errno
19*4882a593Smuzhiyun
20*4882a593Smuzhiyunscripts_path = os.path.dirname(os.path.realpath(__file__))
21*4882a593Smuzhiyunlib_path = scripts_path + '/lib'
22*4882a593Smuzhiyunsys.path = sys.path + [lib_path]
23*4882a593Smuzhiyunimport scriptutils
24*4882a593Smuzhiyunimport argparse_oe
25*4882a593Smuzhiyunlogger = scriptutils.logger_create('sdktool')
26*4882a593Smuzhiyun
27*4882a593Smuzhiyundef mkdir(d):
28*4882a593Smuzhiyun    try:
29*4882a593Smuzhiyun        os.makedirs(d)
30*4882a593Smuzhiyun    except OSError as e:
31*4882a593Smuzhiyun        if e.errno != errno.EEXIST:
32*4882a593Smuzhiyun            raise e
33*4882a593Smuzhiyun
34*4882a593Smuzhiyundef publish(args):
35*4882a593Smuzhiyun    logger.debug("In publish function")
36*4882a593Smuzhiyun    target_sdk = args.sdk
37*4882a593Smuzhiyun    destination = args.dest
38*4882a593Smuzhiyun    logger.debug("target_sdk = %s, update_server = %s" % (target_sdk, destination))
39*4882a593Smuzhiyun    sdk_basename = os.path.basename(target_sdk)
40*4882a593Smuzhiyun
41*4882a593Smuzhiyun    # Ensure the SDK exists
42*4882a593Smuzhiyun    if not os.path.exists(target_sdk):
43*4882a593Smuzhiyun        logger.error("Specified SDK %s doesn't exist" % target_sdk)
44*4882a593Smuzhiyun        return -1
45*4882a593Smuzhiyun    if os.path.isdir(target_sdk):
46*4882a593Smuzhiyun        logger.error("%s is a directory - expected path to SDK installer file" % target_sdk)
47*4882a593Smuzhiyun        return -1
48*4882a593Smuzhiyun
49*4882a593Smuzhiyun    if ':' in destination:
50*4882a593Smuzhiyun        is_remote = True
51*4882a593Smuzhiyun        host, destdir = destination.split(':')
52*4882a593Smuzhiyun        dest_sdk = os.path.join(destdir, sdk_basename)
53*4882a593Smuzhiyun    else:
54*4882a593Smuzhiyun        is_remote = False
55*4882a593Smuzhiyun        dest_sdk = os.path.join(destination, sdk_basename)
56*4882a593Smuzhiyun        destdir = destination
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun    # Making sure the directory exists
59*4882a593Smuzhiyun    logger.debug("Making sure the destination directory exists")
60*4882a593Smuzhiyun    if not is_remote:
61*4882a593Smuzhiyun        mkdir(destination)
62*4882a593Smuzhiyun    else:
63*4882a593Smuzhiyun        cmd = "ssh %s 'mkdir -p %s'" % (host, destdir)
64*4882a593Smuzhiyun        ret = subprocess.call(cmd, shell=True)
65*4882a593Smuzhiyun        if ret != 0:
66*4882a593Smuzhiyun            logger.error("Making directory %s on %s failed" % (destdir, host))
67*4882a593Smuzhiyun            return ret
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun    # Copying the SDK to the destination
70*4882a593Smuzhiyun    logger.info("Copying the SDK to destination")
71*4882a593Smuzhiyun    if not is_remote:
72*4882a593Smuzhiyun        if os.path.exists(dest_sdk):
73*4882a593Smuzhiyun            os.remove(dest_sdk)
74*4882a593Smuzhiyun        if (os.stat(target_sdk).st_dev == os.stat(destination).st_dev):
75*4882a593Smuzhiyun            os.link(target_sdk, dest_sdk)
76*4882a593Smuzhiyun        else:
77*4882a593Smuzhiyun            shutil.copy(target_sdk, dest_sdk)
78*4882a593Smuzhiyun    else:
79*4882a593Smuzhiyun        cmd = "scp %s %s" % (target_sdk, destination)
80*4882a593Smuzhiyun        ret = subprocess.call(cmd, shell=True)
81*4882a593Smuzhiyun        if ret != 0:
82*4882a593Smuzhiyun            logger.error("scp %s %s failed" % (target_sdk, destination))
83*4882a593Smuzhiyun            return ret
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun    # Unpack the SDK
86*4882a593Smuzhiyun    logger.info("Unpacking SDK")
87*4882a593Smuzhiyun    if not is_remote:
88*4882a593Smuzhiyun        cmd = "sh %s -p -y -d %s" % (dest_sdk, destination)
89*4882a593Smuzhiyun        ret = subprocess.call(cmd, shell=True)
90*4882a593Smuzhiyun        if ret == 0:
91*4882a593Smuzhiyun            logger.info('Successfully unpacked %s to %s' % (dest_sdk, destination))
92*4882a593Smuzhiyun            os.remove(dest_sdk)
93*4882a593Smuzhiyun        else:
94*4882a593Smuzhiyun            logger.error('Failed to unpack %s to %s' % (dest_sdk, destination))
95*4882a593Smuzhiyun            return ret
96*4882a593Smuzhiyun    else:
97*4882a593Smuzhiyun        rm_or_not = " && rm -f %s" % dest_sdk
98*4882a593Smuzhiyun        if args.keep_orig:
99*4882a593Smuzhiyun            rm_or_not = ""
100*4882a593Smuzhiyun        cmd = "ssh %s 'sh %s -p -y -d %s%s'" % (host, dest_sdk, destdir, rm_or_not)
101*4882a593Smuzhiyun        ret = subprocess.call(cmd, shell=True)
102*4882a593Smuzhiyun        if ret == 0:
103*4882a593Smuzhiyun            logger.info('Successfully unpacked %s to %s' % (dest_sdk, destdir))
104*4882a593Smuzhiyun        else:
105*4882a593Smuzhiyun            logger.error('Failed to unpack %s to %s' % (dest_sdk, destdir))
106*4882a593Smuzhiyun            return ret
107*4882a593Smuzhiyun
108*4882a593Smuzhiyun    # Setting up the git repo
109*4882a593Smuzhiyun    if not is_remote:
110*4882a593Smuzhiyun        cmd = 'set -e; mkdir -p %s/layers; cd %s/layers; if [ ! -e .git ]; then git init .; cp .git/hooks/post-update.sample .git/hooks/post-commit; echo "*.pyc\n*.pyo\npyshtables.py" > .gitignore; fi; git config gc.auto 0; git add -A .; git config user.email "oe@oe.oe" && git config user.name "OE" && git commit -q -m "init repo" || true' % (destination, destination)
111*4882a593Smuzhiyun    else:
112*4882a593Smuzhiyun        cmd = "ssh %s 'set -e; mkdir -p %s/layers; cd %s/layers; if [ ! -e .git ]; then git init .; cp .git/hooks/post-update.sample .git/hooks/post-commit; echo '*.pyc' > .gitignore; echo '*.pyo' >> .gitignore; echo 'pyshtables.py' >> .gitignore; fi; git config gc.auto 0; git add -A .; git config user.email 'oe@oe.oe' && git config user.name 'OE' && git commit -q -m \"init repo\" || true'" % (host, destdir, destdir)
113*4882a593Smuzhiyun    ret = subprocess.call(cmd, shell=True)
114*4882a593Smuzhiyun    if ret == 0:
115*4882a593Smuzhiyun        logger.info('SDK published successfully')
116*4882a593Smuzhiyun    else:
117*4882a593Smuzhiyun        logger.error('Failed to set up layer git repo')
118*4882a593Smuzhiyun    return ret
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun
121*4882a593Smuzhiyundef main():
122*4882a593Smuzhiyun    parser = argparse_oe.ArgumentParser(description="OpenEmbedded extensible SDK publishing tool - writes server-side data to support the extensible SDK update process to a specified location")
123*4882a593Smuzhiyun    parser.add_argument('-d', '--debug', help='Enable debug output', action='store_true')
124*4882a593Smuzhiyun    parser.add_argument('-q', '--quiet', help='Print only errors', action='store_true')
125*4882a593Smuzhiyun    parser.add_argument('-k', '--keep-orig', help='When published to a remote host, the eSDK installer gets deleted by default.', action='store_true')
126*4882a593Smuzhiyun
127*4882a593Smuzhiyun    parser.add_argument('sdk', help='Extensible SDK to publish (path to .sh installer file)')
128*4882a593Smuzhiyun    parser.add_argument('dest', help='Destination to publish SDK to; can be local path or remote in the form of user@host:/path (in the latter case ssh/scp will be used).')
129*4882a593Smuzhiyun
130*4882a593Smuzhiyun    parser.set_defaults(func=publish)
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun    args = parser.parse_args()
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun    if args.debug:
135*4882a593Smuzhiyun        logger.setLevel(logging.DEBUG)
136*4882a593Smuzhiyun    elif args.quiet:
137*4882a593Smuzhiyun        logger.setLevel(logging.ERROR)
138*4882a593Smuzhiyun
139*4882a593Smuzhiyun    ret = args.func(args)
140*4882a593Smuzhiyun    return ret
141*4882a593Smuzhiyun
142*4882a593Smuzhiyunif __name__ == "__main__":
143*4882a593Smuzhiyun    try:
144*4882a593Smuzhiyun        ret = main()
145*4882a593Smuzhiyun    except Exception:
146*4882a593Smuzhiyun        ret = 1
147*4882a593Smuzhiyun        import traceback
148*4882a593Smuzhiyun        traceback.print_exc()
149*4882a593Smuzhiyun    sys.exit(ret)
150