mirror of
https://github.com/actions/setup-java.git
synced 2025-04-04 10:50:13 +08:00
On self-hosted runners toolchains.xml may survive multiple runs and unexpectedly grow as a result of the toolchains setup simply appending the JDK definition even if one with the same `type` and `provides.id` already exists. Restructuring the parsing step and filtering the potentially existing list of toolchain definitions prevents this and also fixes toolchain.xml files that already contain duplicates. Fixes #530
201 lines
5.6 KiB
TypeScript
201 lines
5.6 KiB
TypeScript
import * as fs from 'fs';
|
|
import * as os from 'os';
|
|
import * as path from 'path';
|
|
import * as core from '@actions/core';
|
|
import * as io from '@actions/io';
|
|
import * as constants from './constants';
|
|
|
|
import {getBooleanInput} from './util';
|
|
import {create as xmlCreate} from 'xmlbuilder2';
|
|
|
|
interface JdkInfo {
|
|
version: string;
|
|
vendor: string;
|
|
id: string;
|
|
jdkHome: string;
|
|
}
|
|
|
|
export async function configureToolchains(
|
|
version: string,
|
|
distributionName: string,
|
|
jdkHome: string,
|
|
toolchainId?: string
|
|
) {
|
|
const vendor =
|
|
core.getInput(constants.INPUT_MVN_TOOLCHAIN_VENDOR) || distributionName;
|
|
const id = toolchainId || `${vendor}_${version}`;
|
|
const settingsDirectory =
|
|
core.getInput(constants.INPUT_SETTINGS_PATH) ||
|
|
path.join(os.homedir(), constants.M2_DIR);
|
|
const overwriteSettings = getBooleanInput(
|
|
constants.INPUT_OVERWRITE_SETTINGS,
|
|
true
|
|
);
|
|
|
|
await createToolchainsSettings({
|
|
jdkInfo: {
|
|
version,
|
|
vendor,
|
|
id,
|
|
jdkHome
|
|
},
|
|
settingsDirectory,
|
|
overwriteSettings
|
|
});
|
|
}
|
|
|
|
export async function createToolchainsSettings({
|
|
jdkInfo,
|
|
settingsDirectory,
|
|
overwriteSettings
|
|
}: {
|
|
jdkInfo: JdkInfo;
|
|
settingsDirectory: string;
|
|
overwriteSettings: boolean;
|
|
}) {
|
|
core.info(
|
|
`Creating ${constants.MVN_TOOLCHAINS_FILE} for JDK version ${jdkInfo.version} from ${jdkInfo.vendor}`
|
|
);
|
|
// when an alternate m2 location is specified use only that location (no .m2 directory)
|
|
// otherwise use the home/.m2/ path
|
|
await io.mkdirP(settingsDirectory);
|
|
const originalToolchains = await readExistingToolchainsFile(
|
|
settingsDirectory
|
|
);
|
|
const updatedToolchains = generateToolchainDefinition(
|
|
originalToolchains,
|
|
jdkInfo.version,
|
|
jdkInfo.vendor,
|
|
jdkInfo.id,
|
|
jdkInfo.jdkHome
|
|
);
|
|
await writeToolchainsFileToDisk(
|
|
settingsDirectory,
|
|
updatedToolchains,
|
|
overwriteSettings
|
|
);
|
|
}
|
|
|
|
// only exported for testing purposes
|
|
export function generateToolchainDefinition(
|
|
original: string,
|
|
version: string,
|
|
vendor: string,
|
|
id: string,
|
|
jdkHome: string
|
|
) {
|
|
let jsToolchains: Toolchain[] = [
|
|
{
|
|
type: 'jdk',
|
|
provides: {
|
|
version: `${version}`,
|
|
vendor: `${vendor}`,
|
|
id: `${id}`
|
|
},
|
|
configuration: {
|
|
jdkHome: `${jdkHome}`
|
|
}
|
|
}
|
|
];
|
|
if (original?.length) {
|
|
// convert existing toolchains into TS native objects for better handling
|
|
// xmlbuilder2 will convert the document into a `{toolchains: { toolchain: [] | {} }}` structure
|
|
// instead of the desired `toolchains: [{}]` one or simply `[{}]`
|
|
const jsObj = xmlCreate(original)
|
|
.root()
|
|
.toObject() as unknown as ExtractedToolchains;
|
|
if (jsObj.toolchains && jsObj.toolchains.toolchain) {
|
|
// in case only a single child exists xmlbuilder2 will not create an array and using verbose = true equally doesn't work here
|
|
// See https://oozcitak.github.io/xmlbuilder2/serialization.html#js-object-and-map-serializers for details
|
|
if (Array.isArray(jsObj.toolchains.toolchain)) {
|
|
jsToolchains.push(...jsObj.toolchains.toolchain);
|
|
} else {
|
|
jsToolchains.push(jsObj.toolchains.toolchain);
|
|
}
|
|
}
|
|
|
|
// remove potential duplicates based on type & id (which should be a unique combination);
|
|
// self.findIndex will only return the first occurrence, ensuring duplicates are skipped
|
|
jsToolchains = jsToolchains.filter(
|
|
(value, index, self) =>
|
|
// ensure non-jdk toolchains are kept in the results, we must not touch them because they belong to the user
|
|
value.type !== 'jdk' ||
|
|
index ===
|
|
self.findIndex(
|
|
t => t.type === value.type && t.provides.id === value.provides.id
|
|
)
|
|
);
|
|
}
|
|
|
|
// TODO: technically bad because we shouldn't re-create the toolchains root node (with possibly different schema values) if it already exists, however, just overriding the toolchain array with xmlbuilder2 is … uh non-trivial
|
|
return xmlCreate({
|
|
toolchains: {
|
|
'@xmlns': 'http://maven.apache.org/TOOLCHAINS/1.1.0',
|
|
'@xmlns:xsi': 'http://www.w3.org/2001/XMLSchema-instance',
|
|
'@xsi:schemaLocation':
|
|
'http://maven.apache.org/TOOLCHAINS/1.1.0 https://maven.apache.org/xsd/toolchains-1.1.0.xsd',
|
|
toolchain: jsToolchains
|
|
}
|
|
}).end({
|
|
format: 'xml',
|
|
wellFormed: false,
|
|
headless: false,
|
|
prettyPrint: true,
|
|
width: 80
|
|
});
|
|
}
|
|
|
|
async function readExistingToolchainsFile(directory: string) {
|
|
const location = path.join(directory, constants.MVN_TOOLCHAINS_FILE);
|
|
if (fs.existsSync(location)) {
|
|
return fs.readFileSync(location, {
|
|
encoding: 'utf-8',
|
|
flag: 'r'
|
|
});
|
|
}
|
|
return '';
|
|
}
|
|
|
|
async function writeToolchainsFileToDisk(
|
|
directory: string,
|
|
settings: string,
|
|
overwriteSettings: boolean
|
|
) {
|
|
const location = path.join(directory, constants.MVN_TOOLCHAINS_FILE);
|
|
const settingsExists = fs.existsSync(location);
|
|
if (settingsExists && overwriteSettings) {
|
|
core.info(`Overwriting existing file ${location}`);
|
|
} else if (!settingsExists) {
|
|
core.info(`Writing to ${location}`);
|
|
} else {
|
|
core.info(
|
|
`Skipping generation of ${location} because file already exists and overwriting is not enabled`
|
|
);
|
|
return;
|
|
}
|
|
|
|
return fs.writeFileSync(location, settings, {
|
|
encoding: 'utf-8',
|
|
flag: 'w'
|
|
});
|
|
}
|
|
|
|
interface ExtractedToolchains {
|
|
toolchains: {
|
|
toolchain: Toolchain[] | Toolchain;
|
|
};
|
|
}
|
|
|
|
// Toolchain type definition according to Maven Toolchains XSD 1.1.0
|
|
interface Toolchain {
|
|
type: string;
|
|
provides:
|
|
| {
|
|
version: string;
|
|
vendor: string;
|
|
id: string;
|
|
}
|
|
| any;
|
|
configuration: any;
|
|
}
|