1#!/usr/bin/env python3 2# 3# Copyright (C) 2019 The Android Open Source Project 4# 5# Licensed under the Apache License, Version 2.0 (the "License"); 6# you may not use this file except in compliance with the License. 7# You may obtain a copy of the License at 8# 9# http://www.apache.org/licenses/LICENSE-2.0 10# 11# Unless required by applicable law or agreed to in writing, software 12# distributed under the License is distributed on an "AS IS" BASIS, 13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14# See the License for the specific language governing permissions and 15# limitations under the License. 16 17""" 18Generates a self extracting archive with a license click through. 19 20Usage: 21 generate-self-extracting-archive.py $OUTPUT_FILE $INPUT_ARCHIVE $COMMENT $LICENSE_FILE 22 23 The comment will be included at the beginning of the output archive file. 24 25Output: 26 The output of the script is a single executable file that when run will 27 display the provided license and if the user accepts extract the wrapped 28 archive. 29 30 The layout of the output file is roughly: 31 * Executable shell script that extracts the archive 32 * Actual archive contents 33 * Zip file containing the license 34""" 35 36import tempfile 37import sys 38import os 39import zipfile 40 41_HEADER_TEMPLATE = """#!/bin/bash 42# 43{comment_line} 44# 45# Usage is subject to the enclosed license agreement 46 47echo 48echo The license for this software will now be displayed. 49echo You must agree to this license before using this software. 50echo 51echo -n Press Enter to view the license 52read dummy 53echo 54more << EndOfLicense 55{license} 56EndOfLicense 57 58if test $? != 0 59then 60 echo "ERROR: Couldn't display license file" 1>&2 61 exit 1 62fi 63echo 64echo -n 'Type "I ACCEPT" if you agree to the terms of the license: ' 65read typed 66if test "$typed" != "I ACCEPT" 67then 68 echo 69 echo "You didn't accept the license. Extraction aborted." 70 exit 2 71fi 72echo 73{extract_command} 74if test $? != 0 75then 76 echo 77 echo "ERROR: Couldn't extract files." 1>&2 78 exit 3 79else 80 echo 81 echo "Files extracted successfully." 82fi 83exit 0 84""" 85 86_PIPE_CHUNK_SIZE = 1048576 87def _pipe_bytes(src, dst): 88 while True: 89 b = src.read(_PIPE_CHUNK_SIZE) 90 if not b: 91 break 92 dst.write(b) 93 94_MAX_OFFSET_WIDTH = 20 95def _generate_extract_command(start, size, extract_name): 96 """Generate the extract command. 97 98 The length of this string must be constant no matter what the start and end 99 offsets are so that its length can be computed before the actual command is 100 generated. 101 102 Args: 103 start: offset in bytes of the start of the wrapped file 104 size: size in bytes of the wrapped file 105 extract_name: of the file to create when extracted 106 107 """ 108 # start gets an extra character for the '+' 109 # for tail +1 is the start of the file, not +0 110 start_str = ('+%d' % (start + 1)).rjust(_MAX_OFFSET_WIDTH + 1) 111 if len(start_str) != _MAX_OFFSET_WIDTH + 1: 112 raise Exception('Start offset too large (%d)' % start) 113 114 size_str = ('%d' % size).rjust(_MAX_OFFSET_WIDTH) 115 if len(size_str) != _MAX_OFFSET_WIDTH: 116 raise Exception('Size too large (%d)' % size) 117 118 return "tail -c %s $0 | head -c %s > %s\n" % (start_str, size_str, extract_name) 119 120 121def main(argv): 122 if len(argv) != 5: 123 print('generate-self-extracting-archive.py expects exactly 4 arguments') 124 sys.exit(1) 125 126 output_filename = argv[1] 127 input_archive_filename = argv[2] 128 comment = argv[3] 129 license_filename = argv[4] 130 131 input_archive_size = os.stat(input_archive_filename).st_size 132 133 with open(license_filename, 'r') as license_file: 134 license = license_file.read() 135 136 if not license: 137 print('License file was empty') 138 sys.exit(1) 139 140 if 'SOFTWARE LICENSE AGREEMENT' not in license: 141 print('License does not look like a license') 142 sys.exit(1) 143 144 comment_line = '# %s\n' % comment 145 extract_name = os.path.basename(input_archive_filename) 146 147 # Compute the size of the header before writing the file out. This is required 148 # so that the extract command, which uses the contents offset, can be created 149 # and included inside the header. 150 header_for_size = _HEADER_TEMPLATE.format( 151 comment_line=comment_line, 152 license=license, 153 extract_command=_generate_extract_command(0, 0, extract_name), 154 ) 155 header_size = len(header_for_size.encode('utf-8')) 156 157 # write the final output 158 with open(output_filename, 'wb') as output: 159 output.write(_HEADER_TEMPLATE.format( 160 comment_line=comment_line, 161 license=license, 162 extract_command=_generate_extract_command(header_size, input_archive_size, extract_name), 163 ).encode('utf-8')) 164 165 with open(input_archive_filename, 'rb') as input_file: 166 _pipe_bytes(input_file, output) 167 168 with tempfile.TemporaryFile() as trailing_zip: 169 with zipfile.ZipFile(trailing_zip, 'w') as myzip: 170 myzip.writestr('license.txt', license, compress_type=zipfile.ZIP_STORED) 171 172 # append the trailing zip to the end of the file 173 trailing_zip.seek(0) 174 _pipe_bytes(trailing_zip, output) 175 176 umask = os.umask(0) 177 os.umask(umask) 178 os.chmod(output_filename, 0o777 & ~umask) 179 180if __name__ == "__main__": 181 main(sys.argv) 182