blob: f0afcfcbf945600a1cfdf183c61c806c8fd64b53 [file] [log] [blame]
Bartek Grzybowskib9d8c4c2019-07-11 08:31:56 +02001#! /usr/bin/env python3
Mateusz Pilat84ef3b32019-06-17 07:45:24 +02002# -*- coding: utf-8 -*-
3
4# COPYRIGHT NOTICE STARTS HERE
5
6# Copyright 2019 . Samsung Electronics Co., Ltd.
7#
8# Licensed under the Apache License, Version 2.0 (the "License");
9# you may not use this file except in compliance with the License.
10# You may obtain a copy of the License at
11#
12# http://www.apache.org/licenses/LICENSE-2.0
13#
14# Unless required by applicable law or agreed to in writing, software
15# distributed under the License is distributed on an "AS IS" BASIS,
16# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17# See the License for the specific language governing permissions and
18# limitations under the License.
19
20# COPYRIGHT NOTICE ENDS HERE
21
22from datetime import datetime
23import subprocess
24import argparse
25import logging
26import shutil
27import glob
28import json
29import sys
30import os
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +010031import hashlib
Mateusz Pilat84ef3b32019-06-17 07:45:24 +020032
33import tarfile
34import git
35
36log = logging.getLogger(__name__)
Denis Kasanic32ecaae2019-11-14 10:43:56 +010037script_location = os.path.abspath(os.path.join(__file__, '..'))
38offline_repository_dir = os.path.abspath(os.path.join(script_location, '..'))
Mateusz Pilat84ef3b32019-06-17 07:45:24 +020039
40
41def prepare_application_repository(directory, url, refspec, patch_path):
42 """
43 Downloads git repository according to refspec, applies patch if provided
44 :param directory: path to repository
45 :param url: url to repository
46 :param refspec: refspec to fetch
47 :param patch_path: path git patch to be applied over repository
48 :return: repository - git repository object
49 """
50
51 try:
52 shutil.rmtree(directory)
53 except FileNotFoundError:
54 pass
55
56 log.info('Cloning {} with refspec {} '.format(url, refspec))
57 repository = git.Repo.init(directory)
58 origin = repository.create_remote('origin', url)
59 origin.pull(refspec)
60 repository.git.submodule('update', '--init')
61
62 if patch_path:
63 log.info('Applying {} over {} {}'.format(patch_path,
64 url,
65 refspec))
66 repository.git.apply(patch_path)
67 else:
68 log.info('No patch file provided, skipping patching')
69
70 return repository
71
72
Ondřej Šmalec6e594412020-01-22 07:38:29 +010073def create_package_info_file(output_file, repository_list, tag, metadata):
Mateusz Pilat84ef3b32019-06-17 07:45:24 +020074 """
75 Generates text file in json format containing basic information about the build
76 :param output_file:
77 :param repository_list: list of repositories to be included in package info
Denis Kasanic32ecaae2019-11-14 10:43:56 +010078 :param tag: build version of packages
Ondřej Šmalec6e594412020-01-22 07:38:29 +010079 :param metadata: additional metadata into package.info
Mateusz Pilat84ef3b32019-06-17 07:45:24 +020080 :return:
81 """
82 log.info('Generating package.info file')
83 build_info = {
84 'Build_info': {
Mateusz Pilatf7d2f572019-08-29 13:28:56 +020085 'build_date': datetime.now().strftime('%Y-%m-%d_%H-%M'),
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +010086 'Version': tag,
87 'Packages': {}
Mateusz Pilat84ef3b32019-06-17 07:45:24 +020088 }
89 }
90 for repository in repository_list:
91 build_info['Build_info'][
92 repository.config_reader().get_value('remote "origin"', 'url')] = repository.head.commit.hexsha
93
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +010094 if metadata:
95 for meta in metadata:
96 build_info['Build_info'].update(meta)
Ondřej Šmalec6e594412020-01-22 07:38:29 +010097
Mateusz Pilat84ef3b32019-06-17 07:45:24 +020098 with open(output_file, 'w') as outfile:
99 json.dump(build_info, outfile, indent=4)
100
101
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +0100102def add_checksum_info(output_dir):
103 """
104 Add checksum information into package.info file
105 :param output_dir: directory where are packages
106 """
107 tar_files = ['resources_package.tar', 'aux_package.tar', 'sw_package.tar']
108 for tar_file in tar_files:
109 try:
Mateusz Pilatf1d98162020-02-14 18:37:35 +0100110 checksum = hashlib.md5()
111 with open(os.path.join(output_dir, tar_file), 'rb') as f:
112 for chunk in iter(lambda: f.read(4096), b""):
113 checksum.update(chunk)
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +0100114 with open(os.path.join(output_dir, 'package.info'), 'r') as f:
115 json_data = json.load(f)
Mateusz Pilatf1d98162020-02-14 18:37:35 +0100116 json_data['Build_info']['Packages'].update({tar_file: checksum.hexdigest()})
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +0100117 with open(os.path.join(output_dir, 'package.info'), 'w') as f:
118 json.dump(json_data, f, indent=4)
119 except FileNotFoundError:
120 pass
121
122
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200123def create_package(tar_content, file_name):
124 """
125 Creates packages
126 :param tar_content: list of dictionaries defining src file and destination tar file
127 :param file_name: output file
128 """
129 log.info('Creating package {}'.format(file_name))
130 with tarfile.open(file_name, 'w') as output_tar_file:
131 for src, dst in tar_content.items():
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200132 if src != '':
Denis Kasanic32ecaae2019-11-14 10:43:56 +0100133 output_tar_file.add(src, dst)
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200134
135
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +0100136def metadata_validation(param):
137 """
138 Validation of metadata parameters
139 :param param: parameter to be checked needs to be in format key=value
140 """
141 try:
142 key, value = param.split('=')
143 assert (key and value)
144 return {key: value}
145 except (ValueError, AssertionError):
146 msg = "%r is not a valid parameter. Needs to be in format key=value" % param
147 raise argparse.ArgumentTypeError(msg)
148
149
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200150def build_offline_deliverables(build_version,
151 application_repository_url,
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200152 application_repository_reference,
153 application_patch_file,
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200154 application_charts_dir,
155 application_configuration,
156 application_patch_role,
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200157 output_dir,
158 resources_directory,
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200159 aux_directory,
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200160 skip_sw,
161 skip_resources,
162 skip_aux,
Ondřej Šmalec6e594412020-01-22 07:38:29 +0100163 overwrite,
164 metadata):
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200165 """
166 Prepares offline deliverables
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200167 :param build_version: Version for packages tagging
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200168 :param application_repository_url: git repository hosting application helm charts
169 :param application_repository_reference: git refspec for repository hosting application helm charts
170 :param application_patch_file: git patch file to be applied over application repository
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200171 :param application_charts_dir: path to directory under application repository containing helm charts
172 :param application_configuration: path to application configuration file (helm override configuration)
173 :param application_patch_role: path to application patch role (executed just before helm deploy)
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200174 :param output_dir: Destination directory for saving packages
175 :param resources_directory: Path to resource directory
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200176 :param aux_directory: Path to aux binary directory
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200177 :param skip_sw: skip sw package generation
178 :param skip_resources: skip resources package generation
179 :param skip_aux: skip aux package generation
180 :param overwrite: overwrite files in output directory
Ondřej Šmalec6e594412020-01-22 07:38:29 +0100181 :param metadata: add metadata info into package.info
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200182 :return:
183 """
184
185 if os.path.exists(output_dir) and os.listdir(output_dir):
186 if not overwrite:
187 log.error('Output directory is not empty, use overwrite to force build')
Denis Kasanic32ecaae2019-11-14 10:43:56 +0100188 raise FileExistsError(output_dir)
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200189 shutil.rmtree(output_dir)
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200190
191 # Git
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200192 offline_repository = git.Repo(offline_repository_dir)
193
194 application_dir = os.path.join(output_dir, 'application_repository')
195 application_repository = prepare_application_repository(application_dir,
196 application_repository_url,
197 application_repository_reference,
198 application_patch_file)
199
200 # Package info
201 info_file = os.path.join(output_dir, 'package.info')
Ondřej Šmalec6e594412020-01-22 07:38:29 +0100202 create_package_info_file(info_file, [application_repository, offline_repository], build_version, metadata)
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200203
204 # packages layout as dictionaries. <file> : <file location under tar archive>
205 sw_content = {
206 os.path.join(offline_repository_dir, 'ansible'): 'ansible',
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200207 application_configuration: 'ansible/application/application_configuration.yml',
208 application_patch_role: 'ansible/application/onap-patch-role',
209 os.path.join(application_dir, application_charts_dir): 'ansible/application/helm_charts',
210 info_file: 'package.info'
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200211 }
212 resources_content = {
213 resources_directory: '',
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200214 info_file: 'package.info'
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200215 }
216 aux_content = {
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200217 aux_directory: '',
218 info_file: 'package.info'
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200219 }
220
221 if not skip_sw:
222 log.info('Building offline installer')
223 os.chdir(os.path.join(offline_repository_dir, 'ansible', 'docker'))
224 installer_build = subprocess.run(
225 os.path.join(offline_repository_dir, 'ansible', 'docker', 'build_ansible_image.sh'))
226 installer_build.check_returncode()
227 os.chdir(script_location)
Ondřej Šmalec47c59052020-01-24 14:03:38 +0100228 sw_package_tar_path = os.path.join(output_dir, 'sw_package.tar')
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200229 create_package(sw_content, sw_package_tar_path)
230
231 if not skip_resources:
232 log.info('Building own dns image')
233 dns_build = subprocess.run([
234 os.path.join(offline_repository_dir, 'build', 'creating_data', 'create_nginx_image', '01create-image.sh'),
235 os.path.join(resources_directory, 'offline_data', 'docker_images_infra')])
236 dns_build.check_returncode()
237
238 # Workaround for downloading without "flat" option
239 log.info('Binaries - workaround')
240 download_dir_path = os.path.join(resources_directory, 'downloads')
241 os.chdir(download_dir_path)
Denis Kasanic32ecaae2019-11-14 10:43:56 +0100242 for file in os.listdir(download_dir_path):
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200243 if os.path.islink(file):
244 os.unlink(file)
245
Bartek Grzybowskibcc82e62021-02-03 10:33:14 +0100246 bin_pattern_list = ['**/rke_linux-amd64',
247 '**/helm-*-linux-amd64.tar.gz',
248 '**/kubectl',
Bartek Grzybowski3f3a7a12021-05-31 14:25:19 +0200249 '**/helm-push_*_linux_amd64.tar.gz',
Bartek Grzybowski1e416742021-11-10 12:18:38 +0100250 '**/kube-prometheus-stack-*.tgz',
Bartek Grzybowski7dafe0b2021-11-15 14:27:23 +0100251 '**/cert-manager-*.tgz',
252 '**/cmctl-linux-amd64.tar.gz']
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200253
Bartek Grzybowskibcc82e62021-02-03 10:33:14 +0100254 for pattern in bin_pattern_list:
255 for bin_file in glob.glob(os.path.join('.', pattern), recursive=True):
256 os.symlink(bin_file, os.path.join(download_dir_path, bin_file.split('/')[-1]))
Bartek Grzybowskia6462282021-01-29 14:52:13 +0100257
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200258 os.chdir(script_location)
259 # End of workaround
260
Ondřej Šmalec47c59052020-01-24 14:03:38 +0100261 resources_package_tar_path = os.path.join(output_dir, 'resources_package.tar')
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200262 create_package(resources_content, resources_package_tar_path)
263
264 if not skip_aux:
Ondřej Šmalec47c59052020-01-24 14:03:38 +0100265 aux_package_tar_path = os.path.join(output_dir, 'aux_package.tar')
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200266 create_package(aux_content, aux_package_tar_path)
267
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +0100268 add_checksum_info(output_dir)
Bartek Grzybowski2240a2a2020-09-23 15:50:15 +0200269 shutil.rmtree(application_dir, ignore_errors=True)
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200270
271
272def run_cli():
273 """
274 Run as cli tool
275 """
276 parser = argparse.ArgumentParser(description='Create Package For Offline Installer')
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200277 parser.add_argument('--build-version',
Denis Kasanic32ecaae2019-11-14 10:43:56 +0100278 help='version of the build', default='')
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200279 parser.add_argument('application_repository_url', metavar='application-repository-url',
280 help='git repository hosting application helm charts')
281 parser.add_argument('--application-repository_reference', default='master',
282 help='git refspec for repository hosting application helm charts')
283 parser.add_argument('--application-patch_file',
284 help='git patch file to be applied over application repository', default='')
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200285 parser.add_argument('--application-charts_dir',
Denis Kasanic32ecaae2019-11-14 10:43:56 +0100286 help='path to directory under application repository containing helm charts ',
287 default='kubernetes')
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200288 parser.add_argument('--application-configuration',
289 help='path to application configuration file (helm override configuration)',
Denis Kasanic32ecaae2019-11-14 10:43:56 +0100290 default=os.path.join(offline_repository_dir, 'config/application_configuration.yml'))
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200291 parser.add_argument('--application-patch-role',
292 help='path to application patch role file (ansible role) to be executed right before installation',
293 default='')
Denis Kasanic32ecaae2019-11-14 10:43:56 +0100294 parser.add_argument('--output-dir', '-o', default=os.path.join(offline_repository_dir, '../packages'),
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200295 help='Destination directory for saving packages')
Denis Kasanic32ecaae2019-11-14 10:43:56 +0100296 parser.add_argument('--resources-directory', default=os.path.join(offline_repository_dir, '../resources'),
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200297 help='Path to resource directory')
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200298 parser.add_argument('--aux-directory',
299 help='Path to aux binary directory', default='')
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200300 parser.add_argument('--skip-sw', action='store_true', default=False,
301 help='Set to skip sw package generation')
302 parser.add_argument('--skip-resources', action='store_true', default=False,
303 help='Set to skip resources package generation')
304 parser.add_argument('--skip-aux', action='store_true', default=False,
305 help='Set to skip aux package generation')
306 parser.add_argument('--overwrite', action='store_true', default=False,
307 help='overwrite files in output directory')
308 parser.add_argument('--debug', action='store_true', default=False,
309 help='Turn on debug output')
Ondřej Šmalecd8b17a12020-02-04 15:30:48 +0100310 parser.add_argument('--add-metadata', nargs="+", type=metadata_validation,
311 help='additional metadata added into package.info, format: key=value')
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200312 args = parser.parse_args()
313
314 if args.debug:
315 logging.basicConfig(stream=sys.stdout, level=logging.DEBUG)
316 else:
317 logging.basicConfig(stream=sys.stdout, level=logging.INFO, format='%(message)s')
318
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200319 build_offline_deliverables(args.build_version,
320 args.application_repository_url,
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200321 args.application_repository_reference,
322 args.application_patch_file,
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200323 args.application_charts_dir,
324 args.application_configuration,
325 args.application_patch_role,
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200326 args.output_dir,
327 args.resources_directory,
Mateusz Pilatf7d2f572019-08-29 13:28:56 +0200328 args.aux_directory,
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200329 args.skip_sw,
330 args.skip_resources,
331 args.skip_aux,
Ondřej Šmalec6e594412020-01-22 07:38:29 +0100332 args.overwrite,
333 args.add_metadata)
Mateusz Pilat84ef3b32019-06-17 07:45:24 +0200334
335
336if __name__ == '__main__':
337 run_cli()