From 82b4ccd605ae89a242a7f79f81fe08c334a35a33 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Emilio=20L=C3=B3pez?=
<2642849+elopez@users.noreply.github.com>
Date: Fri, 1 Dec 2023 01:20:57 -0300
Subject: [PATCH] Install the correct ghcup binary on aarch64 (#47)
This enables native compilations on M1 Mac runners.
Closes: #46
---
dist/index.js | 77 ++++++++++++++++++++++++-----------------
lib/installer.d.ts | 8 ++---
lib/installer.js | 70 ++++++++++++++++++++++---------------
lib/opts.d.ts | 1 +
lib/setup-haskell.js | 7 ++--
src/installer.ts | 82 ++++++++++++++++++++++++++++----------------
src/opts.ts | 1 +
src/setup-haskell.ts | 9 ++---
8 files changed, 156 insertions(+), 99 deletions(-)
diff --git a/dist/index.js b/dist/index.js
index dff1358..37158f1 100644
--- a/dist/index.js
+++ b/dist/index.js
@@ -13437,7 +13437,7 @@ function aptVersion(tool, version) {
// This regex is intentionally liberal to accommodate unusual cases like "head".
return tool === 'cabal' ? /[^.]*\.?[^.]*/.exec(version)[0] : version;
}
-async function isInstalled(tool, version, os) {
+async function isInstalled(tool, version, os, arch) {
const toolPath = tc.find(tool, version);
if (toolPath)
return success(tool, version, toolPath, os);
@@ -13462,7 +13462,7 @@ async function isInstalled(tool, version, os) {
}[os]
};
core.debug(`isInstalled ${tool} ${version} ${locations[tool]}`);
- const f = await exec(await ghcupBin(os), ['whereis', tool, version]);
+ const f = await exec(await ghcupBin(os, arch), ['whereis', tool, version]);
core.info(`\n`);
core.debug(`isInstalled whereis ${f}`);
for (const p of locations[tool]) {
@@ -13484,7 +13484,7 @@ async function isInstalled(tool, version, os) {
if (installedPath === ghcupPath) {
// If the result of this `ghcup set` is non-zero, the version we want
// is probably not actually installed
- const ghcupSetResult = await exec(await ghcupBin(os), [
+ const ghcupSetResult = await exec(await ghcupBin(os, arch), [
'set',
tool,
version
@@ -13495,7 +13495,7 @@ async function isInstalled(tool, version, os) {
else {
// Andreas, 2023-05-03, issue #245.
// Since we do not have the correct version, disable any default version.
- await exec(await ghcupBin(os), ['unset', tool]);
+ await exec(await ghcupBin(os, arch), ['unset', tool]);
}
}
else {
@@ -13507,13 +13507,13 @@ async function isInstalled(tool, version, os) {
}
return false;
}
-async function installTool(tool, version, os) {
- if (await isInstalled(tool, version, os))
+async function installTool(tool, version, os, arch) {
+ if (await isInstalled(tool, version, os, arch))
return;
warn(tool, version);
if (tool === 'stack') {
- await stack(version, os);
- if (await isInstalled(tool, version, os))
+ await stack(version, os, arch);
+ if (await isInstalled(tool, version, os, arch))
return;
return failed(tool, version);
}
@@ -13535,27 +13535,27 @@ async function installTool(tool, version, os) {
// if (!(await aptLibCurses5())) break;
await aptLibNCurses5();
}
- await ghcup(tool, version, os);
- if (await isInstalled(tool, version, os))
+ await ghcup(tool, version, os, arch);
+ if (await isInstalled(tool, version, os, arch))
return;
await apt(tool, version);
break;
case 'win32':
await choco(tool, version);
- if (await isInstalled(tool, version, os))
+ if (await isInstalled(tool, version, os, arch))
return;
- await ghcup(tool, version, os);
+ await ghcup(tool, version, os, arch);
break;
case 'darwin':
- await ghcup(tool, version, os);
+ await ghcup(tool, version, os, arch);
break;
}
- if (await isInstalled(tool, version, os))
+ if (await isInstalled(tool, version, os, arch))
return;
return failed(tool, version);
}
exports.installTool = installTool;
-async function resetTool(tool, _version, os) {
+async function resetTool(tool, _version, os, arch) {
if (tool === 'stack') {
// We don't need to do anything here... yet
// (Once we switch to utilizing ghcup for stack when possible, we can
@@ -13565,11 +13565,11 @@ async function resetTool(tool, _version, os) {
let bin = '';
switch (os) {
case 'linux':
- bin = await ghcupBin(os);
+ bin = await ghcupBin(os, arch);
await exec(bin, ['unset', tool]);
return;
case 'darwin':
- bin = await ghcupBin(os);
+ bin = await ghcupBin(os, arch);
await exec(bin, ['unset', tool]);
return;
case 'win32':
@@ -13578,11 +13578,24 @@ async function resetTool(tool, _version, os) {
}
}
exports.resetTool = resetTool;
-async function stack(version, os) {
- core.info(`Attempting to install stack ${version}`);
+async function stackArchString(arch) {
+ switch (arch) {
+ case 'arm64':
+ return Promise.resolve('aarch64');
+ case 'x64':
+ return Promise.resolve('x86_64');
+ default:
+ const err = `Unsupported architecture: ${arch}`;
+ core.error(err);
+ return Promise.reject(err);
+ }
+}
+async function stack(version, os, arch) {
+ const binArch = await stackArchString(arch);
+ core.info(`Attempting to install stack ${version} for arch ${binArch}`);
const build = {
- linux: `linux-x86_64${(0, compare_versions_1.compareVersions)(version, '2.3.1') >= 0 ? '' : '-static'}`,
- darwin: 'osx-x86_64',
+ linux: `linux-${binArch}${(0, compare_versions_1.compareVersions)(version, '2.3.1') >= 0 ? '' : '-static'}`,
+ darwin: `osx-${binArch}`,
win32: 'windows-x86_64'
}[os];
const url = `https://github.com/commercialhaskell/stack/releases/download/v${version}/stack-${version}-${build}.tar.gz`;
@@ -13642,7 +13655,7 @@ async function choco(tool, version) {
if (tool == 'ghc')
core.addPath(chocoPath);
}
-async function ghcupBin(os) {
+async function ghcupBin(os, arch) {
core.debug(`ghcupBin : ${os}`);
if (os === 'win32') {
return 'ghcup';
@@ -13650,19 +13663,20 @@ async function ghcupBin(os) {
const cachedBin = tc.find('ghcup', opts_1.ghcup_version);
if (cachedBin)
return (0, path_1.join)(cachedBin, 'ghcup');
- const bin = await tc.downloadTool(`https://downloads.haskell.org/ghcup/${opts_1.ghcup_version}/x86_64-${os === 'darwin' ? 'apple-darwin' : 'linux'}-ghcup-${opts_1.ghcup_version}`);
+ const binArch = await stackArchString(arch);
+ const bin = await tc.downloadTool(`https://downloads.haskell.org/ghcup/${opts_1.ghcup_version}/${binArch}-${os === 'darwin' ? 'apple-darwin' : 'linux'}-ghcup-${opts_1.ghcup_version}`);
await fs_1.promises.chmod(bin, 0o755);
return (0, path_1.join)(await tc.cacheFile(bin, 'ghcup', 'ghcup', opts_1.ghcup_version), 'ghcup');
}
-async function addGhcupReleaseChannel(channel, os) {
+async function addGhcupReleaseChannel(channel, os, arch) {
core.info(`Adding ghcup release channel: ${channel}`);
- const bin = await ghcupBin(os);
+ const bin = await ghcupBin(os, arch);
await exec(bin, ['config', 'add-release-channel', channel.toString()]);
}
exports.addGhcupReleaseChannel = addGhcupReleaseChannel;
-async function ghcup(tool, version, os) {
+async function ghcup(tool, version, os, arch) {
core.info(`Attempting to install ${tool} ${version} using ghcup`);
- const bin = await ghcupBin(os);
+ const bin = await ghcupBin(os, arch);
if (tool === 'cabal' && version === 'head') {
await ghcupCabalHead(os, bin);
}
@@ -13697,7 +13711,7 @@ async function ghcupCabalHead(os, bin) {
}
async function ghcupGHCHead() {
core.info(`Attempting to install ghc head using ghcup`);
- const bin = await ghcupBin('linux');
+ const bin = await ghcupBin('linux', 'x64');
const returnCode = await exec(bin, [
'install',
'ghc',
@@ -14032,16 +14046,17 @@ async function run(inputs) {
try {
core.info('Preparing to setup a Haskell environment');
const os = process.platform;
+ const arch = process.arch;
const opts = (0, opts_1.getOpts)((0, opts_1.getDefaults)(os), os, inputs);
core.debug(`run: inputs = ${JSON.stringify(inputs)}`);
core.debug(`run: os = ${JSON.stringify(os)}`);
core.debug(`run: opts = ${JSON.stringify(opts)}`);
if (opts.ghcup.releaseChannel) {
- await core.group(`Preparing ghcup environment`, async () => (0, installer_1.addGhcupReleaseChannel)(opts.ghcup.releaseChannel, os));
+ await core.group(`Preparing ghcup environment`, async () => (0, installer_1.addGhcupReleaseChannel)(opts.ghcup.releaseChannel, os, arch));
}
for (const [t, { resolved }] of Object.entries(opts).filter(o => o[1].enable)) {
- await core.group(`Preparing ${t} environment`, async () => (0, installer_1.resetTool)(t, resolved, os));
- await core.group(`Installing ${t} version ${resolved}`, async () => (0, installer_1.installTool)(t, resolved, os));
+ await core.group(`Preparing ${t} environment`, async () => (0, installer_1.resetTool)(t, resolved, os, arch));
+ await core.group(`Installing ${t} version ${resolved}`, async () => (0, installer_1.installTool)(t, resolved, os, arch));
}
if (opts.stack.setup)
await core.group('Pre-installing GHC with stack', async () => (0, exec_1.exec)('stack', ['setup', opts.ghc.resolved]));
diff --git a/lib/installer.d.ts b/lib/installer.d.ts
index 930d630..96a95b6 100644
--- a/lib/installer.d.ts
+++ b/lib/installer.d.ts
@@ -1,5 +1,5 @@
///
-import { OS, Tool } from './opts';
-export declare function installTool(tool: Tool, version: string, os: OS): Promise;
-export declare function resetTool(tool: Tool, _version: string, os: OS): Promise;
-export declare function addGhcupReleaseChannel(channel: URL, os: OS): Promise;
+import { Arch, OS, Tool } from './opts';
+export declare function installTool(tool: Tool, version: string, os: OS, arch: Arch): Promise;
+export declare function resetTool(tool: Tool, _version: string, os: OS, arch: Arch): Promise;
+export declare function addGhcupReleaseChannel(channel: URL, os: OS, arch: Arch): Promise;
diff --git a/lib/installer.js b/lib/installer.js
index c5f3ca5..8830ae2 100644
--- a/lib/installer.js
+++ b/lib/installer.js
@@ -73,7 +73,7 @@ function aptVersion(tool, version) {
// This regex is intentionally liberal to accommodate unusual cases like "head".
return tool === 'cabal' ? /[^.]*\.?[^.]*/.exec(version)[0] : version;
}
-async function isInstalled(tool, version, os) {
+async function isInstalled(tool, version, os, arch) {
const toolPath = tc.find(tool, version);
if (toolPath)
return success(tool, version, toolPath, os);
@@ -98,7 +98,7 @@ async function isInstalled(tool, version, os) {
}[os]
};
core.debug(`isInstalled ${tool} ${version} ${locations[tool]}`);
- const f = await exec(await ghcupBin(os), ['whereis', tool, version]);
+ const f = await exec(await ghcupBin(os, arch), ['whereis', tool, version]);
core.info(`\n`);
core.debug(`isInstalled whereis ${f}`);
for (const p of locations[tool]) {
@@ -120,7 +120,7 @@ async function isInstalled(tool, version, os) {
if (installedPath === ghcupPath) {
// If the result of this `ghcup set` is non-zero, the version we want
// is probably not actually installed
- const ghcupSetResult = await exec(await ghcupBin(os), [
+ const ghcupSetResult = await exec(await ghcupBin(os, arch), [
'set',
tool,
version
@@ -131,7 +131,7 @@ async function isInstalled(tool, version, os) {
else {
// Andreas, 2023-05-03, issue #245.
// Since we do not have the correct version, disable any default version.
- await exec(await ghcupBin(os), ['unset', tool]);
+ await exec(await ghcupBin(os, arch), ['unset', tool]);
}
}
else {
@@ -143,13 +143,13 @@ async function isInstalled(tool, version, os) {
}
return false;
}
-async function installTool(tool, version, os) {
- if (await isInstalled(tool, version, os))
+async function installTool(tool, version, os, arch) {
+ if (await isInstalled(tool, version, os, arch))
return;
warn(tool, version);
if (tool === 'stack') {
- await stack(version, os);
- if (await isInstalled(tool, version, os))
+ await stack(version, os, arch);
+ if (await isInstalled(tool, version, os, arch))
return;
return failed(tool, version);
}
@@ -171,27 +171,27 @@ async function installTool(tool, version, os) {
// if (!(await aptLibCurses5())) break;
await aptLibNCurses5();
}
- await ghcup(tool, version, os);
- if (await isInstalled(tool, version, os))
+ await ghcup(tool, version, os, arch);
+ if (await isInstalled(tool, version, os, arch))
return;
await apt(tool, version);
break;
case 'win32':
await choco(tool, version);
- if (await isInstalled(tool, version, os))
+ if (await isInstalled(tool, version, os, arch))
return;
- await ghcup(tool, version, os);
+ await ghcup(tool, version, os, arch);
break;
case 'darwin':
- await ghcup(tool, version, os);
+ await ghcup(tool, version, os, arch);
break;
}
- if (await isInstalled(tool, version, os))
+ if (await isInstalled(tool, version, os, arch))
return;
return failed(tool, version);
}
exports.installTool = installTool;
-async function resetTool(tool, _version, os) {
+async function resetTool(tool, _version, os, arch) {
if (tool === 'stack') {
// We don't need to do anything here... yet
// (Once we switch to utilizing ghcup for stack when possible, we can
@@ -201,11 +201,11 @@ async function resetTool(tool, _version, os) {
let bin = '';
switch (os) {
case 'linux':
- bin = await ghcupBin(os);
+ bin = await ghcupBin(os, arch);
await exec(bin, ['unset', tool]);
return;
case 'darwin':
- bin = await ghcupBin(os);
+ bin = await ghcupBin(os, arch);
await exec(bin, ['unset', tool]);
return;
case 'win32':
@@ -214,11 +214,24 @@ async function resetTool(tool, _version, os) {
}
}
exports.resetTool = resetTool;
-async function stack(version, os) {
- core.info(`Attempting to install stack ${version}`);
+async function stackArchString(arch) {
+ switch (arch) {
+ case 'arm64':
+ return Promise.resolve('aarch64');
+ case 'x64':
+ return Promise.resolve('x86_64');
+ default:
+ const err = `Unsupported architecture: ${arch}`;
+ core.error(err);
+ return Promise.reject(err);
+ }
+}
+async function stack(version, os, arch) {
+ const binArch = await stackArchString(arch);
+ core.info(`Attempting to install stack ${version} for arch ${binArch}`);
const build = {
- linux: `linux-x86_64${(0, compare_versions_1.compareVersions)(version, '2.3.1') >= 0 ? '' : '-static'}`,
- darwin: 'osx-x86_64',
+ linux: `linux-${binArch}${(0, compare_versions_1.compareVersions)(version, '2.3.1') >= 0 ? '' : '-static'}`,
+ darwin: `osx-${binArch}`,
win32: 'windows-x86_64'
}[os];
const url = `https://github.com/commercialhaskell/stack/releases/download/v${version}/stack-${version}-${build}.tar.gz`;
@@ -278,7 +291,7 @@ async function choco(tool, version) {
if (tool == 'ghc')
core.addPath(chocoPath);
}
-async function ghcupBin(os) {
+async function ghcupBin(os, arch) {
core.debug(`ghcupBin : ${os}`);
if (os === 'win32') {
return 'ghcup';
@@ -286,19 +299,20 @@ async function ghcupBin(os) {
const cachedBin = tc.find('ghcup', opts_1.ghcup_version);
if (cachedBin)
return (0, path_1.join)(cachedBin, 'ghcup');
- const bin = await tc.downloadTool(`https://downloads.haskell.org/ghcup/${opts_1.ghcup_version}/x86_64-${os === 'darwin' ? 'apple-darwin' : 'linux'}-ghcup-${opts_1.ghcup_version}`);
+ const binArch = await stackArchString(arch);
+ const bin = await tc.downloadTool(`https://downloads.haskell.org/ghcup/${opts_1.ghcup_version}/${binArch}-${os === 'darwin' ? 'apple-darwin' : 'linux'}-ghcup-${opts_1.ghcup_version}`);
await fs_1.promises.chmod(bin, 0o755);
return (0, path_1.join)(await tc.cacheFile(bin, 'ghcup', 'ghcup', opts_1.ghcup_version), 'ghcup');
}
-async function addGhcupReleaseChannel(channel, os) {
+async function addGhcupReleaseChannel(channel, os, arch) {
core.info(`Adding ghcup release channel: ${channel}`);
- const bin = await ghcupBin(os);
+ const bin = await ghcupBin(os, arch);
await exec(bin, ['config', 'add-release-channel', channel.toString()]);
}
exports.addGhcupReleaseChannel = addGhcupReleaseChannel;
-async function ghcup(tool, version, os) {
+async function ghcup(tool, version, os, arch) {
core.info(`Attempting to install ${tool} ${version} using ghcup`);
- const bin = await ghcupBin(os);
+ const bin = await ghcupBin(os, arch);
if (tool === 'cabal' && version === 'head') {
await ghcupCabalHead(os, bin);
}
@@ -333,7 +347,7 @@ async function ghcupCabalHead(os, bin) {
}
async function ghcupGHCHead() {
core.info(`Attempting to install ghc head using ghcup`);
- const bin = await ghcupBin('linux');
+ const bin = await ghcupBin('linux', 'x64');
const returnCode = await exec(bin, [
'install',
'ghc',
diff --git a/lib/opts.d.ts b/lib/opts.d.ts
index 5b5a360..c003555 100644
--- a/lib/opts.d.ts
+++ b/lib/opts.d.ts
@@ -7,6 +7,7 @@ export type Revisions = Record>>;
export type OS = 'linux' | 'darwin' | 'win32';
+export type Arch = 'arm64' | 'x64';
export type Tool = 'cabal' | 'ghc' | 'stack';
export interface ProgramOpt {
enable: boolean;
diff --git a/lib/setup-haskell.js b/lib/setup-haskell.js
index 4285907..f74417e 100644
--- a/lib/setup-haskell.js
+++ b/lib/setup-haskell.js
@@ -48,16 +48,17 @@ async function run(inputs) {
try {
core.info('Preparing to setup a Haskell environment');
const os = process.platform;
+ const arch = process.arch;
const opts = (0, opts_1.getOpts)((0, opts_1.getDefaults)(os), os, inputs);
core.debug(`run: inputs = ${JSON.stringify(inputs)}`);
core.debug(`run: os = ${JSON.stringify(os)}`);
core.debug(`run: opts = ${JSON.stringify(opts)}`);
if (opts.ghcup.releaseChannel) {
- await core.group(`Preparing ghcup environment`, async () => (0, installer_1.addGhcupReleaseChannel)(opts.ghcup.releaseChannel, os));
+ await core.group(`Preparing ghcup environment`, async () => (0, installer_1.addGhcupReleaseChannel)(opts.ghcup.releaseChannel, os, arch));
}
for (const [t, { resolved }] of Object.entries(opts).filter(o => o[1].enable)) {
- await core.group(`Preparing ${t} environment`, async () => (0, installer_1.resetTool)(t, resolved, os));
- await core.group(`Installing ${t} version ${resolved}`, async () => (0, installer_1.installTool)(t, resolved, os));
+ await core.group(`Preparing ${t} environment`, async () => (0, installer_1.resetTool)(t, resolved, os, arch));
+ await core.group(`Installing ${t} version ${resolved}`, async () => (0, installer_1.installTool)(t, resolved, os, arch));
}
if (opts.stack.setup)
await core.group('Pre-installing GHC with stack', async () => (0, exec_1.exec)('stack', ['setup', opts.ghc.resolved]));
diff --git a/src/installer.ts b/src/installer.ts
index 96e664c..93a3744 100644
--- a/src/installer.ts
+++ b/src/installer.ts
@@ -4,7 +4,7 @@ import {which} from '@actions/io';
import * as tc from '@actions/tool-cache';
import {promises as afs} from 'fs';
import {join, dirname} from 'path';
-import {ghcup_version, OS, Tool, releaseRevision} from './opts';
+import {ghcup_version, Arch, OS, Tool, releaseRevision} from './opts';
import process from 'process';
import * as glob from '@actions/glob';
import * as fs from 'fs';
@@ -69,7 +69,8 @@ function aptVersion(tool: string, version: string): string {
async function isInstalled(
tool: Tool,
version: string,
- os: OS
+ os: OS,
+ arch: Arch
): Promise {
const toolPath = tc.find(tool, version);
if (toolPath) return success(tool, version, toolPath, os);
@@ -103,7 +104,7 @@ async function isInstalled(
}[os]
};
core.debug(`isInstalled ${tool} ${version} ${locations[tool]}`);
- const f = await exec(await ghcupBin(os), ['whereis', tool, version]);
+ const f = await exec(await ghcupBin(os, arch), ['whereis', tool, version]);
core.info(`\n`);
core.debug(`isInstalled whereis ${f}`);
@@ -127,7 +128,7 @@ async function isInstalled(
if (installedPath === ghcupPath) {
// If the result of this `ghcup set` is non-zero, the version we want
// is probably not actually installed
- const ghcupSetResult = await exec(await ghcupBin(os), [
+ const ghcupSetResult = await exec(await ghcupBin(os, arch), [
'set',
tool,
version
@@ -137,7 +138,7 @@ async function isInstalled(
} else {
// Andreas, 2023-05-03, issue #245.
// Since we do not have the correct version, disable any default version.
- await exec(await ghcupBin(os), ['unset', tool]);
+ await exec(await ghcupBin(os, arch), ['unset', tool]);
}
} else {
// Install methods apt and choco have precise install paths,
@@ -152,14 +153,15 @@ async function isInstalled(
export async function installTool(
tool: Tool,
version: string,
- os: OS
+ os: OS,
+ arch: Arch
): Promise {
- if (await isInstalled(tool, version, os)) return;
+ if (await isInstalled(tool, version, os, arch)) return;
warn(tool, version);
if (tool === 'stack') {
- await stack(version, os);
- if (await isInstalled(tool, version, os)) return;
+ await stack(version, os, arch);
+ if (await isInstalled(tool, version, os, arch)) return;
return failed(tool, version);
}
@@ -183,28 +185,29 @@ export async function installTool(
// if (!(await aptLibCurses5())) break;
await aptLibNCurses5();
}
- await ghcup(tool, version, os);
- if (await isInstalled(tool, version, os)) return;
+ await ghcup(tool, version, os, arch);
+ if (await isInstalled(tool, version, os, arch)) return;
await apt(tool, version);
break;
case 'win32':
await choco(tool, version);
- if (await isInstalled(tool, version, os)) return;
- await ghcup(tool, version, os);
+ if (await isInstalled(tool, version, os, arch)) return;
+ await ghcup(tool, version, os, arch);
break;
case 'darwin':
- await ghcup(tool, version, os);
+ await ghcup(tool, version, os, arch);
break;
}
- if (await isInstalled(tool, version, os)) return;
+ if (await isInstalled(tool, version, os, arch)) return;
return failed(tool, version);
}
export async function resetTool(
tool: Tool,
_version: string,
- os: OS
+ os: OS,
+ arch: Arch
): Promise {
if (tool === 'stack') {
// We don't need to do anything here... yet
@@ -216,11 +219,11 @@ export async function resetTool(
let bin = '';
switch (os) {
case 'linux':
- bin = await ghcupBin(os);
+ bin = await ghcupBin(os, arch);
await exec(bin, ['unset', tool]);
return;
case 'darwin':
- bin = await ghcupBin(os);
+ bin = await ghcupBin(os, arch);
await exec(bin, ['unset', tool]);
return;
case 'win32':
@@ -229,13 +232,27 @@ export async function resetTool(
}
}
-async function stack(version: string, os: OS): Promise {
- core.info(`Attempting to install stack ${version}`);
+async function stackArchString(arch: Arch): Promise {
+ switch (arch) {
+ case 'arm64':
+ return Promise.resolve('aarch64');
+ case 'x64':
+ return Promise.resolve('x86_64');
+ default:
+ const err = `Unsupported architecture: ${arch}`;
+ core.error(err);
+ return Promise.reject(err);
+ }
+}
+
+async function stack(version: string, os: OS, arch: Arch): Promise {
+ const binArch = await stackArchString(arch);
+ core.info(`Attempting to install stack ${version} for arch ${binArch}`);
const build = {
- linux: `linux-x86_64${
+ linux: `linux-${binArch}${
compareVersions(version, '2.3.1') >= 0 ? '' : '-static'
}`,
- darwin: 'osx-x86_64',
+ darwin: `osx-${binArch}`,
win32: 'windows-x86_64'
}[os];
@@ -314,7 +331,7 @@ async function choco(tool: Tool, version: string): Promise {
if (tool == 'ghc') core.addPath(chocoPath);
}
-async function ghcupBin(os: OS): Promise {
+async function ghcupBin(os: OS, arch: Arch): Promise {
core.debug(`ghcupBin : ${os}`);
if (os === 'win32') {
return 'ghcup';
@@ -322,8 +339,9 @@ async function ghcupBin(os: OS): Promise {
const cachedBin = tc.find('ghcup', ghcup_version);
if (cachedBin) return join(cachedBin, 'ghcup');
+ const binArch = await stackArchString(arch);
const bin = await tc.downloadTool(
- `https://downloads.haskell.org/ghcup/${ghcup_version}/x86_64-${
+ `https://downloads.haskell.org/ghcup/${ghcup_version}/${binArch}-${
os === 'darwin' ? 'apple-darwin' : 'linux'
}-ghcup-${ghcup_version}`
);
@@ -336,16 +354,22 @@ async function ghcupBin(os: OS): Promise {
export async function addGhcupReleaseChannel(
channel: URL,
- os: OS
+ os: OS,
+ arch: Arch
): Promise {
core.info(`Adding ghcup release channel: ${channel}`);
- const bin = await ghcupBin(os);
+ const bin = await ghcupBin(os, arch);
await exec(bin, ['config', 'add-release-channel', channel.toString()]);
}
-async function ghcup(tool: Tool, version: string, os: OS): Promise {
+async function ghcup(
+ tool: Tool,
+ version: string,
+ os: OS,
+ arch: Arch
+): Promise {
core.info(`Attempting to install ${tool} ${version} using ghcup`);
- const bin = await ghcupBin(os);
+ const bin = await ghcupBin(os, arch);
if (tool === 'cabal' && version === 'head') {
await ghcupCabalHead(os, bin);
} else {
@@ -380,7 +404,7 @@ async function ghcupCabalHead(os: OS, bin: string): Promise {
async function ghcupGHCHead(): Promise {
core.info(`Attempting to install ghc head using ghcup`);
- const bin = await ghcupBin('linux');
+ const bin = await ghcupBin('linux', 'x64');
const returnCode = await exec(bin, [
'install',
'ghc',
diff --git a/src/opts.ts b/src/opts.ts
index 9fe2dc1..b919513 100644
--- a/src/opts.ts
+++ b/src/opts.ts
@@ -14,6 +14,7 @@ export type Revisions = Record<
Record>
>;
export type OS = 'linux' | 'darwin' | 'win32';
+export type Arch = 'arm64' | 'x64';
export type Tool = 'cabal' | 'ghc' | 'stack';
export interface ProgramOpt {
diff --git a/src/setup-haskell.ts b/src/setup-haskell.ts
index a0502e4..158d6f1 100644
--- a/src/setup-haskell.ts
+++ b/src/setup-haskell.ts
@@ -6,7 +6,7 @@ import * as path from 'path';
import {EOL} from 'os';
import {getOpts, getDefaults, Tool} from './opts';
import {addGhcupReleaseChannel, installTool, resetTool} from './installer';
-import type {OS} from './opts';
+import type {Arch, OS} from './opts';
import {exec} from '@actions/exec';
async function cabalConfig(): Promise {
@@ -25,6 +25,7 @@ export default async function run(
try {
core.info('Preparing to setup a Haskell environment');
const os = process.platform as OS;
+ const arch = process.arch as Arch;
const opts = getOpts(getDefaults(os), os, inputs);
core.debug(`run: inputs = ${JSON.stringify(inputs)}`);
core.debug(`run: os = ${JSON.stringify(os)}`);
@@ -32,7 +33,7 @@ export default async function run(
if (opts.ghcup.releaseChannel) {
await core.group(`Preparing ghcup environment`, async () =>
- addGhcupReleaseChannel(opts.ghcup.releaseChannel!, os)
+ addGhcupReleaseChannel(opts.ghcup.releaseChannel!, os, arch)
);
}
@@ -40,10 +41,10 @@ export default async function run(
o => o[1].enable
)) {
await core.group(`Preparing ${t} environment`, async () =>
- resetTool(t as Tool, resolved, os)
+ resetTool(t as Tool, resolved, os, arch)
);
await core.group(`Installing ${t} version ${resolved}`, async () =>
- installTool(t as Tool, resolved, os)
+ installTool(t as Tool, resolved, os, arch)
);
}