Files
DistMaker/script/linuxUtils.py
Norberto Lopez 48e4ea799b Changes associated with version 0.70:
[1] Updated to Java version 17.
[2] Updated Glum library version 2.0.0.
[3] Updates relating to the public release of DistMaker.
2024-09-27 19:23:51 -04:00

198 lines
7.3 KiB
Python

#! /usr/bin/env python3
# Copyright (C) 2024 The Johns Hopkins University Applied Physics Laboratory LLC
#
# 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 copy
import os
import shutil
import subprocess
import tempfile
import glob
import jreUtils
import miscUtils
import deployJreDist
def buildRelease(aArgs, aBuildPath, aJreNodeL):
# We mutate args - thus make a custom copy
args = copy.copy(aArgs)
# Retrieve vars of interest
appName = args.name
version = args.version
jreVerSpec = args.jreVerSpec
archStr = 'x64'
platStr = 'linux'
# Determine the types of builds we should do
platformType = miscUtils.getPlatformTypes(args.platform, platStr)
if platformType.nonJre == False and platformType.withJre == False:
return;
# Check our system environment before proceeding
if checkSystemEnvironment() == False:
return
# Form the list of distributions to build (dynamic and static JREs)
distL = []
if platformType.nonJre == True:
distL = [(appName + '-' + version, None)]
if platformType.withJre == True:
# Select the JreNode to utilize for static releases
tmpJreNode = jreUtils.getJreNode(aJreNodeL, archStr, platStr, jreVerSpec)
if tmpJreNode == None:
# Let the user know that a compatible JRE was not found - thus no static release will be made.
print('[Warning] No compatible JRE ({}) is available for the ({}) {} platform. A static release will not be provided for the platform.'.format(jreVerSpec, archStr, platStr.capitalize()))
else:
distL.append((appName + '-' + version + '-jre', tmpJreNode))
# Create a tmp (working) folder
tmpPath = tempfile.mkdtemp(prefix=platStr, dir=aBuildPath)
# Create the various distributions
for (aDistName, aJreNode) in distL:
print('Building {} distribution: {}'.format(platStr.capitalize(), aDistName))
# Let the user know of the JRE release we are going to build with
if aJreNode != None:
print('\tUtilizing JRE: ' + aJreNode.getFile())
# Create the (top level) distribution folder
dstPath = os.path.join(tmpPath, aDistName)
os.mkdir(dstPath)
# Build the contents of the distribution folder
buildDistTree(aBuildPath, dstPath, args, aJreNode)
# Create the tar.gz archive
tarFile = os.path.join(aBuildPath, aDistName + '.tar.gz')
print('\tForming tar.gz file: ' + tarFile)
childPath = aDistName
subprocess.check_call(["tar", "-czf", tarFile, "-C", tmpPath, childPath], stderr=subprocess.STDOUT)
print('\tFinished building release: ' + os.path.basename(tarFile) + '\n')
# Perform cleanup
shutil.rmtree(tmpPath)
def buildDistTree(aBuildPath, aRootPath, aArgs, aJreNode):
# Retrieve vars of interest
appInstallRoot = miscUtils.getInstallRoot()
appInstallRoot = os.path.dirname(appInstallRoot)
appName = aArgs.name
# Form the app contents folder
srcPath = os.path.join(aBuildPath, "delta")
dstPath = os.path.join(aRootPath, "app")
shutil.copytree(srcPath, dstPath, symlinks=True)
# Copy libs to the app directory so they can be found at launch
soDir = os.path.join(aRootPath, 'app', 'code', 'linux')
for libPath in glob.iglob(os.path.join(soDir, "*.so")):
libFileName = os.path.basename(libPath)
srcPath = os.path.join(soDir, libFileName)
linkPath = os.path.join(dstPath, libFileName)
shutil.copy(srcPath, linkPath)
# Setup the launcher contents
dstPath = os.path.join(aRootPath, "launcher/" + deployJreDist.getAppLauncherFileName())
srcPath = os.path.join(appInstallRoot, "template/appLauncher.jar")
os.makedirs(os.path.dirname(dstPath))
shutil.copy(srcPath, dstPath);
# Build the java component of the distribution
if aArgs.javaCode != None:
# Form the executable bash script
dstPath = os.path.join(aRootPath, 'run' + appName)
buildBashScript(dstPath, aArgs, aJreNode)
# Unpack the JRE and set up the JRE tree
if aJreNode != None:
jreUtils.unpackAndRenameToStandard(aJreNode, aRootPath)
def buildBashScript(aDstFile, aArgs, aJreNode):
# Form the jvmArgStr but strip away the -Xmx* component if it is specified
# since the JVM maxMem is dynamically configurable (via DistMaker)
maxMem = None
jvmArgsStr = ''
for aStr in aArgs.jvmArgs:
if aStr.startswith('-Xmx'):
maxMem = aStr[4:]
else:
jvmArgsStr += aStr + ' '
with open(aDstFile, mode='wt', encoding='utf-8', newline='\n') as tmpFO:
# tmpFO.write('#!/bin/bash\n')
tmpFO.write('#!/usr/bin/env bash\n')
tmpFO.write('# Do not remove the opening or closing brackets: {}. This enables safe inline\n')
tmpFO.write('# mutations to this script while it is running\n')
tmpFO.write('{ # Do not remove this bracket! \n\n')
tmpFO.write('# Define where the Java executable is located\n')
if aJreNode == None:
tmpFO.write('javaExe=java\n\n')
else:
jrePath = jreUtils.getBasePathFor(aJreNode)
tmpFO.write('javaExe=../' + jrePath + '/bin/java\n\n')
tmpFO.write('# Define the maximum memory to allow the application to utilize\n')
if maxMem == None:
tmpFO.write('#maxMem=512m # Uncomment out this line to change from defaults.\n\n')
else:
tmpFO.write('maxMem=' + maxMem + '\n\n')
tmpFO.write('# Get the installation path\n')
tmpFO.write('# We support the Linux / Macosx variants explicitly and then default back to Linux\n')
tmpFO.write('if [ "$(uname -s)" == "Darwin" ]; then\n')
tmpFO.write(' # Macosx platform: We assume the coreutils package has been installed...\n')
tmpFO.write(' installPath=$(greadlink -f "$BASH_SOURCE")\n')
tmpFO.write('elif [ "$(uname -s)" == "Linux" ]; then\n')
tmpFO.write(' # Linux platform\n')
tmpFO.write(' installPath=$(readlink -f "$BASH_SOURCE")\n')
tmpFO.write('else\n')
tmpFO.write(' # Other platform: ---> Defaults back to Linux platform\n')
tmpFO.write(' installPath=$(readlink -f "$BASH_SOURCE")\n')
tmpFO.write('fi\n')
tmpFO.write('installPath=$(dirname "$installPath")\n\n')
tmpFO.write('# Change the working directory to the app folder in the installation path\n')
tmpFO.write('cd "$installPath"/app\n\n')
tmpFO.write('# Setup the xmxStr to define the maximum JVM memory.\n')
tmpFO.write('if [ -z ${maxMem+x} ]; then\n')
tmpFO.write(' xmxStr=""\n')
tmpFO.write('else\n')
tmpFO.write(' xmxStr=\'-Xmx\'$maxMem\n')
tmpFO.write('fi\n\n')
exeCmd = '$javaExe ' + jvmArgsStr + '$xmxStr -Djava.system.class.loader=appLauncher.RootClassLoader '
exeCmd = exeCmd + '-cp ../launcher/' + deployJreDist.getAppLauncherFileName() + ' appLauncher.AppLauncher $*'
tmpFO.write('# Run the application\n')
tmpFO.write(exeCmd + '\n\n')
tmpFO.write('exit # Do not remove this exit! (just before the bracket)\n')
tmpFO.write('} # Do not remove this bracket! \n\n')
# Make the script executable
os.chmod(aDstFile, 0o755)
def checkSystemEnvironment():
"""Checks to ensure that all system application / environment variables needed to build a Linux distribution are installed
and properly configured. Returns False if the system environment is insufficient"""
return True