2026-01-30 13:31:20 +01:00
|
|
|
import * as os from 'os';
|
|
|
|
|
import { toCommandValue } from './utils.js';
|
2020-02-29 23:01:03 +01:00
|
|
|
/**
|
2025-12-25 10:58:28 +01:00
|
|
|
* Issues a command to the GitHub Actions runner
|
|
|
|
|
*
|
|
|
|
|
* @param command - The command name to issue
|
|
|
|
|
* @param properties - Additional properties for the command (key-value pairs)
|
|
|
|
|
* @param message - The message to include with the command
|
|
|
|
|
* @remarks
|
|
|
|
|
* This function outputs a specially formatted string to stdout that the Actions
|
|
|
|
|
* runner interprets as a command. These commands can control workflow behavior,
|
|
|
|
|
* set outputs, create annotations, mask values, and more.
|
2020-02-29 23:01:03 +01:00
|
|
|
*
|
|
|
|
|
* Command Format:
|
|
|
|
|
* ::name key=value,key=value::message
|
|
|
|
|
*
|
2025-12-25 10:58:28 +01:00
|
|
|
* @example
|
|
|
|
|
* ```typescript
|
|
|
|
|
* // Issue a warning annotation
|
|
|
|
|
* issueCommand('warning', {}, 'This is a warning message');
|
|
|
|
|
* // Output: ::warning::This is a warning message
|
|
|
|
|
*
|
|
|
|
|
* // Set an environment variable
|
|
|
|
|
* issueCommand('set-env', { name: 'MY_VAR' }, 'some value');
|
|
|
|
|
* // Output: ::set-env name=MY_VAR::some value
|
|
|
|
|
*
|
|
|
|
|
* // Add a secret mask
|
|
|
|
|
* issueCommand('add-mask', {}, 'secretValue123');
|
|
|
|
|
* // Output: ::add-mask::secretValue123
|
|
|
|
|
* ```
|
|
|
|
|
*
|
|
|
|
|
* @internal
|
|
|
|
|
* This is an internal utility function that powers the public API functions
|
|
|
|
|
* such as setSecret, warning, error, and exportVariable.
|
2020-02-29 23:01:03 +01:00
|
|
|
*/
|
2026-01-30 13:31:20 +01:00
|
|
|
export function issueCommand(command, properties, message) {
|
2020-02-29 23:01:03 +01:00
|
|
|
const cmd = new Command(command, properties, message);
|
|
|
|
|
process.stdout.write(cmd.toString() + os.EOL);
|
|
|
|
|
}
|
2026-01-30 13:31:20 +01:00
|
|
|
export function issue(name, message = '') {
|
2020-02-29 23:01:03 +01:00
|
|
|
issueCommand(name, {}, message);
|
|
|
|
|
}
|
|
|
|
|
const CMD_STRING = '::';
|
|
|
|
|
class Command {
|
|
|
|
|
constructor(command, properties, message) {
|
|
|
|
|
if (!command) {
|
|
|
|
|
command = 'missing.command';
|
|
|
|
|
}
|
|
|
|
|
this.command = command;
|
|
|
|
|
this.properties = properties;
|
|
|
|
|
this.message = message;
|
|
|
|
|
}
|
|
|
|
|
toString() {
|
|
|
|
|
let cmdStr = CMD_STRING + this.command;
|
|
|
|
|
if (this.properties && Object.keys(this.properties).length > 0) {
|
|
|
|
|
cmdStr += ' ';
|
|
|
|
|
let first = true;
|
|
|
|
|
for (const key in this.properties) {
|
|
|
|
|
if (this.properties.hasOwnProperty(key)) {
|
|
|
|
|
const val = this.properties[key];
|
|
|
|
|
if (val) {
|
|
|
|
|
if (first) {
|
|
|
|
|
first = false;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
cmdStr += ',';
|
|
|
|
|
}
|
|
|
|
|
cmdStr += `${key}=${escapeProperty(val)}`;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
cmdStr += `${CMD_STRING}${escapeData(this.message)}`;
|
|
|
|
|
return cmdStr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
function escapeData(s) {
|
2026-01-30 13:31:20 +01:00
|
|
|
return toCommandValue(s)
|
2020-02-29 23:01:03 +01:00
|
|
|
.replace(/%/g, '%25')
|
|
|
|
|
.replace(/\r/g, '%0D')
|
|
|
|
|
.replace(/\n/g, '%0A');
|
|
|
|
|
}
|
|
|
|
|
function escapeProperty(s) {
|
2026-01-30 13:31:20 +01:00
|
|
|
return toCommandValue(s)
|
2020-02-29 23:01:03 +01:00
|
|
|
.replace(/%/g, '%25')
|
|
|
|
|
.replace(/\r/g, '%0D')
|
|
|
|
|
.replace(/\n/g, '%0A')
|
|
|
|
|
.replace(/:/g, '%3A')
|
|
|
|
|
.replace(/,/g, '%2C');
|
|
|
|
|
}
|
|
|
|
|
//# sourceMappingURL=command.js.map
|