From 145c80cf8511e096eef4420b6a2aa5f2679102dc Mon Sep 17 00:00:00 2001 From: "Paulo F. Oliveira" Date: Sat, 24 Jun 2023 17:27:31 +0100 Subject: [PATCH] Test Erlang/OTP 26 + Elixir 1.14 on top Windows (#191) * Add more stuff to test on * Tweak our debugging (more verbose when the runner is in debug mode) * Hopefully make it easier to debug * Have the distribution work :-) * Adapt to a better reality --- .github/workflows/ubuntu.yml | 10 +++- .github/workflows/windows.yml | 13 +++++ dist/index.js | 96 ++++++++++++++++++++++++++++------- src/installer.js | 70 +++++++++++++++++++++---- src/setup-beam.js | 26 +++++++--- test/setup-beam.test.js | 21 ++++---- 6 files changed, 190 insertions(+), 46 deletions(-) diff --git a/.github/workflows/ubuntu.yml b/.github/workflows/ubuntu.yml index b0b0638b..1cf4673c 100644 --- a/.github/workflows/ubuntu.yml +++ b/.github/workflows/ubuntu.yml @@ -20,6 +20,14 @@ jobs: fail-fast: false matrix: combo: + - otp-version: '26.0' + elixir-version: 'v1.14-otp-25' + os: 'ubuntu-latest' + version-type: 'strict' + - otp-version: '26.0' + elixir-version: '1.14.5' + os: 'ubuntu-latest' + version-type: 'strict' - otp-version: '25.2' elixir-version: '1.14.3' os: 'ubuntu-latest' @@ -106,7 +114,7 @@ jobs: - elixir-version: '1.10.3' otp-version: '22.3.4.1' os: 'ubuntu-20.04' - - elixir-version: 'master' + - elixir-version: 'main' otp-version: '23.1' os: 'ubuntu-20.04' - elixir-version: 'main' diff --git a/.github/workflows/windows.yml b/.github/workflows/windows.yml index 350c7703..e8566e47 100644 --- a/.github/workflows/windows.yml +++ b/.github/workflows/windows.yml @@ -20,6 +20,18 @@ jobs: fail-fast: false matrix: combo: + - otp-version: '26.0' + elixir-version: 'main' + os: 'windows-latest' + version-type: 'strict' + - otp-version: '26.0' + elixir-version: 'v1.14-otp-25' + os: 'windows-latest' + version-type: 'strict' + - otp-version: '26.0' + elixir-version: '1.14.5' + os: 'windows-latest' + version-type: 'strict' - otp-version: '25' rebar3-version: 'nightly' os: 'windows-latest' @@ -61,6 +73,7 @@ jobs: elixir-version: ${{matrix.combo.elixir-version}} gleam-version: ${{matrix.combo.gleam-version}} rebar3-version: ${{matrix.combo.rebar3-version}} + version-type: ${{matrix.combo.version-type}} - name: Erlang/OTP version (action) run: echo "Erlang/OTP ${{steps.setup-beam.outputs.otp-version}}" - name: Elixir version (action) diff --git a/dist/index.js b/dist/index.js index 56a65a6a..bcc93d0b 100644 --- a/dist/index.js +++ b/dist/index.js @@ -9719,6 +9719,9 @@ const os = __nccwpck_require__(2037) * @param {string[]} hexMirrors */ async function installOTP(osVersion, otpVersion, hexMirrors) { + let cmd + let args + if (hexMirrors.length === 0) { throw new Error( `Could not install Erlang/OTP ${otpVersion} from any hex.pm mirror`, @@ -9743,7 +9746,9 @@ async function installOTP(osVersion, otpVersion, hexMirrors) { cachedPath = await tc.cacheDir(extractPath, 'otp', fullVersion) } - await exec(path.join(cachedPath, 'Install'), ['-minimal', cachedPath]) + cmd = path.join(cachedPath, 'Install') + args = ['-minimal', cachedPath] + await exec(cmd, args) const otpPath = path.join(cachedPath, 'bin') @@ -9751,7 +9756,9 @@ async function installOTP(osVersion, otpVersion, hexMirrors) { core.exportVariable('INSTALL_DIR_FOR_OTP', cachedPath) core.info('Installed Erlang/OTP version') - await exec(path.join(otpPath, 'erl'), ['-version']) + cmd = path.join(otpPath, 'erl') + args = ['-version'] + await exec(cmd, args) } else if (OS === 'win32') { if (!cachedPath) { const exePath = await tc.downloadTool( @@ -9765,13 +9772,19 @@ async function installOTP(osVersion, otpVersion, hexMirrors) { const otpPath = path.join(otpDir, 'bin') await fs.promises.mkdir(otpDir, { recursive: true }) - await exec(path.join(cachedPath, 'otp.exe'), ['/S', `/D=${otpDir}`]) + + cmd = path.join(cachedPath, 'otp.exe') + args = ['/S', `/D=${otpDir}`] + await exec(cmd, args) core.addPath(otpPath) core.exportVariable('INSTALL_DIR_FOR_OTP', otpDir) core.info('Installed Erlang/OTP version') - await exec(path.join(otpPath, 'erl'), ['+V']) + + cmd = path.join(otpPath, 'erl.exe') + args = ['+V'] + await exec(cmd, args) } } catch (err) { core.info(`Install OTP failed for mirror ${hexMirror}`) @@ -9787,6 +9800,10 @@ async function installOTP(osVersion, otpVersion, hexMirrors) { * @param {string[]} hexMirrors */ async function installElixir(elixirVersion, hexMirrors) { + let cmd + let args + let options + if (hexMirrors.length === 0) { throw new Error( `Could not install Elixir ${elixirVersion} from any hex.pm mirror`, @@ -9796,6 +9813,7 @@ async function installElixir(elixirVersion, hexMirrors) { try { let cachedPath = tc.find('elixir', elixirVersion) + const OS = process.platform if (!cachedPath) { const zipPath = await tc.downloadTool( @@ -9813,7 +9831,21 @@ async function installElixir(elixirVersion, hexMirrors) { core.exportVariable('INSTALL_DIR_FOR_ELIXIR', cachedPath) core.info('Installed Elixir version') - await exec(path.join(elixirPath, 'elixir'), ['-v']) + + if (debugLoggingEnabled()) { + core.exportVariable('ELIXIR_CLI_ECHO', 'true') + } + + if (OS === 'linux') { + cmd = path.join(elixirPath, 'elixir') + args = ['-v'] + options = {} + } else if (OS === 'win32') { + cmd = path.join(elixirPath, 'elixir.bat') + args = ['-v'] + options = { windowsVerbatimArguments: true } + } + await exec(cmd, args, options) await fs.promises.mkdir(escriptsPath, { recursive: true }) } catch (err) { @@ -9829,12 +9861,18 @@ async function installElixir(elixirVersion, hexMirrors) { * @param {string} gleamVersion */ async function installGleam(gleamVersion) { + let cmd + let args + const OS = process.platform if (OS === 'linux') { - await exec(__nccwpck_require__.ab + "install-gleam.sh", [gleamVersion]) + cmd = __nccwpck_require__.ab + "install-gleam.sh" + args = [gleamVersion] + await exec(__nccwpck_require__.ab + "install-gleam.sh", args) } else if (OS === 'win32') { - const script = __nccwpck_require__.ab + "install-gleam.ps1" - await exec(`pwsh.exe ${script} -VSN:${gleamVersion}`) + cmd = `pwsh.exe ${path.join(__dirname, 'install-gleam.ps1')}` + args = [`-VSN:${gleamVersion}`] + await exec(cmd, args) } } @@ -9844,12 +9882,18 @@ async function installGleam(gleamVersion) { * @param {string} rebar3Version */ async function installRebar3(rebar3Version) { + let cmd + let args + const OS = process.platform if (OS === 'linux') { - await exec(__nccwpck_require__.ab + "install-rebar3.sh", [rebar3Version]) + cmd = __nccwpck_require__.ab + "install-rebar3.sh" + args = [rebar3Version] + await exec(__nccwpck_require__.ab + "install-rebar3.sh", args) } else if (OS === 'win32') { - const script = __nccwpck_require__.ab + "install-rebar3.ps1" - await exec(`pwsh.exe ${script} -VSN:${rebar3Version}`) + cmd = `pwsh.exe ${path.join(__dirname, 'install-rebar3.ps1')}` + args = [`-VSN:${rebar3Version}`] + await exec(cmd, args) } } @@ -9861,6 +9905,10 @@ function checkPlatform() { } } +function debugLoggingEnabled() { + return !!process.env.RUNNER_DEBUG +} + module.exports = { installOTP, installElixir, @@ -10128,7 +10176,6 @@ async function getOTPVersions(osVersion, hexMirrors) { 'https://api.github.com/repos/erlang/otp/releases?per_page=100' otpVersionsListings = await get(originListing, [1, 2, 3]) } - debugLog(`OTP versions listings from ${originListing}`, otpVersionsListings) const otpVersions = new Map() @@ -10141,6 +10188,7 @@ async function getOTPVersions(osVersion, hexMirrors) { .match(/^([^ ]+)?( .+)/)[1] .match(/^([^-]+-)?(.+)$/) const otpVersion = otpMatch[2] + debugLog('OTP line and parsing', [line, otpVersion, otpMatch]) otpVersions.set(otpVersion, otpMatch[0]) // we keep the original for later reference }) } else if (process.platform === 'win32') { @@ -10152,10 +10200,17 @@ async function getOTPVersions(osVersion, hexMirrors) { .forEach((x) => { const otpMatch = x.name.match(/^otp_win64_(.*).exe$/) const otpVersion = otpMatch[1] + debugLog('OTP line and parsing', [otpMatch, otpVersion]) otpVersions.set(otpVersion, otpVersion) }) }) } + + debugLog( + `OTP versions from ${originListing}`, + JSON.stringify([...otpVersions.entries()]), + ) + return otpVersions } @@ -10262,7 +10317,7 @@ function maybeCoerced(v) { try { ret = semver.coerce(v).version } catch { - // some stuff can't be coerced, like 'master' + // some stuff can't be coerced, like 'main' ret = v } @@ -10445,11 +10500,16 @@ function jsonParseAsList(maybeJson) { } function debugLog(groupName, message) { - if (process.env.RUNNER_DEBUG) { - core.startGroup(`Debugging for ${groupName}`) - core.debug(message) - core.endGroup() - } + const group = `Debugging for ${groupName}` + core.debug( + '┌──────────────────────────────────────────────────────────────────────────', + ) + core.debug(`│ ${group} - start`) + core.debug(message) + core.debug(`│ ${group} - stop`) + core.debug( + '└──────────────────────────────────────────────────────────────────────────', + ) } module.exports = { diff --git a/src/installer.js b/src/installer.js index b24821cc..d7a61f96 100644 --- a/src/installer.js +++ b/src/installer.js @@ -13,6 +13,9 @@ const os = require('os') * @param {string[]} hexMirrors */ async function installOTP(osVersion, otpVersion, hexMirrors) { + let cmd + let args + if (hexMirrors.length === 0) { throw new Error( `Could not install Erlang/OTP ${otpVersion} from any hex.pm mirror`, @@ -37,7 +40,9 @@ async function installOTP(osVersion, otpVersion, hexMirrors) { cachedPath = await tc.cacheDir(extractPath, 'otp', fullVersion) } - await exec(path.join(cachedPath, 'Install'), ['-minimal', cachedPath]) + cmd = path.join(cachedPath, 'Install') + args = ['-minimal', cachedPath] + await exec(cmd, args) const otpPath = path.join(cachedPath, 'bin') @@ -45,7 +50,9 @@ async function installOTP(osVersion, otpVersion, hexMirrors) { core.exportVariable('INSTALL_DIR_FOR_OTP', cachedPath) core.info('Installed Erlang/OTP version') - await exec(path.join(otpPath, 'erl'), ['-version']) + cmd = path.join(otpPath, 'erl') + args = ['-version'] + await exec(cmd, args) } else if (OS === 'win32') { if (!cachedPath) { const exePath = await tc.downloadTool( @@ -59,13 +66,19 @@ async function installOTP(osVersion, otpVersion, hexMirrors) { const otpPath = path.join(otpDir, 'bin') await fs.promises.mkdir(otpDir, { recursive: true }) - await exec(path.join(cachedPath, 'otp.exe'), ['/S', `/D=${otpDir}`]) + + cmd = path.join(cachedPath, 'otp.exe') + args = ['/S', `/D=${otpDir}`] + await exec(cmd, args) core.addPath(otpPath) core.exportVariable('INSTALL_DIR_FOR_OTP', otpDir) core.info('Installed Erlang/OTP version') - await exec(path.join(otpPath, 'erl'), ['+V']) + + cmd = path.join(otpPath, 'erl.exe') + args = ['+V'] + await exec(cmd, args) } } catch (err) { core.info(`Install OTP failed for mirror ${hexMirror}`) @@ -81,6 +94,10 @@ async function installOTP(osVersion, otpVersion, hexMirrors) { * @param {string[]} hexMirrors */ async function installElixir(elixirVersion, hexMirrors) { + let cmd + let args + let options + if (hexMirrors.length === 0) { throw new Error( `Could not install Elixir ${elixirVersion} from any hex.pm mirror`, @@ -90,6 +107,7 @@ async function installElixir(elixirVersion, hexMirrors) { try { let cachedPath = tc.find('elixir', elixirVersion) + const OS = process.platform if (!cachedPath) { const zipPath = await tc.downloadTool( @@ -107,7 +125,21 @@ async function installElixir(elixirVersion, hexMirrors) { core.exportVariable('INSTALL_DIR_FOR_ELIXIR', cachedPath) core.info('Installed Elixir version') - await exec(path.join(elixirPath, 'elixir'), ['-v']) + + if (debugLoggingEnabled()) { + core.exportVariable('ELIXIR_CLI_ECHO', 'true') + } + + if (OS === 'linux') { + cmd = path.join(elixirPath, 'elixir') + args = ['-v'] + options = {} + } else if (OS === 'win32') { + cmd = path.join(elixirPath, 'elixir.bat') + args = ['-v'] + options = { windowsVerbatimArguments: true } + } + await exec(cmd, args, options) await fs.promises.mkdir(escriptsPath, { recursive: true }) } catch (err) { @@ -123,12 +155,18 @@ async function installElixir(elixirVersion, hexMirrors) { * @param {string} gleamVersion */ async function installGleam(gleamVersion) { + let cmd + let args + const OS = process.platform if (OS === 'linux') { - await exec(path.join(__dirname, 'install-gleam.sh'), [gleamVersion]) + cmd = path.join(__dirname, 'install-gleam.sh') + args = [gleamVersion] + await exec(cmd, args) } else if (OS === 'win32') { - const script = path.join(__dirname, 'install-gleam.ps1') - await exec(`pwsh.exe ${script} -VSN:${gleamVersion}`) + cmd = `pwsh.exe ${path.join(__dirname, 'install-gleam.ps1')}` + args = [`-VSN:${gleamVersion}`] + await exec(cmd, args) } } @@ -138,12 +176,18 @@ async function installGleam(gleamVersion) { * @param {string} rebar3Version */ async function installRebar3(rebar3Version) { + let cmd + let args + const OS = process.platform if (OS === 'linux') { - await exec(path.join(__dirname, 'install-rebar3.sh'), [rebar3Version]) + cmd = path.join(__dirname, 'install-rebar3.sh') + args = [rebar3Version] + await exec(cmd, args) } else if (OS === 'win32') { - const script = path.join(__dirname, 'install-rebar3.ps1') - await exec(`pwsh.exe ${script} -VSN:${rebar3Version}`) + cmd = `pwsh.exe ${path.join(__dirname, 'install-rebar3.ps1')}` + args = [`-VSN:${rebar3Version}`] + await exec(cmd, args) } } @@ -155,6 +199,10 @@ function checkPlatform() { } } +function debugLoggingEnabled() { + return !!process.env.RUNNER_DEBUG +} + module.exports = { installOTP, installElixir, diff --git a/src/setup-beam.js b/src/setup-beam.js index cc97f9a7..91108fb9 100644 --- a/src/setup-beam.js +++ b/src/setup-beam.js @@ -256,7 +256,6 @@ async function getOTPVersions(osVersion, hexMirrors) { 'https://api.github.com/repos/erlang/otp/releases?per_page=100' otpVersionsListings = await get(originListing, [1, 2, 3]) } - debugLog(`OTP versions listings from ${originListing}`, otpVersionsListings) const otpVersions = new Map() @@ -269,6 +268,7 @@ async function getOTPVersions(osVersion, hexMirrors) { .match(/^([^ ]+)?( .+)/)[1] .match(/^([^-]+-)?(.+)$/) const otpVersion = otpMatch[2] + debugLog('OTP line and parsing', [line, otpVersion, otpMatch]) otpVersions.set(otpVersion, otpMatch[0]) // we keep the original for later reference }) } else if (process.platform === 'win32') { @@ -280,10 +280,17 @@ async function getOTPVersions(osVersion, hexMirrors) { .forEach((x) => { const otpMatch = x.name.match(/^otp_win64_(.*).exe$/) const otpVersion = otpMatch[1] + debugLog('OTP line and parsing', [otpMatch, otpVersion]) otpVersions.set(otpVersion, otpVersion) }) }) } + + debugLog( + `OTP versions from ${originListing}`, + JSON.stringify([...otpVersions.entries()]), + ) + return otpVersions } @@ -390,7 +397,7 @@ function maybeCoerced(v) { try { ret = semver.coerce(v).version } catch { - // some stuff can't be coerced, like 'master' + // some stuff can't be coerced, like 'main' ret = v } @@ -573,11 +580,16 @@ function jsonParseAsList(maybeJson) { } function debugLog(groupName, message) { - if (process.env.RUNNER_DEBUG) { - core.startGroup(`Debugging for ${groupName}`) - core.debug(message) - core.endGroup() - } + const group = `Debugging for ${groupName}` + core.debug( + '┌──────────────────────────────────────────────────────────────────────────', + ) + core.debug(`│ ${group} - start`) + core.debug(message) + core.debug(`│ ${group} - stop`) + core.debug( + '└──────────────────────────────────────────────────────────────────────────', + ) } module.exports = { diff --git a/test/setup-beam.test.js b/test/setup-beam.test.js index b6acf801..5e41496e 100644 --- a/test/setup-beam.test.js +++ b/test/setup-beam.test.js @@ -199,7 +199,7 @@ async function testElixirVersions() { before = simulateInput('version-type', 'strict') spec = '1.14.0' - otpVersion = 'master' + otpVersion = 'main' expected = 'v1.14.0' got = await setupBeam.getElixirVersion(spec, otpVersion, hexMirrors) assert.deepStrictEqual(got, expected) @@ -213,6 +213,14 @@ async function testElixirVersions() { assert.deepStrictEqual(got, expected) simulateInput('version-type', before) + before = simulateInput('version-type', 'strict') + spec = 'v1.11.0-rc.0-otp-23' + otpVersion = 'OTP-23' + expected = 'v1.11.0-rc.0-otp-23' + got = await setupBeam.getElixirVersion(spec, otpVersion, hexMirrors) + assert.deepStrictEqual(got, expected) + simulateInput('version-type', before) + before = simulateInput('version-type', 'strict') spec = 'v1.11.0' otpVersion = '22.3.4.2' @@ -309,7 +317,7 @@ async function testGetVersionFromSpec() { '22.3.4.9.1', '22.3.4.12.1', '22.3.4.10.1', - 'master', + 'main', 'v11.11.0-rc.0-otp-23', '22.3.4.2', ] @@ -396,17 +404,12 @@ async function testGetVersionFromSpec() { assert.deepStrictEqual(got, expected) before = simulateInput('version-type', 'strict') - spec = 'master' - expected = 'master' + spec = 'main' + expected = 'main' got = setupBeam.getVersionFromSpec(spec, versions) assert.deepStrictEqual(got, expected) simulateInput('version-type', before) - spec = 'master' - expected = 'master' - got = setupBeam.getVersionFromSpec(spec, versions) - assert.deepStrictEqual(got, expected) - before = simulateInput('version-type', 'strict') spec = '22.3.4.2' expected = '22.3.4.2'