2020-07-16 10:57:13 +02:00
|
|
|
import * as core from '@actions/core'
|
2020-07-18 10:55:42 +02:00
|
|
|
import * as fs from 'fs'
|
2020-07-16 10:57:13 +02:00
|
|
|
import * as path from 'path'
|
|
|
|
|
|
|
|
export function getInputAsArray(
|
|
|
|
name: string,
|
|
|
|
options?: core.InputOptions
|
|
|
|
): string[] {
|
|
|
|
return getStringAsArray(core.getInput(name, options))
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getStringAsArray(str: string): string[] {
|
|
|
|
return str
|
|
|
|
.split(/[\n,]+/)
|
|
|
|
.map(s => s.trim())
|
|
|
|
.filter(x => x !== '')
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getRepoPath(relativePath?: string): string {
|
|
|
|
let githubWorkspacePath = process.env['GITHUB_WORKSPACE']
|
|
|
|
if (!githubWorkspacePath) {
|
|
|
|
throw new Error('GITHUB_WORKSPACE not defined')
|
|
|
|
}
|
|
|
|
githubWorkspacePath = path.resolve(githubWorkspacePath)
|
|
|
|
core.debug(`githubWorkspacePath: ${githubWorkspacePath}`)
|
|
|
|
|
|
|
|
let repoPath = githubWorkspacePath
|
|
|
|
if (relativePath) repoPath = path.resolve(repoPath, relativePath)
|
|
|
|
|
|
|
|
core.debug(`repoPath: ${repoPath}`)
|
|
|
|
return repoPath
|
|
|
|
}
|
|
|
|
|
|
|
|
interface RemoteDetail {
|
2022-11-24 03:52:04 +01:00
|
|
|
hostname: string
|
2020-07-16 10:57:13 +02:00
|
|
|
protocol: string
|
|
|
|
repository: string
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getRemoteDetail(remoteUrl: string): RemoteDetail {
|
|
|
|
// Parse the protocol and github repository from a URL
|
|
|
|
// e.g. HTTPS, peter-evans/create-pull-request
|
2021-01-25 19:16:19 +01:00
|
|
|
const githubUrl = process.env['GITHUB_SERVER_URL'] || 'https://github.com'
|
2021-01-25 18:25:20 +01:00
|
|
|
|
2021-01-25 19:16:19 +01:00
|
|
|
const githubServerMatch = githubUrl.match(/^https?:\/\/(.+)$/i)
|
2021-01-25 18:25:20 +01:00
|
|
|
if (!githubServerMatch) {
|
|
|
|
throw new Error('Could not parse GitHub Server name')
|
|
|
|
}
|
|
|
|
|
2022-11-24 03:52:04 +01:00
|
|
|
const hostname = githubServerMatch[1]
|
|
|
|
|
2021-01-25 18:25:20 +01:00
|
|
|
const httpsUrlPattern = new RegExp(
|
2022-11-24 03:52:04 +01:00
|
|
|
'^https?://.*@?' + hostname + '/(.+/.+?)(\\.git)?$',
|
2021-01-25 18:25:20 +01:00
|
|
|
'i'
|
|
|
|
)
|
2022-11-24 03:52:04 +01:00
|
|
|
const sshUrlPattern = new RegExp('^git@' + hostname + ':(.+/.+)\\.git$', 'i')
|
2020-07-16 10:57:13 +02:00
|
|
|
|
|
|
|
const httpsMatch = remoteUrl.match(httpsUrlPattern)
|
|
|
|
if (httpsMatch) {
|
|
|
|
return {
|
2022-11-24 03:52:04 +01:00
|
|
|
hostname,
|
2020-07-16 10:57:13 +02:00
|
|
|
protocol: 'HTTPS',
|
|
|
|
repository: httpsMatch[1]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const sshMatch = remoteUrl.match(sshUrlPattern)
|
|
|
|
if (sshMatch) {
|
|
|
|
return {
|
2022-11-24 03:52:04 +01:00
|
|
|
hostname,
|
2020-07-16 10:57:13 +02:00
|
|
|
protocol: 'SSH',
|
|
|
|
repository: sshMatch[1]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error(
|
|
|
|
`The format of '${remoteUrl}' is not a valid GitHub repository URL`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-11-24 03:52:04 +01:00
|
|
|
export function getRemoteUrl(
|
|
|
|
protocol: string,
|
|
|
|
hostname: string,
|
|
|
|
repository: string
|
|
|
|
): string {
|
2020-07-18 08:33:46 +02:00
|
|
|
return protocol == 'HTTPS'
|
2022-11-24 03:52:04 +01:00
|
|
|
? `https://${hostname}/${repository}`
|
|
|
|
: `git@${hostname}:${repository}.git`
|
2020-07-18 08:33:46 +02:00
|
|
|
}
|
|
|
|
|
2020-07-16 10:57:13 +02:00
|
|
|
export function secondsSinceEpoch(): number {
|
|
|
|
const now = new Date()
|
|
|
|
return Math.round(now.getTime() / 1000)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function randomString(): string {
|
|
|
|
return Math.random().toString(36).substr(2, 7)
|
|
|
|
}
|
|
|
|
|
|
|
|
interface DisplayNameEmail {
|
|
|
|
name: string
|
|
|
|
email: string
|
|
|
|
}
|
|
|
|
|
|
|
|
export function parseDisplayNameEmail(
|
|
|
|
displayNameEmail: string
|
|
|
|
): DisplayNameEmail {
|
|
|
|
// Parse the name and email address from a string in the following format
|
|
|
|
// Display Name <email@address.com>
|
|
|
|
const pattern = /^([^<]+)\s*<([^>]+)>$/i
|
|
|
|
|
|
|
|
// Check we have a match
|
|
|
|
const match = displayNameEmail.match(pattern)
|
|
|
|
if (!match) {
|
|
|
|
throw new Error(
|
|
|
|
`The format of '${displayNameEmail}' is not a valid email address with display name`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that name and email are not just whitespace
|
|
|
|
const name = match[1].trim()
|
|
|
|
const email = match[2].trim()
|
|
|
|
if (!name || !email) {
|
|
|
|
throw new Error(
|
|
|
|
`The format of '${displayNameEmail}' is not a valid email address with display name`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
name: name,
|
|
|
|
email: email
|
|
|
|
}
|
|
|
|
}
|
2020-07-18 10:55:42 +02:00
|
|
|
|
|
|
|
export function fileExistsSync(path: string): boolean {
|
|
|
|
if (!path) {
|
|
|
|
throw new Error("Arg 'path' must not be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
let stats: fs.Stats
|
|
|
|
try {
|
|
|
|
stats = fs.statSync(path)
|
2022-09-21 08:42:50 +02:00
|
|
|
} catch (error) {
|
|
|
|
if (hasErrorCode(error) && error.code === 'ENOENT') {
|
2020-07-18 10:55:42 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error(
|
2022-09-21 08:42:50 +02:00
|
|
|
`Encountered an error when checking whether path '${path}' exists: ${getErrorMessage(
|
|
|
|
error
|
|
|
|
)}`
|
2020-07-18 10:55:42 +02:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!stats.isDirectory()) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
2022-09-21 08:42:50 +02:00
|
|
|
|
|
|
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
|
|
function hasErrorCode(error: any): error is {code: string} {
|
|
|
|
return typeof (error && error.code) === 'string'
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getErrorMessage(error: unknown) {
|
|
|
|
if (error instanceof Error) return error.message
|
|
|
|
return String(error)
|
|
|
|
}
|