Compare commits
97 Commits
v0.1.9-rc1
...
v0.1.12-rc
| Author | SHA1 | Date | |
|---|---|---|---|
| 53f5a08924 | |||
| d368863c3d | |||
| c76cce5830 | |||
| 4fef93b7b1 | |||
| 241dadbeb5 | |||
| 12459c2544 | |||
| e250d6013f | |||
| 0d41d180d6 | |||
|
|
ef251ce17e | ||
| 956e11c3f8 | |||
| 951b0843b2 | |||
| a4e4080a1b | |||
| 2aa91c5d91 | |||
| 6471b3877f | |||
| 778ee142d5 | |||
| 1140132a25 | |||
| c5c299179d | |||
| 0ce4549eb8 | |||
| 97d568ddd6 | |||
| ea58465240 | |||
| 7afd262b1b | |||
| 34ceaa0823 | |||
| 57c34a07f2 | |||
| 73d1d97f1f | |||
| f7ed1a4bde | |||
| bdd80d0229 | |||
| 0238f70c64 | |||
| 24ff430c45 | |||
| 281fb14d4c | |||
| 03bac93929 | |||
| 1ae0c2a654 | |||
| 8140936bd3 | |||
| 8786acf476 | |||
| c460d4c743 | |||
| 5294adf0d7 | |||
| 00f3fa35fd | |||
| 9c9aa4f9c0 | |||
| e23a187cc4 | |||
| 3e429945dc | |||
| 8be2a8eed7 | |||
| 6b65167581 | |||
| 9d7914e69a | |||
| 6c62884b24 | |||
| 965d2a3ba8 | |||
| 40a1cc98c6 | |||
| 4c2d4ee6bd | |||
| 9276664465 | |||
| a94bcdb92d | |||
| 5da5fabfef | |||
| 05cc55c52d | |||
| 571df1349c | |||
| cbbb75062c | |||
| bb7c4205db | |||
| b2027f1625 | |||
| 65945c87df | |||
| 081582d3e1 | |||
| bf240af518 | |||
| a269131e2d | |||
| 59ece98fdc | |||
| 563924ff26 | |||
| 8ee3f55428 | |||
| 93c17607b5 | |||
| 8b4c239444 | |||
| 8bef17bf59 | |||
| a649146a39 | |||
| 9d6a5313ab | |||
| de09c950d5 | |||
| 47838b1bd9 | |||
| 02b360e2a9 | |||
| c10ab15e0c | |||
| 46f3da1a94 | |||
| 7ec9d90aab | |||
| 326bf510c9 | |||
| ce3d1f4309 | |||
| b31ba883e4 | |||
| e5d1c04616 | |||
| 34ff0ed9cf | |||
| 85bd87d5f3 | |||
| 8b274214af | |||
| 069e3102f4 | |||
| 8623b32721 | |||
| 6342e8edf0 | |||
| bbd8f0c84c | |||
| 873c951d6e | |||
| d9c864d3c5 | |||
| 4280d7109a | |||
| c8855c068f | |||
| 90503061e9 | |||
| 672ebf6426 | |||
| fd76fde23a | |||
| e24c9a3ffe | |||
| 2641d50c21 | |||
| 202f3ea3ba | |||
| 4f09e3ff7e | |||
| 1148219130 | |||
| 4b47800dfb | |||
| e2c4db9132 |
@@ -17,7 +17,7 @@ variables:
|
|||||||
BIT: "64"
|
BIT: "64"
|
||||||
|
|
||||||
.alpine:64bit:
|
.alpine:64bit:
|
||||||
image: "alpine:edge"
|
image: "alpine:3.12"
|
||||||
tags:
|
tags:
|
||||||
- x86_64-linux
|
- x86_64-linux
|
||||||
variables:
|
variables:
|
||||||
@@ -25,7 +25,7 @@ variables:
|
|||||||
BIT: "64"
|
BIT: "64"
|
||||||
|
|
||||||
.alpine:32bit:
|
.alpine:32bit:
|
||||||
image: "i386/alpine:edge"
|
image: "i386/alpine:3.12"
|
||||||
tags:
|
tags:
|
||||||
- x86_64-linux
|
- x86_64-linux
|
||||||
variables:
|
variables:
|
||||||
@@ -60,7 +60,12 @@ variables:
|
|||||||
script:
|
script:
|
||||||
- ./.gitlab/script/ghcup_version.sh
|
- ./.gitlab/script/ghcup_version.sh
|
||||||
variables:
|
variables:
|
||||||
JSON_VERSION: "0.0.2"
|
JSON_VERSION: "0.0.3"
|
||||||
|
artifacts:
|
||||||
|
expire_in: 2 week
|
||||||
|
paths:
|
||||||
|
- golden
|
||||||
|
when: on_failure
|
||||||
|
|
||||||
.test_ghcup_version:linux:
|
.test_ghcup_version:linux:
|
||||||
extends:
|
extends:
|
||||||
@@ -102,6 +107,29 @@ variables:
|
|||||||
only:
|
only:
|
||||||
- tags
|
- tags
|
||||||
|
|
||||||
|
######## stack test ########
|
||||||
|
|
||||||
|
test:linux:stack:
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps_minimal.sh
|
||||||
|
script:
|
||||||
|
- ./.gitlab/script/ghcup_stack.sh
|
||||||
|
extends:
|
||||||
|
- .debian
|
||||||
|
|
||||||
|
######## bootstrap test ########
|
||||||
|
|
||||||
|
test:linux:bootstrap_script:
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps_minimal.sh
|
||||||
|
script:
|
||||||
|
- ./.gitlab/script/ghcup_bootstrap.sh
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.8.4"
|
||||||
|
CABAL_VERSION: "3.2.0.0"
|
||||||
|
extends:
|
||||||
|
- .debian
|
||||||
|
|
||||||
######## linux test ########
|
######## linux test ########
|
||||||
|
|
||||||
test:linux:recommended:
|
test:linux:recommended:
|
||||||
@@ -113,7 +141,7 @@ test:linux:recommended:
|
|||||||
test:linux:latest:
|
test:linux:latest:
|
||||||
extends: .test_ghcup_version:linux
|
extends: .test_ghcup_version:linux
|
||||||
variables:
|
variables:
|
||||||
GHC_VERSION: "8.10.1"
|
GHC_VERSION: "8.10.2"
|
||||||
CABAL_VERSION: "3.2.0.0"
|
CABAL_VERSION: "3.2.0.0"
|
||||||
allow_failure: true
|
allow_failure: true
|
||||||
|
|
||||||
@@ -136,7 +164,7 @@ test:mac:recommended:
|
|||||||
test:mac:latest:
|
test:mac:latest:
|
||||||
extends: .test_ghcup_version:darwin
|
extends: .test_ghcup_version:darwin
|
||||||
variables:
|
variables:
|
||||||
GHC_VERSION: "8.10.1"
|
GHC_VERSION: "8.10.2"
|
||||||
CABAL_VERSION: "3.2.0.0"
|
CABAL_VERSION: "3.2.0.0"
|
||||||
allow_failure: true
|
allow_failure: true
|
||||||
|
|
||||||
@@ -152,7 +180,7 @@ test:freebsd:recommended:
|
|||||||
test:freebsd:latest:
|
test:freebsd:latest:
|
||||||
extends: .test_ghcup_version:freebsd
|
extends: .test_ghcup_version:freebsd
|
||||||
variables:
|
variables:
|
||||||
GHC_VERSION: "8.10.1"
|
GHC_VERSION: "8.10.2"
|
||||||
CABAL_VERSION: "3.2.0.0"
|
CABAL_VERSION: "3.2.0.0"
|
||||||
allow_failure: true
|
allow_failure: true
|
||||||
|
|
||||||
|
|||||||
10
.gitlab/before_script/linux/install_deps_minimal.sh
Executable file
10
.gitlab/before_script/linux/install_deps_minimal.sh
Executable file
@@ -0,0 +1,10 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "${TMPDIR}"
|
||||||
|
|
||||||
|
sudo apt-get update -y
|
||||||
|
sudo apt-get install -y libnuma-dev zlib1g-dev libgmp-dev libgmp10 libssl-dev liblzma-dev git wget
|
||||||
30
.gitlab/script/ghcup_bootstrap.sh
Executable file
30
.gitlab/script/ghcup_bootstrap.sh
Executable file
@@ -0,0 +1,30 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "$CI_PROJECT_DIR"/.local/bin
|
||||||
|
|
||||||
|
ecabal() {
|
||||||
|
cabal --store-dir="$(pwd)"/.store "$@"
|
||||||
|
}
|
||||||
|
|
||||||
|
eghcup() {
|
||||||
|
ghcup -v -c -s file://$(pwd)/ghcup-${JSON_VERSION}.yaml "$@"
|
||||||
|
}
|
||||||
|
|
||||||
|
git describe --always
|
||||||
|
|
||||||
|
### build
|
||||||
|
|
||||||
|
ecabal update
|
||||||
|
|
||||||
|
export BOOTSTRAP_HASKELL_NONINTERACTIVE=yes
|
||||||
|
export BOOTSTRAP_HASKELL_GHC_VERSION=$GHC_VERSION
|
||||||
|
export BOOTSTRAP_HASKELL_CABAL_VERSION=$CABAL_VERSION
|
||||||
|
|
||||||
|
./bootstrap-haskell
|
||||||
|
|
||||||
|
[ "$(ghc --numeric-version)" = "${GHC_VERSION}" ]
|
||||||
|
|
||||||
21
.gitlab/script/ghcup_stack.sh
Executable file
21
.gitlab/script/ghcup_stack.sh
Executable file
@@ -0,0 +1,21 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "$CI_PROJECT_DIR"/.local/bin
|
||||||
|
|
||||||
|
git describe --always
|
||||||
|
|
||||||
|
### build
|
||||||
|
|
||||||
|
curl -L -O https://get.haskellstack.org/stable/linux-x86_64.tar.gz
|
||||||
|
tar xf linux-x86_64.tar.gz
|
||||||
|
cp stack-*-linux-*/stack "$CI_PROJECT_DIR"/.local/bin/stack
|
||||||
|
chmod +x "$CI_PROJECT_DIR"/.local/bin/stack
|
||||||
|
|
||||||
|
mkdir -p "$CI_PROJECT_DIR"/.stack_root
|
||||||
|
export TAR_OPTIONS=--no-same-owner
|
||||||
|
stack --allow-different-user --stack-root "$CI_PROJECT_DIR"/.stack_root build
|
||||||
|
stack --allow-different-user --stack-root "$CI_PROJECT_DIR"/.stack_root test
|
||||||
@@ -20,22 +20,28 @@ git describe --always
|
|||||||
|
|
||||||
ecabal update
|
ecabal update
|
||||||
|
|
||||||
|
ecabal install -w ghc-${GHC_VERSION} --installdir="$CI_PROJECT_DIR"/.local/bin hspec-discover
|
||||||
|
|
||||||
if [ "${OS}" = "DARWIN" ] ; then
|
if [ "${OS}" = "DARWIN" ] ; then
|
||||||
ecabal build -w ghc-${GHC_VERSION} -ftui
|
ecabal build -w ghc-${GHC_VERSION} -ftui
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -ftui ghcup-test
|
||||||
elif [ "${OS}" = "LINUX" ] ; then
|
elif [ "${OS}" = "LINUX" ] ; then
|
||||||
if [ "${BIT}" = "32" ] ; then
|
if [ "${BIT}" = "32" ] ; then
|
||||||
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui -ftar
|
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui -ftar
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -finternal-downloader -ftui -ftar ghcup-test
|
||||||
else
|
else
|
||||||
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui
|
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -finternal-downloader -ftui ghcup-test
|
||||||
fi
|
fi
|
||||||
else
|
else
|
||||||
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui
|
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -finternal-downloader -ftui ghcup-test
|
||||||
fi
|
fi
|
||||||
|
|
||||||
ecabal haddock -w ghc-${GHC_VERSION} -ftar
|
ecabal haddock -w ghc-${GHC_VERSION} -ftar
|
||||||
|
|
||||||
cp "$(ecabal new-exec -w ghc-${GHC_VERSION} --enable-tests --verbose=0 --offline sh -- -c 'command -v ghcup')" .
|
cp "$(ecabal new-exec -w ghc-${GHC_VERSION} --verbose=0 --offline sh -- -c 'command -v ghcup')" .
|
||||||
cp "$(ecabal new-exec -w ghc-${GHC_VERSION} --enable-tests --verbose=0 --offline sh -- -c 'command -v ghcup-gen')" .
|
cp "$(ecabal new-exec -w ghc-${GHC_VERSION} --verbose=0 --offline sh -- -c 'command -v ghcup-gen')" .
|
||||||
|
|
||||||
cp ./ghcup "$CI_PROJECT_DIR"/.local/bin/ghcup
|
cp ./ghcup "$CI_PROJECT_DIR"/.local/bin/ghcup
|
||||||
cp ./ghcup-gen "$CI_PROJECT_DIR"/.local/bin/ghcup-gen
|
cp ./ghcup-gen "$CI_PROJECT_DIR"/.local/bin/ghcup-gen
|
||||||
@@ -87,6 +93,18 @@ eghcup set ${GHC_VERSION}
|
|||||||
eghcup rm 8.4.4
|
eghcup rm 8.4.4
|
||||||
[ "$(ghc --numeric-version)" = "${ghc_ver}" ]
|
[ "$(ghc --numeric-version)" = "${ghc_ver}" ]
|
||||||
|
|
||||||
|
# install hls
|
||||||
|
if [ "${OS}" = "DARWIN" ] ; then
|
||||||
|
eghcup install hls
|
||||||
|
haskell-language-server-wrapper --version
|
||||||
|
elif [ "${OS}" = "LINUX" ] ; then
|
||||||
|
if [ "${BIT}" = "64" ] ; then
|
||||||
|
eghcup install hls
|
||||||
|
haskell-language-server-wrapper --version
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
eghcup rm $(ghc --numeric-version)
|
eghcup rm $(ghc --numeric-version)
|
||||||
|
|
||||||
eghcup upgrade
|
eghcup upgrade
|
||||||
|
|||||||
30
CHANGELOG.md
30
CHANGELOG.md
@@ -1,13 +1,39 @@
|
|||||||
# Revision history for ghcup
|
# Revision history for ghcup
|
||||||
|
|
||||||
## 0.1.9 -- yyyy-mm-dd
|
## 0.1.12 -- ????-??-??
|
||||||
|
|
||||||
|
* improve TUI:
|
||||||
|
- separators between tools sections
|
||||||
|
- reverse list order so latest is on top
|
||||||
|
- expand the blues selected bar
|
||||||
|
- show new latest versions in bright white
|
||||||
|
* allow configuration file and settings TUI hotkeys wrt #41
|
||||||
|
|
||||||
|
## 0.1.11 -- 2020-09-23
|
||||||
|
|
||||||
|
* Add support for installing haskell-language-server, wrt #65
|
||||||
|
* When compiling GHC from source create a bindist first, store that bindist in `~/.ghcup/cache` and install it, wrt #51
|
||||||
|
* Allow to compile over existing version (`ghcup compile ghc -v 8.6.5 -b 8.6.5`) and replace it wrt #59
|
||||||
|
* simplify installing from custom bindist wrt #60
|
||||||
|
- `ghcup install ghc -u <url> <version>`
|
||||||
|
* fix bug when cabal isn't marked executable in bindist
|
||||||
|
* fix bug when `~/.ghcup` is a valid symlink wrt #49
|
||||||
|
* Drop support for compiling cabal from source (the old bootstrap script is discontinued)
|
||||||
|
|
||||||
|
## 0.1.10 -- 2020-08-14
|
||||||
|
|
||||||
|
* Show stray Cabals (useful for pre-releases or compiled ones)
|
||||||
|
|
||||||
|
## 0.1.9 -- 2020-08-14
|
||||||
|
|
||||||
|
* Fix bug when uninstalling all cabal versions
|
||||||
|
* Fix bug when setting a non-installed ghc version as current default
|
||||||
* Use yaml instead of generated json for download info for ease of adding new GHC versions #44
|
* Use yaml instead of generated json for download info for ease of adding new GHC versions #44
|
||||||
* Allow pre-release versions of GHC/cabal
|
* Allow pre-release versions of GHC/cabal
|
||||||
* Add XDG dirs support (set `GHCUP_USE_XDG_DIRS`) wrt #39
|
* Add XDG dirs support (set `GHCUP_USE_XDG_DIRS`) wrt #39
|
||||||
* Allow to specify regex for tarball subdir (e.g. `ghc-.*`)
|
* Allow to specify regex for tarball subdir (e.g. `ghc-.*`)
|
||||||
* Allow installing arbitrary bindists more seamlessly:
|
* Allow installing arbitrary bindists more seamlessly:
|
||||||
- e.g. installing GHC HEAD: `ghcup -c -n install ghc -u '{"dlHash": "", "dlSubdir": { "RegexDir": "ghc-.*"}, "dlUri": "https://gitlab.haskell.org/api/v4/projects/1/jobs/artifacts/master/raw/ghc-x86_64-fedora27-linux.tar.xz?job=validate-x86_64-linux-fedora27" }' head`
|
- e.g. installing GHC HEAD: `ghcup -n install ghc -u '{"dlHash": "", "dlSubdir": { "RegexDir": "ghc-.*"}, "dlUri": "https://gitlab.haskell.org/api/v4/projects/1/jobs/artifacts/master/raw/ghc-x86_64-fedora27-linux.tar.xz?job=validate-x86_64-linux-fedora27" }' head`
|
||||||
* Avoid duplicate edits to .bashrc/.zshrc wrt #43
|
* Avoid duplicate edits to .bashrc/.zshrc wrt #43
|
||||||
|
|
||||||
## 0.1.8 -- 2020-07-21
|
## 0.1.8 -- 2020-07-21
|
||||||
|
|||||||
45
README.md
45
README.md
@@ -9,11 +9,16 @@ Similar in scope to [rustup](https://github.com/rust-lang-nursery/rustup.rs), [p
|
|||||||
## Table of Contents
|
## Table of Contents
|
||||||
|
|
||||||
* [Installation](#installation)
|
* [Installation](#installation)
|
||||||
|
* [Simple bootstrap](#simple-bootstrap)
|
||||||
|
* [Manual install](#manual-install)
|
||||||
|
* [Vim integration](#vim-integration)
|
||||||
* [Usage](#usage)
|
* [Usage](#usage)
|
||||||
|
* [Configuration](#configuration)
|
||||||
* [Manpages](#manpages)
|
* [Manpages](#manpages)
|
||||||
* [Shell-completion](#shell-completion)
|
* [Shell-completion](#shell-completion)
|
||||||
* [Cross support](#cross-support)
|
* [Cross support](#cross-support)
|
||||||
* [XDG support](#xdg-support)
|
* [XDG support](#xdg-support)
|
||||||
|
* [Installing custom bindists](#installing-custom-bindists)
|
||||||
* [Design goals](#design-goals)
|
* [Design goals](#design-goals)
|
||||||
* [How](#how)
|
* [How](#how)
|
||||||
* [Known users](#known-users)
|
* [Known users](#known-users)
|
||||||
@@ -37,6 +42,10 @@ Then adjust your `PATH` in `~/.bashrc` (or similar, depending on your shell) lik
|
|||||||
export PATH="$HOME/.cabal/bin:$HOME/.ghcup/bin:$PATH"
|
export PATH="$HOME/.cabal/bin:$HOME/.ghcup/bin:$PATH"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Vim integration
|
||||||
|
|
||||||
|
See [ghcup.vim](https://github.com/hasufell/ghcup.vim).
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
See `ghcup --help`.
|
See `ghcup --help`.
|
||||||
@@ -72,6 +81,13 @@ ghcup upgrade
|
|||||||
Generally this is meant to be used with [`cabal-install`](https://hackage.haskell.org/package/cabal-install), which
|
Generally this is meant to be used with [`cabal-install`](https://hackage.haskell.org/package/cabal-install), which
|
||||||
handles your haskell packages and can demand that [a specific version](https://cabal.readthedocs.io/en/latest/nix-local-build.html#cfg-flag---with-compiler) of `ghc` is available, which `ghcup` can do.
|
handles your haskell packages and can demand that [a specific version](https://cabal.readthedocs.io/en/latest/nix-local-build.html#cfg-flag---with-compiler) of `ghc` is available, which `ghcup` can do.
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
A configuration file can be put in `~/.ghcup/config.yaml`. The default config file
|
||||||
|
explaining all possible configurations can be found in this repo: [config.yaml](./config.yaml).
|
||||||
|
|
||||||
|
Partial configuration is fine. Command line options always overwrite the config file settings.
|
||||||
|
|
||||||
### Manpages
|
### Manpages
|
||||||
|
|
||||||
For man pages to work you need [man-db](http://man-db.nongnu.org/) as your `man` provider, then issue `man ghc`. Manpages only work for the currently set ghc.
|
For man pages to work you need [man-db](http://man-db.nongnu.org/) as your `man` provider, then issue `man ghc`. Manpages only work for the currently set ghc.
|
||||||
@@ -97,7 +113,7 @@ For distributions with non-standard locations of cross toolchain and
|
|||||||
libraries, this may need some tweaking of `build.mk` or configure args.
|
libraries, this may need some tweaking of `build.mk` or configure args.
|
||||||
See `ghcup compile ghc --help` for further information.
|
See `ghcup compile ghc --help` for further information.
|
||||||
|
|
||||||
### Cross support
|
### XDG support
|
||||||
|
|
||||||
To enable XDG style directories, set the environment variable `GHCUP_USE_XDG_DIRS` to anything.
|
To enable XDG style directories, set the environment variable `GHCUP_USE_XDG_DIRS` to anything.
|
||||||
|
|
||||||
@@ -107,6 +123,22 @@ Then you can control the locations via XDG environment variables as such:
|
|||||||
* `XDG_CACHE_HOME`: logs and download files will be stored in `ghcup` subdir
|
* `XDG_CACHE_HOME`: logs and download files will be stored in `ghcup` subdir
|
||||||
* `XDG_BIN_HOME`: binaries end up here (default: `~/.local/bin`)
|
* `XDG_BIN_HOME`: binaries end up here (default: `~/.local/bin`)
|
||||||
|
|
||||||
|
### Installing custom bindists
|
||||||
|
|
||||||
|
There are a couple of good use cases to install custom bindists:
|
||||||
|
|
||||||
|
1. manually built bindists (e.g. with patches)
|
||||||
|
- example: `ghcup install ghc -u 'file:///home/mearwald/tmp/ghc-eff-patches/ghc-8.10.2-x86_64-deb10-linux.tar.xz' 8.10.2-eff`
|
||||||
|
2. GHC head CI bindists
|
||||||
|
- example: `ghcup install ghc -u 'https://gitlab.haskell.org/api/v4/projects/1/jobs/artifacts/master/raw/ghc-x86_64-fedora27-linux.tar.xz?job=validate-x86_64-linux-fedora27' head`
|
||||||
|
3. DWARF bindists
|
||||||
|
- example: `ghcup install ghc -u 'https://downloads.haskell.org/~ghc/8.10.2/ghc-8.10.2-x86_64-deb10-linux-dwarf.tar.xz' 8.10.2-dwarf`
|
||||||
|
|
||||||
|
Since the version parser is pretty lax, `8.10.2-eff` and `head` are both valid versions
|
||||||
|
and produce the binaries `ghc-8.10.2-eff` and `ghc-head` respectively.
|
||||||
|
GHCup always needs to know which version the bindist corresponds to (this is not automatically
|
||||||
|
detected).
|
||||||
|
|
||||||
## Design goals
|
## Design goals
|
||||||
|
|
||||||
1. simplicity
|
1. simplicity
|
||||||
@@ -139,6 +171,17 @@ In addition this script can also install `cabal-install`.
|
|||||||
|
|
||||||
## Known problems
|
## Known problems
|
||||||
|
|
||||||
|
### Custom ghc version names
|
||||||
|
|
||||||
|
When installing ghc bindists with custom version names as outlined in
|
||||||
|
[installing custom bindists](#installing-custom-bindists), then cabal might
|
||||||
|
be unable to find the correct `ghc-pkg` (also see [#73](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/73))
|
||||||
|
if you use `cabal build --with-compiler=ghc-foo`. Instead, point it to the full path, such as:
|
||||||
|
`cabal build --with-compiler=$HOME/.ghcup/ghc/<version-name>/bin/ghc` or set that GHC version
|
||||||
|
as the current one via: `ghcup set ghc <version-name>`.
|
||||||
|
|
||||||
|
This problem doesn't exist for regularly installed GHC versions.
|
||||||
|
|
||||||
### Limited distributions supported
|
### Limited distributions supported
|
||||||
|
|
||||||
Currently only GNU/Linux distributions compatible with the [upstream GHC](https://www.haskell.org/ghc/download_ghc_8_6_1.html#binaries) binaries are supported.
|
Currently only GNU/Linux distributions compatible with the [upstream GHC](https://www.haskell.org/ghc/download_ghc_8_6_1.html#binaries) binaries are supported.
|
||||||
|
|||||||
@@ -55,7 +55,7 @@ validate :: (Monad m, MonadLogger m, MonadThrow m, MonadIO m, MonadUnliftIO m)
|
|||||||
validate dls = do
|
validate dls = do
|
||||||
ref <- liftIO $ newIORef 0
|
ref <- liftIO $ newIORef 0
|
||||||
|
|
||||||
-- * verify binary downloads * --
|
-- verify binary downloads --
|
||||||
flip runReaderT ref $ do
|
flip runReaderT ref $ do
|
||||||
-- unique tags
|
-- unique tags
|
||||||
forM_ (M.toList dls) $ \(t, _) -> checkUniqueTags t
|
forM_ (M.toList dls) $ \(t, _) -> checkUniqueTags t
|
||||||
@@ -122,6 +122,7 @@ validate dls = do
|
|||||||
where
|
where
|
||||||
isUniqueTag Latest = True
|
isUniqueTag Latest = True
|
||||||
isUniqueTag Recommended = True
|
isUniqueTag Recommended = True
|
||||||
|
isUniqueTag Old = False
|
||||||
isUniqueTag Prerelease = False
|
isUniqueTag Prerelease = False
|
||||||
isUniqueTag (Base _) = False
|
isUniqueTag (Base _) = False
|
||||||
isUniqueTag (UnknownTag _) = False
|
isUniqueTag (UnknownTag _) = False
|
||||||
@@ -192,7 +193,7 @@ validateTarballs dls = do
|
|||||||
where
|
where
|
||||||
downloadAll dli = do
|
downloadAll dli = do
|
||||||
dirs <- liftIO getDirs
|
dirs <- liftIO getDirs
|
||||||
let settings = Settings True False Never Curl False dirs
|
let settings = AppState (Settings True False Never Curl False GHCupURL) dirs defaultKeyBindings
|
||||||
let runLogger = myLoggerT LoggerConfig { lcPrintDebug = True
|
let runLogger = myLoggerT LoggerConfig { lcPrintDebug = True
|
||||||
, colorOutter = B.hPut stderr
|
, colorOutter = B.hPut stderr
|
||||||
, rawOutter = (\_ -> pure ())
|
, rawOutter = (\_ -> pure ())
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
{-# LANGUAGE CPP #-}
|
{-# LANGUAGE CPP #-}
|
||||||
{-# LANGUAGE DataKinds #-}
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE TypeApplications #-}
|
|
||||||
{-# LANGUAGE TemplateHaskell #-}
|
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
{-# LANGUAGE TypeApplications #-}
|
||||||
|
{-# LANGUAGE ViewPatterns #-}
|
||||||
|
|
||||||
module BrickMain where
|
module BrickMain where
|
||||||
|
|
||||||
@@ -19,7 +21,10 @@ import Brick
|
|||||||
import Brick.Widgets.Border
|
import Brick.Widgets.Border
|
||||||
import Brick.Widgets.Border.Style
|
import Brick.Widgets.Border.Style
|
||||||
import Brick.Widgets.Center
|
import Brick.Widgets.Center
|
||||||
import Brick.Widgets.List
|
import Brick.Widgets.List ( listSelectedFocusedAttr
|
||||||
|
, listSelectedAttr
|
||||||
|
, listAttr
|
||||||
|
)
|
||||||
#if !defined(TAR)
|
#if !defined(TAR)
|
||||||
import Codec.Archive
|
import Codec.Archive
|
||||||
#endif
|
#endif
|
||||||
@@ -31,10 +36,11 @@ import Data.Bool
|
|||||||
import Data.Functor
|
import Data.Functor
|
||||||
import Data.List
|
import Data.List
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.Char
|
|
||||||
import Data.IORef
|
import Data.IORef
|
||||||
import Data.String.Interpolate
|
import Data.String.Interpolate
|
||||||
import Data.Vector ( Vector )
|
import Data.Vector ( Vector
|
||||||
|
, (!?)
|
||||||
|
)
|
||||||
import Data.Versions hiding ( str )
|
import Data.Versions hiding ( str )
|
||||||
import Haskus.Utils.Variant.Excepts
|
import Haskus.Utils.Variant.Excepts
|
||||||
import Prelude hiding ( appendFile )
|
import Prelude hiding ( appendFile )
|
||||||
@@ -47,43 +53,94 @@ import qualified Graphics.Vty as Vty
|
|||||||
import qualified Data.Vector as V
|
import qualified Data.Vector as V
|
||||||
|
|
||||||
|
|
||||||
data AppState = AppState {
|
|
||||||
lr :: LR
|
data BrickData = BrickData
|
||||||
, dls :: GHCupDownloads
|
{ lr :: [ListResult]
|
||||||
|
, dls :: GHCupDownloads
|
||||||
, pfreq :: PlatformRequest
|
, pfreq :: PlatformRequest
|
||||||
}
|
}
|
||||||
|
deriving Show
|
||||||
|
|
||||||
type LR = GenericList String Vector ListResult
|
data BrickSettings = BrickSettings
|
||||||
|
{ showAll :: Bool
|
||||||
|
}
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
data BrickInternalState = BrickInternalState
|
||||||
|
{ clr :: Vector ListResult
|
||||||
|
, ix :: Int
|
||||||
|
}
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
data BrickState = BrickState
|
||||||
|
{ appData :: BrickData
|
||||||
|
, appSettings :: BrickSettings
|
||||||
|
, appState :: BrickInternalState
|
||||||
|
, appKeys :: KeyBindings
|
||||||
|
}
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
|
||||||
keyHandlers :: [(Char, String, AppState -> EventM n (Next AppState))]
|
keyHandlers :: KeyBindings
|
||||||
keyHandlers =
|
-> [ ( Vty.Key
|
||||||
[ ('q', "Quit" , halt)
|
, BrickSettings -> String
|
||||||
, ('i', "Install" , withIOAction install')
|
, BrickState -> EventM n (Next BrickState)
|
||||||
, ('u', "Uninstall", withIOAction del')
|
)
|
||||||
, ('s', "Set" , withIOAction set')
|
]
|
||||||
, ('c', "ChangeLog", withIOAction changelog')
|
keyHandlers KeyBindings {..} =
|
||||||
|
[ (bQuit, const "Quit" , halt)
|
||||||
|
, (bInstall, const "Install" , withIOAction install')
|
||||||
|
, (bUninstall, const "Uninstall", withIOAction del')
|
||||||
|
, (bSet, const "Set" , withIOAction set')
|
||||||
|
, (bChangelog, const "ChangeLog", withIOAction changelog')
|
||||||
|
, ( bShowAll
|
||||||
|
, (\BrickSettings {..} ->
|
||||||
|
if showAll then "Hide old versions" else "Show all versions"
|
||||||
|
)
|
||||||
|
, hideShowHandler
|
||||||
|
)
|
||||||
|
, (bUp, const "Up", \BrickState {..} -> continue (BrickState { appState = (moveCursor 1 appState Up), .. }))
|
||||||
|
, (bDown, const "Down", \BrickState {..} -> continue (BrickState { appState = (moveCursor 1 appState Down), .. }))
|
||||||
]
|
]
|
||||||
|
where
|
||||||
|
hideShowHandler (BrickState {..}) =
|
||||||
|
let newAppSettings = appSettings { showAll = not . showAll $ appSettings }
|
||||||
|
newInternalState = constructList appData newAppSettings (Just appState)
|
||||||
|
in continue (BrickState appData newAppSettings newInternalState appKeys)
|
||||||
|
|
||||||
|
|
||||||
ui :: AppState -> Widget String
|
showKey :: Vty.Key -> String
|
||||||
ui AppState {..} =
|
showKey (Vty.KChar c) = [c]
|
||||||
( padBottom Max
|
showKey (Vty.KUp) = "↑"
|
||||||
|
showKey (Vty.KDown) = "↓"
|
||||||
|
showKey key = tail (show key)
|
||||||
|
|
||||||
|
|
||||||
|
ui :: BrickState -> Widget String
|
||||||
|
ui BrickState { appSettings = as@(BrickSettings {}), ..}
|
||||||
|
= ( padBottom Max
|
||||||
$ ( withBorderStyle unicode
|
$ ( withBorderStyle unicode
|
||||||
$ borderWithLabel (str "GHCup")
|
$ borderWithLabel (str "GHCup")
|
||||||
$ (center $ renderList renderItem True lr)
|
$ (center $ (header <=> hBorder <=> renderList' appState))
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
<=> ( withAttr "help"
|
<=> footer
|
||||||
. txtWrap
|
|
||||||
. T.pack
|
|
||||||
. foldr1 (\x y -> x <> " " <> y)
|
|
||||||
. (++ ["↑↓:Navigation"])
|
|
||||||
$ (fmap (\(c, s, _) -> (c : ':' : s)) keyHandlers)
|
|
||||||
)
|
|
||||||
|
|
||||||
where
|
where
|
||||||
renderItem b ListResult {..} =
|
footer =
|
||||||
|
withAttr "help"
|
||||||
|
. txtWrap
|
||||||
|
. T.pack
|
||||||
|
. foldr1 (\x y -> x <> " " <> y)
|
||||||
|
$ (fmap (\(key, s, _) -> (showKey key <> ":" <> s as)) $ keyHandlers appKeys)
|
||||||
|
header =
|
||||||
|
(minHSize 2 $ emptyWidget)
|
||||||
|
<+> (padLeft (Pad 2) $ minHSize 6 $ str "Tool")
|
||||||
|
<+> (minHSize 15 $ str "Version")
|
||||||
|
<+> (padLeft (Pad 1) $ minHSize 25 $ str "Tags")
|
||||||
|
<+> (padLeft (Pad 5) $ str "Notes")
|
||||||
|
renderList' = withDefAttr listAttr . drawListElements renderItem True
|
||||||
|
renderItem _ b listResult@(ListResult {..}) =
|
||||||
let marks = if
|
let marks = if
|
||||||
| lSet -> (withAttr "set" $ str "✔✔")
|
| lSet -> (withAttr "set" $ str "✔✔")
|
||||||
| lInstalled -> (withAttr "installed" $ str "✓ ")
|
| lInstalled -> (withAttr "installed" $ str "✓ ")
|
||||||
@@ -94,34 +151,93 @@ ui AppState {..} =
|
|||||||
dim = if lNoBindist
|
dim = if lNoBindist
|
||||||
then updateAttrMap (const dimAttributes) . withAttr "no-bindist"
|
then updateAttrMap (const dimAttributes) . withAttr "no-bindist"
|
||||||
else id
|
else id
|
||||||
in dim
|
hooray
|
||||||
|
| elem Latest lTag && not lInstalled =
|
||||||
|
withAttr "hooray"
|
||||||
|
| otherwise = id
|
||||||
|
active = if b then forceAttr "active" else id
|
||||||
|
in hooray $ active $ dim
|
||||||
( marks
|
( marks
|
||||||
<+> ( padLeft (Pad 2)
|
<+> (( padLeft (Pad 2)
|
||||||
$ minHSize 20
|
$ minHSize 6
|
||||||
$ ((if b then withAttr "active" else id)
|
$ (printTool lTool)
|
||||||
(str $ (fmap toLower . show $ lTool) <> " " <> ver)
|
)
|
||||||
)
|
|
||||||
)
|
)
|
||||||
<+> (padLeft (Pad 1) $ if null lTag
|
<+> (minHSize 15 $ (str ver))
|
||||||
then emptyWidget
|
<+> (let l = catMaybes . fmap printTag $ sort lTag
|
||||||
else
|
in padLeft (Pad 1) $ minHSize 25 $ if null l
|
||||||
foldr1 (\x y -> x <+> str "," <+> y)
|
then emptyWidget
|
||||||
$ (fmap printTag $ sort lTag)
|
else foldr1 (\x y -> x <+> str "," <+> y) l
|
||||||
)
|
)
|
||||||
|
<+> ( padLeft (Pad 5)
|
||||||
|
$ let notes = printNotes listResult
|
||||||
|
in if null notes
|
||||||
|
then emptyWidget
|
||||||
|
else foldr1 (\x y -> x <+> str "," <+> y) $ notes
|
||||||
|
)
|
||||||
|
<+> (vLimit 1 $ fill ' ')
|
||||||
)
|
)
|
||||||
|
|
||||||
printTag Recommended = withAttr "recommended" $ str "recommended"
|
printTag Recommended = Just $ withAttr "recommended" $ str "recommended"
|
||||||
printTag Latest = withAttr "latest" $ str "latest"
|
printTag Latest = Just $ withAttr "latest" $ str "latest"
|
||||||
printTag Prerelease = withAttr "prerelease" $ str "prerelease"
|
printTag Prerelease = Just $ withAttr "prerelease" $ str "prerelease"
|
||||||
printTag (Base pvp'') = str ("base-" ++ T.unpack (prettyPVP pvp''))
|
printTag (Base pvp'') = Just $ str ("base-" ++ T.unpack (prettyPVP pvp''))
|
||||||
printTag (UnknownTag t ) = str t
|
printTag Old = Nothing
|
||||||
|
printTag (UnknownTag t) = Just $ str t
|
||||||
|
|
||||||
|
printTool Cabal = str "cabal"
|
||||||
|
printTool GHC = str "GHC"
|
||||||
|
printTool GHCup = str "GHCup"
|
||||||
|
printTool HLS = str "HLS"
|
||||||
|
|
||||||
|
printNotes ListResult {..} =
|
||||||
|
(if hlsPowered then [withAttr "hls-powered" $ str "hls-powered"] else mempty
|
||||||
|
)
|
||||||
|
++ (if fromSrc then [withAttr "compiled" $ str "compiled"] else mempty)
|
||||||
|
++ (if lStray then [withAttr "stray" $ str "stray"] else mempty)
|
||||||
|
|
||||||
|
-- | Draws the list elements.
|
||||||
|
--
|
||||||
|
-- Evaluates the underlying container up to, and a bit beyond, the
|
||||||
|
-- selected element. The exact amount depends on available height
|
||||||
|
-- for drawing and 'listItemHeight'. At most, it will evaluate up to
|
||||||
|
-- element @(i + h + 1)@ where @i@ is the selected index and @h@ is the
|
||||||
|
-- available height.
|
||||||
|
drawListElements :: (Int -> Bool -> ListResult -> Widget String)
|
||||||
|
-> Bool
|
||||||
|
-> BrickInternalState
|
||||||
|
-> Widget String
|
||||||
|
drawListElements drawElem foc is@(BrickInternalState clr _) =
|
||||||
|
Widget Greedy Greedy $
|
||||||
|
let
|
||||||
|
es = clr
|
||||||
|
listSelected = fmap fst $ listSelectedElement' is
|
||||||
|
|
||||||
|
drawnElements = flip V.imap es $ \i' e ->
|
||||||
|
let addSeparator w = case es !? (i' - 1) of
|
||||||
|
Just e' | lTool e' /= lTool e ->
|
||||||
|
hBorder <=> w
|
||||||
|
_ -> w
|
||||||
|
|
||||||
|
isSelected = Just i' == listSelected
|
||||||
|
elemWidget = drawElem i' isSelected e
|
||||||
|
selItemAttr = if foc
|
||||||
|
then withDefAttr listSelectedFocusedAttr
|
||||||
|
else withDefAttr listSelectedAttr
|
||||||
|
makeVisible = if isSelected then visible . selItemAttr else id
|
||||||
|
in addSeparator $ makeVisible elemWidget
|
||||||
|
|
||||||
|
in render
|
||||||
|
$ viewport "GHCup" Vertical
|
||||||
|
$ vBox
|
||||||
|
$ V.toList drawnElements
|
||||||
|
|
||||||
|
|
||||||
minHSize :: Int -> Widget n -> Widget n
|
minHSize :: Int -> Widget n -> Widget n
|
||||||
minHSize s' = hLimit s' . vLimit 1 . (<+> fill ' ')
|
minHSize s' = hLimit s' . vLimit 1 . (<+> fill ' ')
|
||||||
|
|
||||||
|
|
||||||
app :: App AppState e String
|
app :: App BrickState e String
|
||||||
app = App { appDraw = \st -> [ui st]
|
app = App { appDraw = \st -> [ui st]
|
||||||
, appHandleEvent = eventHandler
|
, appHandleEvent = eventHandler
|
||||||
, appStartEvent = return
|
, appStartEvent = return
|
||||||
@@ -137,9 +253,13 @@ defaultAttributes = attrMap
|
|||||||
, ("set" , Vty.defAttr `Vty.withForeColor` Vty.green)
|
, ("set" , Vty.defAttr `Vty.withForeColor` Vty.green)
|
||||||
, ("installed" , Vty.defAttr `Vty.withForeColor` Vty.green)
|
, ("installed" , Vty.defAttr `Vty.withForeColor` Vty.green)
|
||||||
, ("recommended" , Vty.defAttr `Vty.withForeColor` Vty.green)
|
, ("recommended" , Vty.defAttr `Vty.withForeColor` Vty.green)
|
||||||
|
, ("hls-powered" , Vty.defAttr `Vty.withForeColor` Vty.green)
|
||||||
, ("latest" , Vty.defAttr `Vty.withForeColor` Vty.yellow)
|
, ("latest" , Vty.defAttr `Vty.withForeColor` Vty.yellow)
|
||||||
, ("prerelease" , Vty.defAttr `Vty.withForeColor` Vty.red)
|
, ("prerelease" , Vty.defAttr `Vty.withForeColor` Vty.red)
|
||||||
|
, ("compiled" , Vty.defAttr `Vty.withForeColor` Vty.blue)
|
||||||
|
, ("stray" , Vty.defAttr `Vty.withForeColor` Vty.blue)
|
||||||
, ("help" , Vty.defAttr `Vty.withStyle` Vty.italic)
|
, ("help" , Vty.defAttr `Vty.withStyle` Vty.italic)
|
||||||
|
, ("hooray" , Vty.defAttr `Vty.withForeColor` Vty.brightWhite)
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
@@ -151,78 +271,144 @@ dimAttributes = attrMap
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
eventHandler :: BrickState -> BrickEvent n e -> EventM n (Next BrickState)
|
||||||
|
eventHandler st@(BrickState {..}) ev = do
|
||||||
|
AppState { keyBindings = kb } <- liftIO $ readIORef settings'
|
||||||
|
case ev of
|
||||||
|
(MouseDown _ Vty.BScrollUp _ _) ->
|
||||||
|
continue (BrickState { appState = moveCursor 1 appState Up, .. })
|
||||||
|
(MouseDown _ Vty.BScrollDown _ _) ->
|
||||||
|
continue (BrickState { appState = moveCursor 1 appState Down, .. })
|
||||||
|
(VtyEvent (Vty.EvResize _ _)) -> continue st
|
||||||
|
(VtyEvent (Vty.EvKey Vty.KUp _)) ->
|
||||||
|
continue (BrickState { appState = (moveCursor 1 appState Up), .. })
|
||||||
|
(VtyEvent (Vty.EvKey Vty.KDown _)) ->
|
||||||
|
continue (BrickState { appState = (moveCursor 1 appState Down), .. })
|
||||||
|
(VtyEvent (Vty.EvKey key _)) ->
|
||||||
|
case find (\(key', _, _) -> key' == key) (keyHandlers kb) of
|
||||||
|
Nothing -> continue st
|
||||||
|
Just (_, _, handler) -> handler st
|
||||||
|
_ -> continue st
|
||||||
|
|
||||||
eventHandler :: AppState -> BrickEvent n e -> EventM n (Next AppState)
|
|
||||||
eventHandler st (VtyEvent (Vty.EvResize _ _)) = continue st
|
moveCursor :: Int -> BrickInternalState -> Direction -> BrickInternalState
|
||||||
eventHandler st (VtyEvent (Vty.EvKey (Vty.KChar 'q') _)) = halt st
|
moveCursor steps ais@(BrickInternalState {..}) direction =
|
||||||
eventHandler st (VtyEvent (Vty.EvKey Vty.KEsc _)) = halt st
|
let newIx = if direction == Down then ix + steps else ix - steps
|
||||||
eventHandler AppState {..} (VtyEvent (Vty.EvKey (Vty.KUp) _)) =
|
in case clr !? newIx of
|
||||||
continue (AppState (listMoveUp lr) dls pfreq)
|
Just _ -> BrickInternalState { ix = newIx, .. }
|
||||||
eventHandler AppState {..} (VtyEvent (Vty.EvKey (Vty.KDown) _)) =
|
Nothing -> ais
|
||||||
continue (AppState (listMoveDown lr) dls pfreq)
|
|
||||||
eventHandler as (VtyEvent (Vty.EvKey (Vty.KChar c) _)) =
|
|
||||||
case find (\(c', _, _) -> c' == c) keyHandlers of
|
|
||||||
Nothing -> continue as
|
|
||||||
Just (_, _, handler) -> handler as
|
|
||||||
eventHandler st _ = continue st
|
|
||||||
|
|
||||||
|
|
||||||
-- | Suspend the current UI and run an IO action in terminal. If the
|
-- | Suspend the current UI and run an IO action in terminal. If the
|
||||||
-- IO action returns a Left value, then it's thrown as userError.
|
-- IO action returns a Left value, then it's thrown as userError.
|
||||||
withIOAction :: (AppState -> (Int, ListResult) -> IO (Either String a))
|
withIOAction :: (BrickState -> (Int, ListResult) -> IO (Either String a))
|
||||||
-> AppState
|
-> BrickState
|
||||||
-> EventM n (Next AppState)
|
-> EventM n (Next BrickState)
|
||||||
withIOAction action as = case listSelectedElement (lr as) of
|
withIOAction action as = case listSelectedElement' (appState as) of
|
||||||
Nothing -> continue as
|
Nothing -> continue as
|
||||||
Just (ix, e) -> suspendAndResume $ do
|
Just (ix, e) -> suspendAndResume $ do
|
||||||
action as (ix, e) >>= \case
|
action as (ix, e) >>= \case
|
||||||
Left err -> putStrLn $ ("Error: " <> err)
|
Left err -> putStrLn $ ("Error: " <> err)
|
||||||
Right _ -> putStrLn "Success"
|
Right _ -> putStrLn "Success"
|
||||||
apps <- (fmap . fmap)
|
getAppData Nothing (pfreq . appData $ as) >>= \case
|
||||||
(\AppState {..} -> AppState { lr = listMoveTo ix lr, .. })
|
Right data' -> do
|
||||||
$ getAppState Nothing (pfreq as)
|
|
||||||
case apps of
|
|
||||||
Right nas -> do
|
|
||||||
putStrLn "Press enter to continue"
|
putStrLn "Press enter to continue"
|
||||||
_ <- getLine
|
_ <- getLine
|
||||||
pure nas
|
pure (updateList data' as)
|
||||||
Left err -> throwIO $ userError err
|
Left err -> throwIO $ userError err
|
||||||
|
|
||||||
|
|
||||||
install' :: AppState -> (Int, ListResult) -> IO (Either String ())
|
-- | Update app data and list internal state based on new evidence.
|
||||||
install' AppState {..} (_, ListResult {..}) = do
|
-- This synchronises @BrickInternalState@ with @BrickData@
|
||||||
|
-- and @BrickSettings@.
|
||||||
|
updateList :: BrickData -> BrickState -> BrickState
|
||||||
|
updateList appD (BrickState {..}) =
|
||||||
|
let newInternalState = constructList appD appSettings (Just appState)
|
||||||
|
in BrickState { appState = newInternalState
|
||||||
|
, appData = appD
|
||||||
|
, appSettings = appSettings
|
||||||
|
, appKeys = appKeys
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
constructList :: BrickData
|
||||||
|
-> BrickSettings
|
||||||
|
-> Maybe BrickInternalState
|
||||||
|
-> BrickInternalState
|
||||||
|
constructList appD appSettings mapp =
|
||||||
|
replaceLR (filterVisible (showAll appSettings)) (lr appD) mapp
|
||||||
|
|
||||||
|
listSelectedElement' :: BrickInternalState -> Maybe (Int, ListResult)
|
||||||
|
listSelectedElement' (BrickInternalState {..}) = fmap (ix, ) $ clr !? ix
|
||||||
|
|
||||||
|
|
||||||
|
selectLatest :: Vector ListResult -> Int
|
||||||
|
selectLatest v =
|
||||||
|
case V.findIndex (\ListResult {..} -> lTool == GHC && Latest `elem` lTag) v of
|
||||||
|
Just ix -> ix
|
||||||
|
Nothing -> 0
|
||||||
|
|
||||||
|
|
||||||
|
-- | Replace the @appState@ or construct it based on a filter function
|
||||||
|
-- and a new @[ListResult]@ evidence.
|
||||||
|
-- When passed an existing @appState@, tries to keep the selected element.
|
||||||
|
replaceLR :: (ListResult -> Bool)
|
||||||
|
-> [ListResult]
|
||||||
|
-> Maybe BrickInternalState
|
||||||
|
-> BrickInternalState
|
||||||
|
replaceLR filterF lr s =
|
||||||
|
let oldElem = s >>= listSelectedElement'
|
||||||
|
newVec = V.fromList . filter filterF $ lr
|
||||||
|
newSelected =
|
||||||
|
case oldElem >>= \(_, oldE) -> V.findIndex (toolEqual oldE) newVec of
|
||||||
|
Just ix -> ix
|
||||||
|
Nothing -> selectLatest newVec
|
||||||
|
in BrickInternalState newVec newSelected
|
||||||
|
where
|
||||||
|
toolEqual e1 e2 =
|
||||||
|
lTool e1 == lTool e2 && lVer e1 == lVer e2 && lCross e1 == lCross e2
|
||||||
|
|
||||||
|
|
||||||
|
filterVisible :: Bool -> ListResult -> Bool
|
||||||
|
filterVisible showAll e | lInstalled e = True
|
||||||
|
| showAll = True
|
||||||
|
| otherwise = not (elem Old (lTag e))
|
||||||
|
|
||||||
|
|
||||||
|
install' :: BrickState -> (Int, ListResult) -> IO (Either String ())
|
||||||
|
install' BrickState { appData = BrickData {..} } (_, ListResult {..}) = do
|
||||||
settings <- readIORef settings'
|
settings <- readIORef settings'
|
||||||
l <- readIORef logger'
|
l <- readIORef logger'
|
||||||
let runLogger = myLoggerT l
|
let runLogger = myLoggerT l
|
||||||
|
|
||||||
let
|
let run =
|
||||||
run =
|
runLogger
|
||||||
runLogger
|
. flip runReaderT settings
|
||||||
. flip runReaderT settings
|
. runResourceT
|
||||||
. runResourceT
|
. runE
|
||||||
. runE
|
@'[ AlreadyInstalled
|
||||||
@'[AlreadyInstalled
|
|
||||||
, UnknownArchive
|
|
||||||
#if !defined(TAR)
|
#if !defined(TAR)
|
||||||
, ArchiveResult
|
, ArchiveResult
|
||||||
#endif
|
#endif
|
||||||
, FileDoesNotExistError
|
, UnknownArchive
|
||||||
, CopyError
|
, FileDoesNotExistError
|
||||||
, NoDownload
|
, CopyError
|
||||||
, NotInstalled
|
, NoDownload
|
||||||
, BuildFailed
|
, NotInstalled
|
||||||
, TagNotFound
|
, BuildFailed
|
||||||
, DigestError
|
, TagNotFound
|
||||||
, DownloadFailed
|
, DigestError
|
||||||
, NoUpdate
|
, DownloadFailed
|
||||||
, TarDirDoesNotExist
|
, NoUpdate
|
||||||
]
|
, TarDirDoesNotExist
|
||||||
|
]
|
||||||
|
|
||||||
(run $ do
|
(run $ do
|
||||||
case lTool of
|
case lTool of
|
||||||
GHC -> liftE $ installGHCBin dls lVer pfreq
|
GHC -> liftE $ installGHCBin dls lVer pfreq
|
||||||
Cabal -> liftE $ installCabalBin dls lVer pfreq
|
Cabal -> liftE $ installCabalBin dls lVer pfreq
|
||||||
GHCup -> liftE $ upgradeGHCup dls Nothing False pfreq $> ()
|
GHCup -> liftE $ upgradeGHCup dls Nothing False pfreq $> ()
|
||||||
|
HLS -> liftE $ installHLSBin dls lVer pfreq $> ()
|
||||||
)
|
)
|
||||||
>>= \case
|
>>= \case
|
||||||
VRight _ -> pure $ Right ()
|
VRight _ -> pure $ Right ()
|
||||||
@@ -236,7 +422,7 @@ install' AppState {..} (_, ListResult {..}) = do
|
|||||||
Also check the logs in ~/.ghcup/logs|]
|
Also check the logs in ~/.ghcup/logs|]
|
||||||
|
|
||||||
|
|
||||||
set' :: AppState -> (Int, ListResult) -> IO (Either String ())
|
set' :: BrickState -> (Int, ListResult) -> IO (Either String ())
|
||||||
set' _ (_, ListResult {..}) = do
|
set' _ (_, ListResult {..}) = do
|
||||||
settings <- readIORef settings'
|
settings <- readIORef settings'
|
||||||
l <- readIORef logger'
|
l <- readIORef logger'
|
||||||
@@ -245,12 +431,13 @@ set' _ (_, ListResult {..}) = do
|
|||||||
let run =
|
let run =
|
||||||
runLogger
|
runLogger
|
||||||
. flip runReaderT settings
|
. flip runReaderT settings
|
||||||
. runE @'[FileDoesNotExistError, NotInstalled, TagNotFound]
|
. runE @'[FileDoesNotExistError , NotInstalled , TagNotFound]
|
||||||
|
|
||||||
(run $ do
|
(run $ do
|
||||||
case lTool of
|
case lTool of
|
||||||
GHC -> liftE $ setGHC (GHCTargetVersion lCross lVer) SetGHCOnly $> ()
|
GHC -> liftE $ setGHC (GHCTargetVersion lCross lVer) SetGHCOnly $> ()
|
||||||
Cabal -> liftE $ setCabal lVer $> ()
|
Cabal -> liftE $ setCabal lVer $> ()
|
||||||
|
HLS -> liftE $ setHLS lVer $> ()
|
||||||
GHCup -> pure ()
|
GHCup -> pure ()
|
||||||
)
|
)
|
||||||
>>= \case
|
>>= \case
|
||||||
@@ -258,7 +445,7 @@ set' _ (_, ListResult {..}) = do
|
|||||||
VLeft e -> pure $ Left [i|#{e}|]
|
VLeft e -> pure $ Left [i|#{e}|]
|
||||||
|
|
||||||
|
|
||||||
del' :: AppState -> (Int, ListResult) -> IO (Either String ())
|
del' :: BrickState -> (Int, ListResult) -> IO (Either String ())
|
||||||
del' _ (_, ListResult {..}) = do
|
del' _ (_, ListResult {..}) = do
|
||||||
settings <- readIORef settings'
|
settings <- readIORef settings'
|
||||||
l <- readIORef logger'
|
l <- readIORef logger'
|
||||||
@@ -270,6 +457,7 @@ del' _ (_, ListResult {..}) = do
|
|||||||
case lTool of
|
case lTool of
|
||||||
GHC -> liftE $ rmGHCVer (GHCTargetVersion lCross lVer) $> ()
|
GHC -> liftE $ rmGHCVer (GHCTargetVersion lCross lVer) $> ()
|
||||||
Cabal -> liftE $ rmCabalVer lVer $> ()
|
Cabal -> liftE $ rmCabalVer lVer $> ()
|
||||||
|
HLS -> liftE $ rmHLSVer lVer $> ()
|
||||||
GHCup -> pure ()
|
GHCup -> pure ()
|
||||||
)
|
)
|
||||||
>>= \case
|
>>= \case
|
||||||
@@ -277,16 +465,16 @@ del' _ (_, ListResult {..}) = do
|
|||||||
VLeft e -> pure $ Left [i|#{e}|]
|
VLeft e -> pure $ Left [i|#{e}|]
|
||||||
|
|
||||||
|
|
||||||
changelog' :: AppState -> (Int, ListResult) -> IO (Either String ())
|
changelog' :: BrickState -> (Int, ListResult) -> IO (Either String ())
|
||||||
changelog' AppState {..} (_, ListResult {..}) = do
|
changelog' BrickState { appData = BrickData {..} } (_, ListResult {..}) = do
|
||||||
case getChangeLog dls lTool (Left lVer) of
|
case getChangeLog dls lTool (Left lVer) of
|
||||||
Nothing -> pure $ Left
|
Nothing -> pure $ Left
|
||||||
[i|Could not find ChangeLog for #{lTool}, version #{prettyVer lVer}|]
|
[i|Could not find ChangeLog for #{lTool}, version #{prettyVer lVer}|]
|
||||||
Just uri -> do
|
Just uri -> do
|
||||||
let cmd = case _rPlatform pfreq of
|
let cmd = case _rPlatform pfreq of
|
||||||
Darwin -> "open"
|
Darwin -> "open"
|
||||||
Linux _ -> "xdg-open"
|
Linux _ -> "xdg-open"
|
||||||
FreeBSD -> "xdg-open"
|
FreeBSD -> "xdg-open"
|
||||||
exec cmd True [serializeURIRef' uri] Nothing Nothing >>= \case
|
exec cmd True [serializeURIRef' uri] Nothing Nothing >>= \case
|
||||||
Right _ -> pure $ Right ()
|
Right _ -> pure $ Right ()
|
||||||
Left e -> pure $ Left [i|#{e}|]
|
Left e -> pure $ Left [i|#{e}|]
|
||||||
@@ -297,17 +485,21 @@ uri' :: IORef (Maybe URI)
|
|||||||
uri' = unsafePerformIO (newIORef Nothing)
|
uri' = unsafePerformIO (newIORef Nothing)
|
||||||
|
|
||||||
|
|
||||||
settings' :: IORef Settings
|
settings' :: IORef AppState
|
||||||
{-# NOINLINE settings' #-}
|
{-# NOINLINE settings' #-}
|
||||||
settings' = unsafePerformIO $ do
|
settings' = unsafePerformIO $ do
|
||||||
dirs <- getDirs
|
dirs <- getDirs
|
||||||
newIORef Settings { cache = True
|
newIORef $ AppState (Settings { cache = True
|
||||||
, noVerify = False
|
, noVerify = False
|
||||||
, keepDirs = Never
|
, keepDirs = Never
|
||||||
, downloader = Curl
|
, downloader = Curl
|
||||||
, verbose = False
|
, verbose = False
|
||||||
, ..
|
, urlSource = GHCupURL
|
||||||
}
|
, ..
|
||||||
|
})
|
||||||
|
dirs
|
||||||
|
defaultKeyBindings
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
logger' :: IORef LoggerConfig
|
logger' :: IORef LoggerConfig
|
||||||
@@ -320,7 +512,12 @@ logger' = unsafePerformIO
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
brickMain :: Settings -> Maybe URI -> LoggerConfig -> GHCupDownloads -> PlatformRequest -> IO ()
|
brickMain :: AppState
|
||||||
|
-> Maybe URI
|
||||||
|
-> LoggerConfig
|
||||||
|
-> GHCupDownloads
|
||||||
|
-> PlatformRequest
|
||||||
|
-> IO ()
|
||||||
brickMain s muri l av pfreq' = do
|
brickMain s muri l av pfreq' = do
|
||||||
writeIORef uri' muri
|
writeIORef uri' muri
|
||||||
writeIORef settings' s
|
writeIORef settings' s
|
||||||
@@ -328,23 +525,29 @@ brickMain s muri l av pfreq' = do
|
|||||||
writeIORef logger' l
|
writeIORef logger' l
|
||||||
let runLogger = myLoggerT l
|
let runLogger = myLoggerT l
|
||||||
|
|
||||||
eApps <- getAppState (Just av) pfreq'
|
eAppData <- getAppData (Just av) pfreq'
|
||||||
case eApps of
|
case eAppData of
|
||||||
Right as -> defaultMain app (selectLatest as) $> ()
|
Right ad ->
|
||||||
Left e -> do
|
defaultMain
|
||||||
|
app
|
||||||
|
(BrickState ad
|
||||||
|
defaultAppSettings
|
||||||
|
(constructList ad defaultAppSettings Nothing)
|
||||||
|
(keyBindings s)
|
||||||
|
|
||||||
|
)
|
||||||
|
$> ()
|
||||||
|
Left e -> do
|
||||||
runLogger ($(logError) [i|Error building app state: #{show e}|])
|
runLogger ($(logError) [i|Error building app state: #{show e}|])
|
||||||
exitWith $ ExitFailure 2
|
exitWith $ ExitFailure 2
|
||||||
where
|
|
||||||
selectLatest :: AppState -> AppState
|
|
||||||
selectLatest AppState {..} =
|
|
||||||
(\ix -> AppState { lr = listMoveTo ix lr, .. })
|
|
||||||
. fromJust
|
|
||||||
. V.findIndex (\ListResult {..} -> lTool == GHC && Latest `elem` lTag)
|
|
||||||
$ (listElements lr)
|
|
||||||
|
|
||||||
|
|
||||||
getAppState :: Maybe GHCupDownloads -> PlatformRequest -> IO (Either String AppState)
|
defaultAppSettings :: BrickSettings
|
||||||
getAppState mg pfreq' = do
|
defaultAppSettings = BrickSettings { showAll = False }
|
||||||
|
|
||||||
|
|
||||||
|
getDownloads' :: IO (Either String GHCupDownloads)
|
||||||
|
getDownloads' = do
|
||||||
muri <- readIORef uri'
|
muri <- readIORef uri'
|
||||||
settings <- readIORef settings'
|
settings <- readIORef settings'
|
||||||
l <- readIORef logger'
|
l <- readIORef logger'
|
||||||
@@ -353,14 +556,30 @@ getAppState mg pfreq' = do
|
|||||||
r <-
|
r <-
|
||||||
runLogger
|
runLogger
|
||||||
. flip runReaderT settings
|
. flip runReaderT settings
|
||||||
. runE
|
. runE @'[JSONError , DownloadFailed , FileDoesNotExistError]
|
||||||
@'[JSONError, DownloadFailed, FileDoesNotExistError]
|
$ fmap _ghcupDownloads
|
||||||
$ do
|
$ liftE
|
||||||
dls <- maybe (fmap _ghcupDownloads $ liftE $ getDownloadsF (maybe GHCupURL OwnSource muri)) pure mg
|
$ getDownloadsF (maybe GHCupURL OwnSource muri)
|
||||||
|
|
||||||
lV <- lift $ listVersions dls Nothing Nothing pfreq'
|
|
||||||
pure $ (AppState (list "Tool versions" (V.fromList lV) 1) dls pfreq')
|
|
||||||
|
|
||||||
case r of
|
case r of
|
||||||
VRight a -> pure $ Right a
|
VRight a -> pure $ Right a
|
||||||
VLeft e -> pure $ Left [i|#{e}|]
|
VLeft e -> pure $ Left [i|#{e}|]
|
||||||
|
|
||||||
|
|
||||||
|
getAppData :: Maybe GHCupDownloads
|
||||||
|
-> PlatformRequest
|
||||||
|
-> IO (Either String BrickData)
|
||||||
|
getAppData mg pfreq' = do
|
||||||
|
settings <- readIORef settings'
|
||||||
|
l <- readIORef logger'
|
||||||
|
let runLogger = myLoggerT l
|
||||||
|
|
||||||
|
r <- maybe getDownloads' (pure . Right) mg
|
||||||
|
|
||||||
|
runLogger . flip runReaderT settings $ do
|
||||||
|
case r of
|
||||||
|
Right dls -> do
|
||||||
|
lV <- listVersions dls Nothing Nothing pfreq'
|
||||||
|
pure $ Right $ (BrickData (reverse lV) dls pfreq')
|
||||||
|
Left e -> pure $ Left [i|#{e}|]
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -59,7 +59,7 @@ _done() {
|
|||||||
download_ghcup() {
|
download_ghcup() {
|
||||||
_plat="$(uname -s)"
|
_plat="$(uname -s)"
|
||||||
_arch=$(uname -m)
|
_arch=$(uname -m)
|
||||||
_ghver="0.1.8"
|
_ghver="0.1.11"
|
||||||
_base_url="https://downloads.haskell.org/~ghcup"
|
_base_url="https://downloads.haskell.org/~ghcup"
|
||||||
|
|
||||||
case "${_plat}" in
|
case "${_plat}" in
|
||||||
@@ -188,7 +188,30 @@ printf "\\033[0;35m%s\\033[0m\\n" ""
|
|||||||
printf "\\033[0;35m%s\\033[0m\\n" "Installation done!"
|
printf "\\033[0;35m%s\\033[0m\\n" "Installation done!"
|
||||||
printf "\\033[0;35m%s\\033[0m\\n" ""
|
printf "\\033[0;35m%s\\033[0m\\n" ""
|
||||||
|
|
||||||
|
|
||||||
if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
||||||
|
printf "\\033[0;35m%s\\033[0m\\n" "Do you want to install haskell-language-server (HLS) now?"
|
||||||
|
printf "\\033[0;35m%s\\033[0m\\n" "HLS is a language-server that provides IDE-like functionality"
|
||||||
|
printf "\\033[0;35m%s\\033[0m\\n" "and can integrate with different editors, such as Vim, Emacs, VS Code, Atom, ..."
|
||||||
|
printf "\\033[0;35m%s\\033[0m\\n" "Also see https://github.com/haskell/haskell-language-server/blob/master/README.md"
|
||||||
|
printf "\\033[0;35m%s\\033[0m\\n" ""
|
||||||
|
printf "\\033[0;35m%s\\033[0m\\n" "Answer with YES or NO and press ENTER."
|
||||||
|
printf "\\033[0;35m%s\\033[0m\\n" ""
|
||||||
|
|
||||||
|
while true; do
|
||||||
|
read -r hls_answer </dev/tty
|
||||||
|
|
||||||
|
case $hls_answer in
|
||||||
|
[Yy]*)
|
||||||
|
eghcup --cache install hls
|
||||||
|
break ;;
|
||||||
|
[Nn]*)
|
||||||
|
break ;;
|
||||||
|
*)
|
||||||
|
echo "Please type YES or NO and press enter.";;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
echo "In order to run ghc and cabal, you need to adjust your PATH variable."
|
echo "In order to run ghc and cabal, you need to adjust your PATH variable."
|
||||||
echo "You may want to source '$GHCUP_DIR/env' in your shell"
|
echo "You may want to source '$GHCUP_DIR/env' in your shell"
|
||||||
echo "configuration to do so (e.g. ~/.bashrc)."
|
echo "configuration to do so (e.g. ~/.bashrc)."
|
||||||
@@ -235,10 +258,23 @@ if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
|||||||
if ! grep -q "ghcup-env" "${GHCUP_PROFILE_FILE}" ; then
|
if ! grep -q "ghcup-env" "${GHCUP_PROFILE_FILE}" ; then
|
||||||
echo "# ghcup-env" >> "${GHCUP_PROFILE_FILE}"
|
echo "# ghcup-env" >> "${GHCUP_PROFILE_FILE}"
|
||||||
echo "set -q GHCUP_INSTALL_BASE_PREFIX[1]; or set GHCUP_INSTALL_BASE_PREFIX \$HOME" >> "${GHCUP_PROFILE_FILE}"
|
echo "set -q GHCUP_INSTALL_BASE_PREFIX[1]; or set GHCUP_INSTALL_BASE_PREFIX \$HOME" >> "${GHCUP_PROFILE_FILE}"
|
||||||
echo "test -f $GHCUP_DIR/env ; and set -gx PATH \$HOME/.cabal/bin $GHCUP_BIN/bin \$PATH" >> "${GHCUP_PROFILE_FILE}"
|
echo "test -f $GHCUP_DIR/env ; and set -gx PATH \$HOME/.cabal/bin $GHCUP_BIN \$PATH" >> "${GHCUP_PROFILE_FILE}"
|
||||||
fi
|
fi
|
||||||
break ;;
|
break ;;
|
||||||
*)
|
bash)
|
||||||
|
if ! grep -q "ghcup-env" "${GHCUP_PROFILE_FILE}" ; then
|
||||||
|
echo "[ -f \"${GHCUP_DIR}/env\" ] && source \"${GHCUP_DIR}/env\" # ghcup-env" >> "${GHCUP_PROFILE_FILE}"
|
||||||
|
fi
|
||||||
|
case "$(uname -s)" in
|
||||||
|
"Darwin"|"darwin")
|
||||||
|
if ! grep -q "ghcup-env" "${HOME}/.bash_profile" ; then
|
||||||
|
echo "[[ -f ~/.bashrc ]] && source ~/.bashrc # ghcup-env" >> "${HOME}/.bash_profile"
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
break ;;
|
||||||
|
|
||||||
|
zsh)
|
||||||
if ! grep -q "ghcup-env" "${GHCUP_PROFILE_FILE}" ; then
|
if ! grep -q "ghcup-env" "${GHCUP_PROFILE_FILE}" ; then
|
||||||
echo "[ -f \"${GHCUP_DIR}/env\" ] && source \"${GHCUP_DIR}/env\" # ghcup-env" >> "${GHCUP_PROFILE_FILE}"
|
echo "[ -f \"${GHCUP_DIR}/env\" ] && source \"${GHCUP_DIR}/env\" # ghcup-env" >> "${GHCUP_PROFILE_FILE}"
|
||||||
fi
|
fi
|
||||||
|
|||||||
47
cabal.ghc884.project
Normal file
47
cabal.ghc884.project
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
-- Generated by stackage-to-hackage
|
||||||
|
|
||||||
|
index-state: 2020-10-24T20:53:55Z
|
||||||
|
|
||||||
|
with-compiler: ghc-8.8.4
|
||||||
|
|
||||||
|
packages:
|
||||||
|
./
|
||||||
|
, 3rdparty/lzma/
|
||||||
|
, 3rdparty/lzma-clib/
|
||||||
|
, 3rdparty/zlib/
|
||||||
|
|
||||||
|
source-repository-package
|
||||||
|
type: git
|
||||||
|
location: https://github.com/haskus/packages.git
|
||||||
|
tag: 80a1c5fc07f7226c424250ec17f674cd4d618f42
|
||||||
|
subdir: haskus-utils-types
|
||||||
|
|
||||||
|
source-repository-package
|
||||||
|
type: git
|
||||||
|
location: https://github.com/hasufell/hpath.git
|
||||||
|
tag: bf6d28cf989b70286e12fecc183d5bbf5454a1a2
|
||||||
|
subdir: hpath-directory
|
||||||
|
hpath-io
|
||||||
|
|
||||||
|
source-repository-package
|
||||||
|
type: git
|
||||||
|
location: https://github.com/hasufell/text-conversions.git
|
||||||
|
tag: 9abf0e5e5664a3178367597c32db19880477a53c
|
||||||
|
|
||||||
|
allow-older: *
|
||||||
|
allow-newer: *
|
||||||
|
|
||||||
|
package lzma
|
||||||
|
ghc-options: -O2
|
||||||
|
|
||||||
|
package lzma-clib
|
||||||
|
ghc-options: -O2
|
||||||
|
|
||||||
|
package zlib
|
||||||
|
ghc-options: -O2
|
||||||
|
|
||||||
|
package ghcup
|
||||||
|
ghc-options: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
|
|
||||||
|
package streamly
|
||||||
|
ghc-options: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
2521
cabal.ghc884.project.freeze
Normal file
2521
cabal.ghc884.project.freeze
Normal file
File diff suppressed because it is too large
Load Diff
@@ -8,6 +8,18 @@ source-repository-package
|
|||||||
tag: 80a1c5fc07f7226c424250ec17f674cd4d618f42
|
tag: 80a1c5fc07f7226c424250ec17f674cd4d618f42
|
||||||
subdir: haskus-utils-types
|
subdir: haskus-utils-types
|
||||||
|
|
||||||
|
source-repository-package
|
||||||
|
type: git
|
||||||
|
location: https://github.com/hasufell/hpath.git
|
||||||
|
tag: bf6d28cf989b70286e12fecc183d5bbf5454a1a2
|
||||||
|
subdir: hpath-io
|
||||||
|
|
||||||
|
source-repository-package
|
||||||
|
type: git
|
||||||
|
location: https://github.com/hasufell/hpath.git
|
||||||
|
tag: bf6d28cf989b70286e12fecc183d5bbf5454a1a2
|
||||||
|
subdir: hpath-directory
|
||||||
|
|
||||||
optimization: 2
|
optimization: 2
|
||||||
|
|
||||||
package streamly
|
package streamly
|
||||||
@@ -19,6 +31,6 @@ package ghcup
|
|||||||
constraints: http-io-streams -brotli
|
constraints: http-io-streams -brotli
|
||||||
|
|
||||||
package libarchive
|
package libarchive
|
||||||
flags: +static
|
flags: -system-libarchive
|
||||||
|
|
||||||
allow-newer: base, ghc-prim, template-haskell
|
allow-newer: base, ghc-prim, template-haskell
|
||||||
|
|||||||
61
config.yaml
Normal file
61
config.yaml
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
# Cache downloads in ~/.ghcup/cache
|
||||||
|
cache: False
|
||||||
|
# Skip tarball checksum verification
|
||||||
|
no-verify: False
|
||||||
|
# enable verbosity
|
||||||
|
verbose: False
|
||||||
|
# When to keep build directories
|
||||||
|
keep-dirs: Errors # Always | Never | Errors
|
||||||
|
# Which downloader to use
|
||||||
|
downloader: Curl # Curl | Wget | Internal
|
||||||
|
|
||||||
|
# TUI key bindings,
|
||||||
|
# see https://hackage.haskell.org/package/vty-5.31/docs/Graphics-Vty-Input-Events.html#t:Key
|
||||||
|
# for possible values.
|
||||||
|
key-bindings:
|
||||||
|
up:
|
||||||
|
KUp: []
|
||||||
|
down:
|
||||||
|
KDown: []
|
||||||
|
quit:
|
||||||
|
KChar: 'q'
|
||||||
|
install:
|
||||||
|
KChar: 'i'
|
||||||
|
uninstall:
|
||||||
|
KChar: 'u'
|
||||||
|
set:
|
||||||
|
KChar: 's'
|
||||||
|
changelog:
|
||||||
|
KChar: 'c'
|
||||||
|
show-all:
|
||||||
|
KChar: 'a'
|
||||||
|
|
||||||
|
# Where to get GHC/cabal/hls download info/versions from. For more detailed explanation
|
||||||
|
# check the 'URLSource' type in the code.
|
||||||
|
url-source:
|
||||||
|
## Use the internal download uri, this is the default
|
||||||
|
GHCupURL: []
|
||||||
|
|
||||||
|
## Example 1: Read download info from this location instead
|
||||||
|
## Accepts file/http/https scheme
|
||||||
|
# OwnSource: "file:///home/jule/git/ghcup-hs/ghcup-0.0.3.yaml"
|
||||||
|
|
||||||
|
## Example 2: Add custom tarballs to the default downloads, overwriting duplicate versions
|
||||||
|
# AddSource:
|
||||||
|
# Left:
|
||||||
|
# toolRequirements: {} # this is ignored
|
||||||
|
# ghcupDownloads:
|
||||||
|
# GHC:
|
||||||
|
# 9.10.2:
|
||||||
|
# viTags: []
|
||||||
|
# viArch:
|
||||||
|
# A_64:
|
||||||
|
# Linux_UnknownLinux:
|
||||||
|
# unknown_versioning:
|
||||||
|
# dlUri: https://downloads.haskell.org/~ghc/7.10.3/ghc-7.10.3-x86_64-deb8-linux.tar.bz2
|
||||||
|
# dlSubdir: ghc-7.10.3
|
||||||
|
# dlHash: 01cfbad8dff1e8b34a5fdca8caeaf843b56e36af919e29cd68870d2588563db5
|
||||||
|
|
||||||
|
## Example 3: Add a custom download file to the default downloads, overwriting duplicate versions
|
||||||
|
# AddSource:
|
||||||
|
# Right: "file:///home/jule/git/ghcup-hs/ghcup-custom.yaml"
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
# !!! if you use RegexDir, then the version must be bumped !!!
|
|
||||||
---
|
---
|
||||||
toolRequirements:
|
toolRequirements:
|
||||||
GHC:
|
GHC:
|
||||||
@@ -1160,7 +1159,7 @@ ghcupDownloads:
|
|||||||
Linux_Alpine:
|
Linux_Alpine:
|
||||||
unknown_versioning:
|
unknown_versioning:
|
||||||
dlUri: https://downloads.haskell.org/~ghc/8.10.2/ghc-8.10.2-x86_64-alpine3.10-linux-integer-simple.tar.xz
|
dlUri: https://downloads.haskell.org/~ghc/8.10.2/ghc-8.10.2-x86_64-alpine3.10-linux-integer-simple.tar.xz
|
||||||
dlSubdir: ghc-8.10.2
|
dlSubdir: ghc-8.10.2-x86_64-unknown-linux
|
||||||
dlHash: 14d09a508f2a3a11875c140be8e6c5f6982ac5cd448f089ca10b7adc955fec76
|
dlHash: 14d09a508f2a3a11875c140be8e6c5f6982ac5cd448f089ca10b7adc955fec76
|
||||||
Linux_AmazonLinux:
|
Linux_AmazonLinux:
|
||||||
unknown_versioning: *ghc-8102-64-centos
|
unknown_versioning: *ghc-8102-64-centos
|
||||||
@@ -1189,14 +1188,15 @@ ghcupDownloads:
|
|||||||
unknown_versioning: *ghc-8102-32-deb9
|
unknown_versioning: *ghc-8102-32-deb9
|
||||||
Linux_UnknownLinux:
|
Linux_UnknownLinux:
|
||||||
unknown_versioning: *ghc-8102-32-deb9
|
unknown_versioning: *ghc-8102-32-deb9
|
||||||
|
Linux_Alpine:
|
||||||
|
unknown_versioning:
|
||||||
|
dlUri: https://files.hasufell.de/ghc/ghc-8.10.2-i386-alpine-linux.tar.xz
|
||||||
|
dlSubdir: ghc-8.10.2
|
||||||
|
dlHash: 9ee1cf1e85e9536088b3c9e80e975074e525ea378cd4eb156071bbc4b7b38327
|
||||||
Cabal:
|
Cabal:
|
||||||
2.4.1.0:
|
2.4.1.0:
|
||||||
viTags: []
|
viTags: []
|
||||||
viChangeLog: https://downloads.haskell.org/~cabal/cabal-install-2.4.1.0/changelog
|
viChangeLog: https://downloads.haskell.org/~cabal/cabal-install-2.4.1.0/changelog
|
||||||
viSourceDL:
|
|
||||||
dlUri: https://github.com/haskell/cabal/archive/cabal-install-v2.4.1.0.tar.gz
|
|
||||||
dlSubdir: cabal-cabal-install-v2.4.1.0/cabal-install
|
|
||||||
dlHash: 61eb64a5addafca026aff9277291f4643fe07e83886f76d059d42c734fed829c
|
|
||||||
viArch:
|
viArch:
|
||||||
A_64:
|
A_64:
|
||||||
Linux_Alpine:
|
Linux_Alpine:
|
||||||
@@ -1228,10 +1228,6 @@ ghcupDownloads:
|
|||||||
3.0.0.0:
|
3.0.0.0:
|
||||||
viTags: []
|
viTags: []
|
||||||
viChangeLog: https://downloads.haskell.org/~cabal/cabal-install-3.0.0.0/changelog
|
viChangeLog: https://downloads.haskell.org/~cabal/cabal-install-3.0.0.0/changelog
|
||||||
viSourceDL:
|
|
||||||
dlUri: https://github.com/haskell/cabal/archive/cabal-install-v3.0.0.0.tar.gz
|
|
||||||
dlSubdir: cabal-cabal-install-v3.0.0.0/cabal-install
|
|
||||||
dlHash: c0b26817a7b7c2907e45cb38235ce1157e732211880f62e92eaff4066202e674
|
|
||||||
viArch:
|
viArch:
|
||||||
A_64:
|
A_64:
|
||||||
Linux_Alpine:
|
Linux_Alpine:
|
||||||
@@ -1264,10 +1260,6 @@ ghcupDownloads:
|
|||||||
- Recommended
|
- Recommended
|
||||||
- Latest
|
- Latest
|
||||||
viChangeLog: https://downloads.haskell.org/~cabal/cabal-install-3.2.0.0/changelog
|
viChangeLog: https://downloads.haskell.org/~cabal/cabal-install-3.2.0.0/changelog
|
||||||
viSourceDL:
|
|
||||||
dlUri: https://github.com/haskell/cabal/archive/cabal-install-v3.2.0.0.tar.gz
|
|
||||||
dlSubdir: cabal-cabal-install-v3.2.0.0/cabal-install
|
|
||||||
dlHash: 77202358bdf0b481c09326268ce18880df14194c5aaa840f99510bdd1a124b75
|
|
||||||
viArch:
|
viArch:
|
||||||
A_64:
|
A_64:
|
||||||
Linux_Alpine:
|
Linux_Alpine:
|
||||||
@@ -1295,8 +1287,32 @@ ghcupDownloads:
|
|||||||
unknown_versioning:
|
unknown_versioning:
|
||||||
dlUri: https://downloads.haskell.org/~cabal/cabal-install-3.2.0.0/cabal-install-3.2.0.0-i386-unknown-linux.tar.xz
|
dlUri: https://downloads.haskell.org/~cabal/cabal-install-3.2.0.0/cabal-install-3.2.0.0-i386-unknown-linux.tar.xz
|
||||||
dlHash: 2b3ac28549916de5f3379241797eaf60e84b6c001f2abbe73d9fadbbaf768e93
|
dlHash: 2b3ac28549916de5f3379241797eaf60e84b6c001f2abbe73d9fadbbaf768e93
|
||||||
|
3.4.0.0-rc4:
|
||||||
|
viTags:
|
||||||
|
- Prerelease
|
||||||
|
viChangeLog: https://github.com/haskell/cabal/blob/master/release-notes/cabal-install-3.4.0.0.md
|
||||||
|
viArch:
|
||||||
|
A_64:
|
||||||
|
Linux_Ubuntu:
|
||||||
|
unknown_versioning: &cabal-3400rc4-ubuntu
|
||||||
|
dlUri: https://oleg.fi/cabal-install-3.4.0.0-rc4/cabal-install-3.4.0.0-x86_64-ubuntu-16.04.tar.xz
|
||||||
|
dlHash: a1be168876816a624b206c55596d9bb5f442541c889ee2438d664698122b9ffe
|
||||||
|
Linux_Alpine:
|
||||||
|
unknown_versioning:
|
||||||
|
dlUri: https://oleg.fi/cabal-install-3.4.0.0-rc4/cabal-install-3.4.0.0-x86_64-alpine-3.11.6-static-noofd.tar.xz
|
||||||
|
dlHash: 49dab6684483594e4c7c3e561ec477268002605253ad34701b471277efbe91bc
|
||||||
|
Linux_UnknownLinux:
|
||||||
|
unknown_versioning: *cabal-3400rc4-ubuntu
|
||||||
|
Darwin:
|
||||||
|
unknown_versioning:
|
||||||
|
dlUri: https://oleg.fi/cabal-install-3.4.0.0-rc4/cabal-install-3.4.0.0-x86_64-darwin-sierra.tar.xz
|
||||||
|
dlHash: a3f809a3388e90b9fdf52444e30ea9aad3894e2cbe53c37fc3311ceb106eda9e
|
||||||
|
FreeBSD:
|
||||||
|
unknown_versioning:
|
||||||
|
dlUri: https://oleg.fi/cabal-install-3.4.0.0-rc4/cabal-install-3.4.0.0-x86_64-freebsd-12.1-release.tar.xz
|
||||||
|
dlHash: 9705e16d03497b46be4ad477e6c64d10890af853eafa8a9adf6dba89aa9e05f7
|
||||||
GHCup:
|
GHCup:
|
||||||
0.1.8:
|
0.1.11:
|
||||||
viTags:
|
viTags:
|
||||||
- Recommended
|
- Recommended
|
||||||
- Latest
|
- Latest
|
||||||
@@ -1306,22 +1322,22 @@ ghcupDownloads:
|
|||||||
A_64:
|
A_64:
|
||||||
Linux_UnknownLinux:
|
Linux_UnknownLinux:
|
||||||
unknown_versioning: &ghcup-64
|
unknown_versioning: &ghcup-64
|
||||||
dlUri: https://downloads.haskell.org/~ghcup/0.1.8/x86_64-linux-ghcup-0.1.8
|
dlUri: https://downloads.haskell.org/~ghcup/0.1.11/x86_64-linux-ghcup-0.1.11
|
||||||
dlHash: 7ffcd4c3de156e895b648c75a36c762be2a4932883f3cd598f7a483c97d4a8a9
|
dlHash: 99d97c9a1dce76892001e5cffd50cc23bf804f2282998c546d1b965aa2179699
|
||||||
Darwin:
|
Darwin:
|
||||||
unknown_versioning:
|
unknown_versioning:
|
||||||
dlUri: https://downloads.haskell.org/~ghcup/0.1.8/x86_64-apple-darwin-ghcup-0.1.8
|
dlUri: https://downloads.haskell.org/~ghcup/0.1.11/x86_64-apple-darwin-ghcup-0.1.11
|
||||||
dlHash: b6efc25013a20734e93ad7ae4ecf319f19eeee2129d515d568ccf0003f26615f
|
dlHash: 4b91dcd9bfdc40534156b8fadea3f317b3c44af1255169895f4911a221f819c6
|
||||||
FreeBSD:
|
FreeBSD:
|
||||||
unknown_versioning:
|
unknown_versioning:
|
||||||
dlUri: https://downloads.haskell.org/~ghcup/0.1.8/x86_64-portbld-freebsd-ghcup-0.1.8
|
dlUri: https://downloads.haskell.org/~ghcup/0.1.11/x86_64-portbld-freebsd-ghcup-0.1.11
|
||||||
dlHash: 442cdfe1b4525a327d9566e6270f909f7deba21c16dd4c7912537cf67e6cd521
|
dlHash: 6f04ce98d3f3eb9299ce74f8264aa956f0dc38a64a3bd12ee048b7f146e9e1b4
|
||||||
Linux_Alpine:
|
Linux_Alpine:
|
||||||
unknown_versioning: *ghcup-64
|
unknown_versioning: *ghcup-64
|
||||||
A_32:
|
A_32:
|
||||||
Linux_UnknownLinux:
|
Linux_UnknownLinux:
|
||||||
unknown_versioning: &ghcup-32
|
unknown_versioning: &ghcup-32
|
||||||
dlUri: https://downloads.haskell.org/~ghcup/0.1.8/i386-linux-ghcup-0.1.8
|
dlUri: https://downloads.haskell.org/~ghcup/0.1.11/i386-linux-ghcup-0.1.11
|
||||||
dlHash: 18ab162920cea662feae4b08f39d3879e9e416fde7b734afd8072c39d3c43cde
|
dlHash: ec339e4c2b8b4d502f66a03c0d3f112cb68cd922dd3c4a6f66323628cf6a76c2
|
||||||
Linux_Alpine:
|
Linux_Alpine:
|
||||||
unknown_versioning: *ghcup-32
|
unknown_versioning: *ghcup-32
|
||||||
|
|||||||
1449
ghcup-0.0.3.yaml
Normal file
1449
ghcup-0.0.3.yaml
Normal file
File diff suppressed because it is too large
Load Diff
68
ghcup.cabal
68
ghcup.cabal
@@ -1,6 +1,6 @@
|
|||||||
cabal-version: 3.0
|
cabal-version: 3.0
|
||||||
name: ghcup
|
name: ghcup
|
||||||
version: 0.1.9
|
version: 0.1.12
|
||||||
synopsis: ghc toolchain installer as an exe/library
|
synopsis: ghc toolchain installer as an exe/library
|
||||||
description:
|
description:
|
||||||
A rewrite of the shell script ghcup, for providing
|
A rewrite of the shell script ghcup, for providing
|
||||||
@@ -72,6 +72,9 @@ common bz2
|
|||||||
common case-insensitive
|
common case-insensitive
|
||||||
build-depends: case-insensitive >=1.2.1.0
|
build-depends: case-insensitive >=1.2.1.0
|
||||||
|
|
||||||
|
common casing
|
||||||
|
build-depends: casing >=0.1.4.1
|
||||||
|
|
||||||
common concurrent-output
|
common concurrent-output
|
||||||
build-depends: concurrent-output >=1.10.11
|
build-depends: concurrent-output >=1.10.11
|
||||||
|
|
||||||
@@ -81,6 +84,9 @@ common containers
|
|||||||
common cryptohash-sha256
|
common cryptohash-sha256
|
||||||
build-depends: cryptohash-sha256 >= 0.11.101.0
|
build-depends: cryptohash-sha256 >= 0.11.101.0
|
||||||
|
|
||||||
|
common generic-arbitrary
|
||||||
|
build-depends: generic-arbitrary >=0.1.0
|
||||||
|
|
||||||
common generics-sop
|
common generics-sop
|
||||||
build-depends: generics-sop >=0.5
|
build-depends: generics-sop >=0.5
|
||||||
|
|
||||||
@@ -94,13 +100,13 @@ common hpath
|
|||||||
build-depends: hpath >=0.11
|
build-depends: hpath >=0.11
|
||||||
|
|
||||||
common hpath-directory
|
common hpath-directory
|
||||||
build-depends: hpath-directory >=0.14
|
build-depends: hpath-directory >=0.14.1
|
||||||
|
|
||||||
common hpath-filepath
|
common hpath-filepath
|
||||||
build-depends: hpath-filepath >=0.10.3
|
build-depends: hpath-filepath >=0.10.3
|
||||||
|
|
||||||
common hpath-io
|
common hpath-io
|
||||||
build-depends: hpath-io >=0.14
|
build-depends: hpath-io >=0.14.1
|
||||||
|
|
||||||
common hpath-posix
|
common hpath-posix
|
||||||
build-depends: hpath-posix >=0.13.2
|
build-depends: hpath-posix >=0.13.2
|
||||||
@@ -108,11 +114,17 @@ common hpath-posix
|
|||||||
common http-io-streams
|
common http-io-streams
|
||||||
build-depends: http-io-streams >=0.1.2.0
|
build-depends: http-io-streams >=0.1.2.0
|
||||||
|
|
||||||
|
common hspec
|
||||||
|
build-depends: hspec >=2.7.4
|
||||||
|
|
||||||
|
common hspec-golden-aeson
|
||||||
|
build-depends: hspec-golden-aeson >=0.7
|
||||||
|
|
||||||
common io-streams
|
common io-streams
|
||||||
build-depends: io-streams >=1.5
|
build-depends: io-streams >=1.5
|
||||||
|
|
||||||
common libarchive
|
common libarchive
|
||||||
build-depends: libarchive >= 2.2.5.0
|
build-depends: libarchive >= 3.0.0.0
|
||||||
|
|
||||||
common lzma
|
common lzma
|
||||||
build-depends: lzma >=0.0.0.3
|
build-depends: lzma >=0.0.0.3
|
||||||
@@ -171,9 +183,6 @@ common strict-base
|
|||||||
common string-interpolate
|
common string-interpolate
|
||||||
build-depends: string-interpolate >=0.2.0.0
|
build-depends: string-interpolate >=0.2.0.0
|
||||||
|
|
||||||
common table-layout
|
|
||||||
build-depends: table-layout >=0.8
|
|
||||||
|
|
||||||
common template-haskell
|
common template-haskell
|
||||||
build-depends: template-haskell >=2.7
|
build-depends: template-haskell >=2.7
|
||||||
|
|
||||||
@@ -195,6 +204,12 @@ common transformers
|
|||||||
common os-release
|
common os-release
|
||||||
build-depends: os-release >=1.0.0
|
build-depends: os-release >=1.0.0
|
||||||
|
|
||||||
|
common QuickCheck
|
||||||
|
build-depends: QuickCheck >=2.14.1
|
||||||
|
|
||||||
|
common quickcheck-arbitrary-adt
|
||||||
|
build-depends: quickcheck-arbitrary-adt >=0.3.1.0
|
||||||
|
|
||||||
common unix
|
common unix
|
||||||
build-depends: unix >=2.7
|
build-depends: unix >=2.7
|
||||||
|
|
||||||
@@ -214,7 +229,7 @@ common vector
|
|||||||
build-depends: vector >=0.12
|
build-depends: vector >=0.12
|
||||||
|
|
||||||
common versions
|
common versions
|
||||||
build-depends: versions >=3.5
|
build-depends: versions >=4.0.1
|
||||||
|
|
||||||
common vty
|
common vty
|
||||||
build-depends: vty >=5.28.2
|
build-depends: vty >=5.28.2
|
||||||
@@ -240,8 +255,6 @@ common config
|
|||||||
PackageImports
|
PackageImports
|
||||||
RecordWildCards
|
RecordWildCards
|
||||||
ScopedTypeVariables
|
ScopedTypeVariables
|
||||||
Strict
|
|
||||||
StrictData
|
|
||||||
TupleSections
|
TupleSections
|
||||||
|
|
||||||
library
|
library
|
||||||
@@ -256,6 +269,7 @@ library
|
|||||||
, bytestring
|
, bytestring
|
||||||
, bz2
|
, bz2
|
||||||
, case-insensitive
|
, case-insensitive
|
||||||
|
, casing
|
||||||
, concurrent-output
|
, concurrent-output
|
||||||
, containers
|
, containers
|
||||||
, cryptohash-sha256
|
, cryptohash-sha256
|
||||||
@@ -297,6 +311,7 @@ library
|
|||||||
, utf8-string
|
, utf8-string
|
||||||
, vector
|
, vector
|
||||||
, versions
|
, versions
|
||||||
|
, vty
|
||||||
, word8
|
, word8
|
||||||
, yaml
|
, yaml
|
||||||
, zlib
|
, zlib
|
||||||
@@ -321,6 +336,10 @@ library
|
|||||||
GHCup.Utils.Version.QQ
|
GHCup.Utils.Version.QQ
|
||||||
GHCup.Version
|
GHCup.Version
|
||||||
|
|
||||||
|
default-extensions:
|
||||||
|
Strict
|
||||||
|
StrictData
|
||||||
|
|
||||||
-- other-modules:
|
-- other-modules:
|
||||||
-- other-extensions:
|
-- other-extensions:
|
||||||
hs-source-dirs: lib
|
hs-source-dirs: lib
|
||||||
@@ -361,7 +380,6 @@ executable ghcup
|
|||||||
, safe
|
, safe
|
||||||
, safe-exceptions
|
, safe-exceptions
|
||||||
, string-interpolate
|
, string-interpolate
|
||||||
, table-layout
|
|
||||||
, template-haskell
|
, template-haskell
|
||||||
, text
|
, text
|
||||||
, uri-bytestring
|
, uri-bytestring
|
||||||
@@ -377,6 +395,10 @@ executable ghcup
|
|||||||
hs-source-dirs: app/ghcup
|
hs-source-dirs: app/ghcup
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
|
|
||||||
|
default-extensions:
|
||||||
|
Strict
|
||||||
|
StrictData
|
||||||
|
|
||||||
if flag(internal-downloader)
|
if flag(internal-downloader)
|
||||||
cpp-options: -DINTERNAL_DOWNLOADER
|
cpp-options: -DINTERNAL_DOWNLOADER
|
||||||
|
|
||||||
@@ -412,7 +434,6 @@ executable ghcup-gen
|
|||||||
, resourcet
|
, resourcet
|
||||||
, safe-exceptions
|
, safe-exceptions
|
||||||
, string-interpolate
|
, string-interpolate
|
||||||
, table-layout
|
|
||||||
, text
|
, text
|
||||||
, transformers
|
, transformers
|
||||||
, uri-bytestring
|
, uri-bytestring
|
||||||
@@ -431,8 +452,25 @@ executable ghcup-gen
|
|||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
|
|
||||||
test-suite ghcup-test
|
test-suite ghcup-test
|
||||||
default-language: Haskell2010
|
import:
|
||||||
|
config
|
||||||
|
, base
|
||||||
|
, bytestring
|
||||||
|
, containers
|
||||||
|
, QuickCheck
|
||||||
|
, generic-arbitrary
|
||||||
|
, hpath
|
||||||
|
, hspec
|
||||||
|
, hspec-golden-aeson
|
||||||
|
, quickcheck-arbitrary-adt
|
||||||
|
, text
|
||||||
|
, uri-bytestring
|
||||||
|
, versions
|
||||||
type: exitcode-stdio-1.0
|
type: exitcode-stdio-1.0
|
||||||
|
build-depends: ghcup
|
||||||
hs-source-dirs: test
|
hs-source-dirs: test
|
||||||
main-is: MyLibTest.hs
|
main-is: Main.hs
|
||||||
build-depends: base >=4.12.0.0
|
other-modules:
|
||||||
|
GHCup.ArbitraryTypes
|
||||||
|
GHCup.Types.JSONSpec
|
||||||
|
Spec
|
||||||
|
|||||||
16179
golden/GHCupInfo.json
Normal file
16179
golden/GHCupInfo.json
Normal file
File diff suppressed because it is too large
Load Diff
19
hie.yaml
19
hie.yaml
@@ -1,4 +1,19 @@
|
|||||||
cradle:
|
cradle:
|
||||||
cabal:
|
cabal:
|
||||||
- path: "."
|
- path: "./lib"
|
||||||
component: "ghcup:lib:ghcup"
|
component: "lib:ghcup"
|
||||||
|
|
||||||
|
- path: "./app/ghcup/Main.hs"
|
||||||
|
component: "ghcup:exe:ghcup"
|
||||||
|
|
||||||
|
- path: "./app/ghcup/BrickMain.hs"
|
||||||
|
component: "ghcup:exe:ghcup"
|
||||||
|
|
||||||
|
- path: "./app/ghcup-gen/Main.hs"
|
||||||
|
component: "ghcup:exe:ghcup-gen"
|
||||||
|
|
||||||
|
- path: "./app/ghcup-gen/Validate.hs"
|
||||||
|
component: "ghcup:exe:ghcup-gen"
|
||||||
|
|
||||||
|
- path: "./test"
|
||||||
|
component: "ghcup:test:ghcup-test"
|
||||||
|
|||||||
776
lib/GHCup.hs
776
lib/GHCup.hs
File diff suppressed because it is too large
Load Diff
@@ -83,9 +83,9 @@ import qualified Crypto.Hash.SHA256 as SHA256
|
|||||||
import qualified Data.ByteString as BS
|
import qualified Data.ByteString as BS
|
||||||
import qualified Data.ByteString.Base16 as B16
|
import qualified Data.ByteString.Base16 as B16
|
||||||
import qualified Data.ByteString.Lazy as L
|
import qualified Data.ByteString.Lazy as L
|
||||||
|
import qualified Data.Map.Strict as M
|
||||||
#if defined(INTERNAL_DOWNLOADER)
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
import qualified Data.CaseInsensitive as CI
|
import qualified Data.CaseInsensitive as CI
|
||||||
import qualified Data.Map.Strict as M
|
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
#endif
|
#endif
|
||||||
import qualified Data.Text.Encoding as E
|
import qualified Data.Text.Encoding as E
|
||||||
@@ -104,8 +104,8 @@ import qualified System.Posix.RawFilePath.Directory
|
|||||||
------------------
|
------------------
|
||||||
|
|
||||||
|
|
||||||
-- | Like 'getDownloads', but tries to fall back to
|
|
||||||
-- cached ~/.ghcup/cache/ghcup-<format-ver>.yaml
|
-- | Downloads the download information! But only if we need to ;P
|
||||||
getDownloadsF :: ( FromJSONKey Tool
|
getDownloadsF :: ( FromJSONKey Tool
|
||||||
, FromJSONKey Version
|
, FromJSONKey Version
|
||||||
, FromJSON VersionInfo
|
, FromJSON VersionInfo
|
||||||
@@ -114,7 +114,7 @@ getDownloadsF :: ( FromJSONKey Tool
|
|||||||
, MonadLogger m
|
, MonadLogger m
|
||||||
, MonadThrow m
|
, MonadThrow m
|
||||||
, MonadFail m
|
, MonadFail m
|
||||||
, MonadReader Settings m
|
, MonadReader AppState m
|
||||||
)
|
)
|
||||||
=> URLSource
|
=> URLSource
|
||||||
-> Excepts
|
-> Excepts
|
||||||
@@ -123,17 +123,24 @@ getDownloadsF :: ( FromJSONKey Tool
|
|||||||
GHCupInfo
|
GHCupInfo
|
||||||
getDownloadsF urlSource = do
|
getDownloadsF urlSource = do
|
||||||
case urlSource of
|
case urlSource of
|
||||||
GHCupURL ->
|
GHCupURL -> liftE getBase
|
||||||
liftE
|
(OwnSource url) -> do
|
||||||
$ handleIO (\_ -> readFromCache)
|
bs <- reThrowAll DownloadFailed $ downloadBS url
|
||||||
$ catchE @_ @'[JSONError , FileDoesNotExistError]
|
lE' JSONDecodeError $ bimap show id $ Y.decodeEither' (L.toStrict bs)
|
||||||
(\(DownloadFailed _) -> readFromCache)
|
(OwnSpec av) -> pure av
|
||||||
$ getDownloads urlSource
|
(AddSource (Left ext)) -> do
|
||||||
(OwnSource _) -> liftE $ getDownloads urlSource
|
base <- liftE getBase
|
||||||
(OwnSpec _) -> liftE $ getDownloads urlSource
|
pure (mergeGhcupInfo base ext)
|
||||||
|
(AddSource (Right uri)) -> do
|
||||||
|
base <- liftE getBase
|
||||||
|
bsExt <- reThrowAll DownloadFailed $ downloadBS uri
|
||||||
|
ext <- lE' JSONDecodeError $ bimap show id $ Y.decodeEither' (L.toStrict bsExt)
|
||||||
|
pure (mergeGhcupInfo base ext)
|
||||||
where
|
where
|
||||||
|
readFromCache :: (MonadIO m, MonadCatch m, MonadLogger m, MonadReader AppState m)
|
||||||
|
=> Excepts '[JSONError, FileDoesNotExistError] m GHCupInfo
|
||||||
readFromCache = do
|
readFromCache = do
|
||||||
Settings {dirs = Dirs {..}} <- lift ask
|
AppState {dirs = Dirs {..}} <- lift ask
|
||||||
lift $ $(logWarn)
|
lift $ $(logWarn)
|
||||||
[i|Could not get download info, trying cached version (this may not be recent!)|]
|
[i|Could not get download info, trying cached version (this may not be recent!)|]
|
||||||
let path = view pathL' ghcupURL
|
let path = view pathL' ghcupURL
|
||||||
@@ -145,32 +152,25 @@ getDownloadsF urlSource = do
|
|||||||
$ readFile yaml_file
|
$ readFile yaml_file
|
||||||
lE' JSONDecodeError $ bimap show id $ Y.decodeEither' (L.toStrict bs)
|
lE' JSONDecodeError $ bimap show id $ Y.decodeEither' (L.toStrict bs)
|
||||||
|
|
||||||
|
getBase :: (MonadFail m, MonadIO m, MonadCatch m, MonadLogger m, MonadReader AppState m)
|
||||||
|
=> Excepts '[JSONError , FileDoesNotExistError] m GHCupInfo
|
||||||
|
getBase =
|
||||||
|
handleIO (\_ -> readFromCache)
|
||||||
|
$ catchE @_ @'[JSONError, FileDoesNotExistError]
|
||||||
|
(\(DownloadFailed _) -> readFromCache)
|
||||||
|
$ ((reThrowAll @_ @_ @'[JSONError, DownloadFailed] DownloadFailed $ smartDl ghcupURL)
|
||||||
|
>>= (liftE . lE' @_ @_ @'[JSONError] JSONDecodeError . bimap show id . Y.decodeEither' . L.toStrict))
|
||||||
|
|
||||||
-- | Downloads the download information! But only if we need to ;P
|
mergeGhcupInfo :: GHCupInfo -- ^ base to merge with
|
||||||
getDownloads :: ( FromJSONKey Tool
|
-> GHCupInfo -- ^ extension overwriting the base
|
||||||
, FromJSONKey Version
|
-> GHCupInfo
|
||||||
, FromJSON VersionInfo
|
mergeGhcupInfo (GHCupInfo tr base) (GHCupInfo _ ext) =
|
||||||
, MonadIO m
|
let new = M.mapWithKey (\k a -> case M.lookup k ext of
|
||||||
, MonadCatch m
|
Just a' -> M.union a' a
|
||||||
, MonadLogger m
|
Nothing -> a
|
||||||
, MonadThrow m
|
) base
|
||||||
, MonadFail m
|
in GHCupInfo tr new
|
||||||
, MonadReader Settings m
|
|
||||||
)
|
|
||||||
=> URLSource
|
|
||||||
-> Excepts '[JSONError , DownloadFailed] m GHCupInfo
|
|
||||||
getDownloads urlSource = do
|
|
||||||
lift $ $(logDebug) [i|Receiving download info from: #{urlSource}|]
|
|
||||||
case urlSource of
|
|
||||||
GHCupURL -> do
|
|
||||||
bs <- reThrowAll DownloadFailed $ smartDl ghcupURL
|
|
||||||
lE' JSONDecodeError $ bimap show id $ Y.decodeEither' (L.toStrict bs)
|
|
||||||
(OwnSource url) -> do
|
|
||||||
bs <- reThrowAll DownloadFailed $ downloadBS url
|
|
||||||
lE' JSONDecodeError $ bimap show id $ Y.decodeEither' (L.toStrict bs)
|
|
||||||
(OwnSpec av) -> pure $ av
|
|
||||||
|
|
||||||
where
|
|
||||||
-- First check if the json file is in the ~/.ghcup/cache dir
|
-- First check if the json file is in the ~/.ghcup/cache dir
|
||||||
-- and check it's access time. If it has been accessed within the
|
-- and check it's access time. If it has been accessed within the
|
||||||
-- last 5 minutes, just reuse it.
|
-- last 5 minutes, just reuse it.
|
||||||
@@ -185,7 +185,7 @@ getDownloads urlSource = do
|
|||||||
, MonadIO m1
|
, MonadIO m1
|
||||||
, MonadFail m1
|
, MonadFail m1
|
||||||
, MonadLogger m1
|
, MonadLogger m1
|
||||||
, MonadReader Settings m1
|
, MonadReader AppState m1
|
||||||
)
|
)
|
||||||
=> URI
|
=> URI
|
||||||
-> Excepts
|
-> Excepts
|
||||||
@@ -200,7 +200,7 @@ getDownloads urlSource = do
|
|||||||
m1
|
m1
|
||||||
L.ByteString
|
L.ByteString
|
||||||
smartDl uri' = do
|
smartDl uri' = do
|
||||||
Settings {dirs = Dirs {..}} <- lift ask
|
AppState {dirs = Dirs {..}} <- lift ask
|
||||||
let path = view pathL' uri'
|
let path = view pathL' uri'
|
||||||
json_file <- (cacheDir </>) <$> urlBaseName path
|
json_file <- (cacheDir </>) <$> urlBaseName path
|
||||||
e <- liftIO $ doesFileExist json_file
|
e <- liftIO $ doesFileExist json_file
|
||||||
@@ -226,7 +226,7 @@ getDownloads urlSource = do
|
|||||||
else -- access in less than 5 minutes, re-use file
|
else -- access in less than 5 minutes, re-use file
|
||||||
liftIO $ readFile json_file
|
liftIO $ readFile json_file
|
||||||
else do
|
else do
|
||||||
liftIO $ createDirRecursive newDirPerms cacheDir
|
liftIO $ createDirRecursive' cacheDir
|
||||||
getModTime >>= \case
|
getModTime >>= \case
|
||||||
Just modTime -> dlWithMod modTime json_file
|
Just modTime -> dlWithMod modTime json_file
|
||||||
Nothing -> do
|
Nothing -> do
|
||||||
@@ -311,7 +311,7 @@ getDownloadInfo t v (PlatformRequest a p mv) dls = maybe
|
|||||||
--
|
--
|
||||||
-- The file must not exist.
|
-- The file must not exist.
|
||||||
download :: ( MonadMask m
|
download :: ( MonadMask m
|
||||||
, MonadReader Settings m
|
, MonadReader AppState m
|
||||||
, MonadThrow m
|
, MonadThrow m
|
||||||
, MonadLogger m
|
, MonadLogger m
|
||||||
, MonadIO m
|
, MonadIO m
|
||||||
@@ -330,7 +330,7 @@ download dli dest mfn
|
|||||||
scheme = view (dlUri % uriSchemeL' % schemeBSL') dli
|
scheme = view (dlUri % uriSchemeL' % schemeBSL') dli
|
||||||
cp = do
|
cp = do
|
||||||
-- destination dir must exist
|
-- destination dir must exist
|
||||||
liftIO $ hideError AlreadyExists $ createDirRecursive newDirPerms dest
|
liftIO $ createDirRecursive' dest
|
||||||
destFile <- getDestFile
|
destFile <- getDestFile
|
||||||
fromFile <- parseAbs path
|
fromFile <- parseAbs path
|
||||||
liftIO $ copyFile fromFile destFile Strict
|
liftIO $ copyFile fromFile destFile Strict
|
||||||
@@ -340,7 +340,7 @@ download dli dest mfn
|
|||||||
lift $ $(logInfo) [i|downloading: #{uri'}|]
|
lift $ $(logInfo) [i|downloading: #{uri'}|]
|
||||||
|
|
||||||
-- destination dir must exist
|
-- destination dir must exist
|
||||||
liftIO $ hideError AlreadyExists $ createDirRecursive newDirPerms dest
|
liftIO $ createDirRecursive' dest
|
||||||
destFile <- getDestFile
|
destFile <- getDestFile
|
||||||
|
|
||||||
-- download
|
-- download
|
||||||
@@ -383,7 +383,7 @@ downloadCached :: ( MonadMask m
|
|||||||
, MonadThrow m
|
, MonadThrow m
|
||||||
, MonadLogger m
|
, MonadLogger m
|
||||||
, MonadIO m
|
, MonadIO m
|
||||||
, MonadReader Settings m
|
, MonadReader AppState m
|
||||||
)
|
)
|
||||||
=> DownloadInfo
|
=> DownloadInfo
|
||||||
-> Maybe (Path Rel) -- ^ optional filename
|
-> Maybe (Path Rel) -- ^ optional filename
|
||||||
@@ -392,7 +392,7 @@ downloadCached dli mfn = do
|
|||||||
cache <- lift getCache
|
cache <- lift getCache
|
||||||
case cache of
|
case cache of
|
||||||
True -> do
|
True -> do
|
||||||
Settings {dirs = Dirs {..}} <- lift ask
|
AppState {dirs = Dirs {..}} <- lift ask
|
||||||
fn <- maybe (urlBaseName $ view (dlUri % pathL') dli) pure mfn
|
fn <- maybe (urlBaseName $ view (dlUri % pathL') dli) pure mfn
|
||||||
let cachfile = cacheDir </> fn
|
let cachfile = cacheDir </> fn
|
||||||
fileExists <- liftIO $ doesFileExist cachfile
|
fileExists <- liftIO $ doesFileExist cachfile
|
||||||
@@ -416,7 +416,7 @@ downloadCached dli mfn = do
|
|||||||
|
|
||||||
|
|
||||||
-- | This is used for downloading the JSON.
|
-- | This is used for downloading the JSON.
|
||||||
downloadBS :: (MonadReader Settings m, MonadCatch m, MonadIO m, MonadLogger m)
|
downloadBS :: (MonadReader AppState m, MonadCatch m, MonadIO m, MonadLogger m)
|
||||||
=> URI
|
=> URI
|
||||||
-> Excepts
|
-> Excepts
|
||||||
'[ FileDoesNotExistError
|
'[ FileDoesNotExistError
|
||||||
@@ -473,12 +473,12 @@ downloadBS uri'
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
checkDigest :: (MonadIO m, MonadThrow m, MonadLogger m, MonadReader Settings m)
|
checkDigest :: (MonadIO m, MonadThrow m, MonadLogger m, MonadReader AppState m)
|
||||||
=> DownloadInfo
|
=> DownloadInfo
|
||||||
-> Path Abs
|
-> Path Abs
|
||||||
-> Excepts '[DigestError] m ()
|
-> Excepts '[DigestError] m ()
|
||||||
checkDigest dli file = do
|
checkDigest dli file = do
|
||||||
verify <- lift ask <&> (not . noVerify)
|
verify <- lift ask <&> (not . noVerify . settings)
|
||||||
when verify $ do
|
when verify $ do
|
||||||
p' <- toFilePath <$> basename file
|
p' <- toFilePath <$> basename file
|
||||||
lift $ $(logInfo) [i|verifying digest of: #{p'}|]
|
lift $ $(logInfo) [i|verifying digest of: #{p'}|]
|
||||||
|
|||||||
@@ -152,3 +152,10 @@ data ParseError = ParseError String
|
|||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
instance Exception ParseError
|
instance Exception ParseError
|
||||||
|
|
||||||
|
|
||||||
|
data UnexpectedListLength = UnexpectedListLength String
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Exception UnexpectedListLength
|
||||||
|
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ prettyRequirements :: Requirements -> T.Text
|
|||||||
prettyRequirements Requirements {..} =
|
prettyRequirements Requirements {..} =
|
||||||
let d = if not . null $ _distroPKGs
|
let d = if not . null $ _distroPKGs
|
||||||
then
|
then
|
||||||
"\n Install the following distro packages: "
|
"\n Please install the following distro packages: "
|
||||||
<> T.intercalate " " _distroPKGs
|
<> T.intercalate " " _distroPKGs
|
||||||
else ""
|
else ""
|
||||||
n = if not . T.null $ _notes then "\n Note: " <> _notes else ""
|
n = if not . T.null $ _notes then "\n Note: " <> _notes else ""
|
||||||
|
|||||||
@@ -19,7 +19,9 @@ import Data.Versions
|
|||||||
import HPath
|
import HPath
|
||||||
import URI.ByteString
|
import URI.ByteString
|
||||||
|
|
||||||
|
import qualified Data.Text as T
|
||||||
import qualified GHC.Generics as GHC
|
import qualified GHC.Generics as GHC
|
||||||
|
import qualified Graphics.Vty as Vty
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -75,6 +77,7 @@ type PlatformVersionSpec = Map (Maybe Versioning) DownloadInfo
|
|||||||
data Tool = GHC
|
data Tool = GHC
|
||||||
| Cabal
|
| Cabal
|
||||||
| GHCup
|
| GHCup
|
||||||
|
| HLS
|
||||||
deriving (Eq, GHC.Generic, Ord, Show)
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
|
||||||
@@ -86,7 +89,7 @@ data VersionInfo = VersionInfo
|
|||||||
, _viSourceDL :: Maybe DownloadInfo -- ^ source tarball
|
, _viSourceDL :: Maybe DownloadInfo -- ^ source tarball
|
||||||
, _viArch :: ArchitectureSpec -- ^ descend for binary downloads per arch
|
, _viArch :: ArchitectureSpec -- ^ descend for binary downloads per arch
|
||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, GHC.Generic, Show)
|
||||||
|
|
||||||
|
|
||||||
-- | A tag. These are currently attached to a version of a tool.
|
-- | A tag. These are currently attached to a version of a tool.
|
||||||
@@ -94,8 +97,9 @@ data Tag = Latest
|
|||||||
| Recommended
|
| Recommended
|
||||||
| Prerelease
|
| Prerelease
|
||||||
| Base PVP
|
| Base PVP
|
||||||
|
| Old -- ^ old version are hidden by default in TUI
|
||||||
| UnknownTag String -- ^ used for upwardscompat
|
| UnknownTag String -- ^ used for upwardscompat
|
||||||
deriving (Ord, Eq, Show) -- FIXME: manual JSON instance
|
deriving (Ord, Eq, GHC.Generic, Show) -- FIXME: manual JSON instance
|
||||||
|
|
||||||
|
|
||||||
data Architecture = A_64
|
data Architecture = A_64
|
||||||
@@ -108,6 +112,15 @@ data Architecture = A_64
|
|||||||
| A_ARM64
|
| A_ARM64
|
||||||
deriving (Eq, GHC.Generic, Ord, Show)
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
prettyArch :: Architecture -> String
|
||||||
|
prettyArch A_64 = "x86_64"
|
||||||
|
prettyArch A_32 = "i386"
|
||||||
|
prettyArch A_PowerPC = "powerpc"
|
||||||
|
prettyArch A_PowerPC64 = "powerpc64"
|
||||||
|
prettyArch A_Sparc = "sparc"
|
||||||
|
prettyArch A_Sparc64 = "sparc64"
|
||||||
|
prettyArch A_ARM = "arm"
|
||||||
|
prettyArch A_ARM64 = "aarch64"
|
||||||
|
|
||||||
data Platform = Linux LinuxDistro
|
data Platform = Linux LinuxDistro
|
||||||
-- ^ must exit
|
-- ^ must exit
|
||||||
@@ -116,6 +129,11 @@ data Platform = Linux LinuxDistro
|
|||||||
| FreeBSD
|
| FreeBSD
|
||||||
deriving (Eq, GHC.Generic, Ord, Show)
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
prettyPlatfrom :: Platform -> String
|
||||||
|
prettyPlatfrom (Linux distro) = "linux-" ++ prettyDistro distro
|
||||||
|
prettyPlatfrom Darwin = "darwin"
|
||||||
|
prettyPlatfrom FreeBSD = "freebsd"
|
||||||
|
|
||||||
data LinuxDistro = Debian
|
data LinuxDistro = Debian
|
||||||
| Ubuntu
|
| Ubuntu
|
||||||
| Mint
|
| Mint
|
||||||
@@ -132,6 +150,19 @@ data LinuxDistro = Debian
|
|||||||
-- ^ must exit
|
-- ^ must exit
|
||||||
deriving (Eq, GHC.Generic, Ord, Show)
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
prettyDistro :: LinuxDistro -> String
|
||||||
|
prettyDistro Debian = "debian"
|
||||||
|
prettyDistro Ubuntu = "ubuntu"
|
||||||
|
prettyDistro Mint= "mint"
|
||||||
|
prettyDistro Fedora = "fedora"
|
||||||
|
prettyDistro CentOS = "centos"
|
||||||
|
prettyDistro RedHat = "redhat"
|
||||||
|
prettyDistro Alpine = "alpine"
|
||||||
|
prettyDistro AmazonLinux = "amazon"
|
||||||
|
prettyDistro Gentoo = "gentoo"
|
||||||
|
prettyDistro Exherbo = "exherbo"
|
||||||
|
prettyDistro UnknownLinux = "unknown"
|
||||||
|
|
||||||
|
|
||||||
-- | An encapsulation of a download. This can be used
|
-- | An encapsulation of a download. This can be used
|
||||||
-- to download, extract and install a tool.
|
-- to download, extract and install a tool.
|
||||||
@@ -140,7 +171,7 @@ data DownloadInfo = DownloadInfo
|
|||||||
, _dlSubdir :: Maybe TarDir
|
, _dlSubdir :: Maybe TarDir
|
||||||
, _dlHash :: Text
|
, _dlHash :: Text
|
||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, GHC.Generic, Show)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -153,34 +184,89 @@ data DownloadInfo = DownloadInfo
|
|||||||
-- | How to descend into a tar archive.
|
-- | How to descend into a tar archive.
|
||||||
data TarDir = RealDir (Path Rel)
|
data TarDir = RealDir (Path Rel)
|
||||||
| RegexDir String -- ^ will be compiled to regex, the first match will "win"
|
| RegexDir String -- ^ will be compiled to regex, the first match will "win"
|
||||||
deriving (Eq, Show)
|
deriving (Eq, GHC.Generic, Show)
|
||||||
|
|
||||||
|
|
||||||
-- | Where to fetch GHCupDownloads from.
|
-- | Where to fetch GHCupDownloads from.
|
||||||
data URLSource = GHCupURL
|
data URLSource = GHCupURL
|
||||||
| OwnSource URI
|
| OwnSource URI
|
||||||
| OwnSpec GHCupInfo
|
| OwnSpec GHCupInfo
|
||||||
deriving Show
|
| AddSource (Either GHCupInfo URI) -- ^ merge with GHCupURL
|
||||||
|
deriving (GHC.Generic, Show)
|
||||||
|
|
||||||
|
|
||||||
|
data UserSettings = UserSettings
|
||||||
|
{ uCache :: Maybe Bool
|
||||||
|
, uNoVerify :: Maybe Bool
|
||||||
|
, uVerbose :: Maybe Bool
|
||||||
|
, uKeepDirs :: Maybe KeepDirs
|
||||||
|
, uDownloader :: Maybe Downloader
|
||||||
|
, uKeyBindings :: Maybe UserKeyBindings
|
||||||
|
, uUrlSource :: Maybe URLSource
|
||||||
|
}
|
||||||
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
|
defaultUserSettings :: UserSettings
|
||||||
|
defaultUserSettings = UserSettings Nothing Nothing Nothing Nothing Nothing Nothing Nothing
|
||||||
|
|
||||||
|
data UserKeyBindings = UserKeyBindings
|
||||||
|
{ kUp :: Maybe Vty.Key
|
||||||
|
, kDown :: Maybe Vty.Key
|
||||||
|
, kQuit :: Maybe Vty.Key
|
||||||
|
, kInstall :: Maybe Vty.Key
|
||||||
|
, kUninstall :: Maybe Vty.Key
|
||||||
|
, kSet :: Maybe Vty.Key
|
||||||
|
, kChangelog :: Maybe Vty.Key
|
||||||
|
, kShowAll :: Maybe Vty.Key
|
||||||
|
}
|
||||||
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
|
data KeyBindings = KeyBindings
|
||||||
|
{ bUp :: Vty.Key
|
||||||
|
, bDown :: Vty.Key
|
||||||
|
, bQuit :: Vty.Key
|
||||||
|
, bInstall :: Vty.Key
|
||||||
|
, bUninstall :: Vty.Key
|
||||||
|
, bSet :: Vty.Key
|
||||||
|
, bChangelog :: Vty.Key
|
||||||
|
, bShowAll :: Vty.Key
|
||||||
|
}
|
||||||
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
|
defaultKeyBindings :: KeyBindings
|
||||||
|
defaultKeyBindings = KeyBindings
|
||||||
|
{ bUp = Vty.KUp
|
||||||
|
, bDown = Vty.KDown
|
||||||
|
, bQuit = Vty.KChar 'q'
|
||||||
|
, bInstall = Vty.KChar 'i'
|
||||||
|
, bUninstall = Vty.KChar 'u'
|
||||||
|
, bSet = Vty.KChar 's'
|
||||||
|
, bChangelog = Vty.KChar 'c'
|
||||||
|
, bShowAll = Vty.KChar 'a'
|
||||||
|
}
|
||||||
|
|
||||||
|
data AppState = AppState
|
||||||
|
{ settings :: Settings
|
||||||
|
, dirs :: Dirs
|
||||||
|
, keyBindings :: KeyBindings
|
||||||
|
} deriving (Show)
|
||||||
|
|
||||||
data Settings = Settings
|
data Settings = Settings
|
||||||
{ -- set by user
|
{ cache :: Bool
|
||||||
cache :: Bool
|
|
||||||
, noVerify :: Bool
|
, noVerify :: Bool
|
||||||
, keepDirs :: KeepDirs
|
, keepDirs :: KeepDirs
|
||||||
, downloader :: Downloader
|
, downloader :: Downloader
|
||||||
, verbose :: Bool
|
, verbose :: Bool
|
||||||
|
, urlSource :: URLSource
|
||||||
-- set on app start
|
|
||||||
, dirs :: Dirs
|
|
||||||
}
|
}
|
||||||
deriving Show
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
data Dirs = Dirs
|
data Dirs = Dirs
|
||||||
{ baseDir :: Path Abs
|
{ baseDir :: Path Abs
|
||||||
, binDir :: Path Abs
|
, binDir :: Path Abs
|
||||||
, cacheDir :: Path Abs
|
, cacheDir :: Path Abs
|
||||||
, logsDir :: Path Abs
|
, logsDir :: Path Abs
|
||||||
|
, confDir :: Path Abs
|
||||||
}
|
}
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
@@ -219,6 +305,12 @@ data PlatformResult = PlatformResult
|
|||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, Show)
|
||||||
|
|
||||||
|
prettyPlatform :: PlatformResult -> String
|
||||||
|
prettyPlatform PlatformResult { _platform = plat, _distroVersion = Just v' }
|
||||||
|
= show plat <> ", " <> show v'
|
||||||
|
prettyPlatform PlatformResult { _platform = plat, _distroVersion = Nothing }
|
||||||
|
= show plat
|
||||||
|
|
||||||
data PlatformRequest = PlatformRequest
|
data PlatformRequest = PlatformRequest
|
||||||
{ _rArch :: Architecture
|
{ _rArch :: Architecture
|
||||||
, _rPlatform :: Platform
|
, _rPlatform :: Platform
|
||||||
@@ -226,6 +318,13 @@ data PlatformRequest = PlatformRequest
|
|||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, Show)
|
||||||
|
|
||||||
|
prettyPfReq :: PlatformRequest -> String
|
||||||
|
prettyPfReq (PlatformRequest arch plat ver) =
|
||||||
|
prettyArch arch ++ "-" ++ prettyPlatfrom plat ++ pver
|
||||||
|
where
|
||||||
|
pver = case ver of
|
||||||
|
Just v' -> "-" ++ (T.unpack $ prettyV v')
|
||||||
|
Nothing -> ""
|
||||||
|
|
||||||
-- | A GHC identified by the target platform triple
|
-- | A GHC identified by the target platform triple
|
||||||
-- and the version.
|
-- and the version.
|
||||||
|
|||||||
@@ -33,14 +33,17 @@ import Data.Versions
|
|||||||
import Data.Word8
|
import Data.Word8
|
||||||
import HPath
|
import HPath
|
||||||
import URI.ByteString
|
import URI.ByteString
|
||||||
|
import Text.Casing
|
||||||
|
|
||||||
import qualified Data.ByteString as BS
|
import qualified Data.ByteString as BS
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
|
import qualified Graphics.Vty as Vty
|
||||||
|
|
||||||
|
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } { fieldLabelModifier = removeLensFieldLabel } ''Architecture
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } { fieldLabelModifier = removeLensFieldLabel } ''Architecture
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''LinuxDistro
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''LinuxDistro
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Mess
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Mess
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''MChunk
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Platform
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Platform
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''SemVer
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''SemVer
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Tool
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Tool
|
||||||
@@ -50,11 +53,18 @@ deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Versio
|
|||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''DownloadInfo
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''DownloadInfo
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''GHCupInfo
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''GHCupInfo
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Requirements
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Requirements
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''KeepDirs
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Downloader
|
||||||
|
deriveJSON defaultOptions { sumEncoding = ObjectWithSingleField } ''URLSource
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = \str' -> maybe str' T.unpack . T.stripPrefix (T.pack "u-") . T.pack . kebab $ str' } ''UserSettings
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = \str' -> maybe str' T.unpack . T.stripPrefix (T.pack "k-") . T.pack . kebab $ str' } ''UserKeyBindings
|
||||||
|
deriveJSON defaultOptions { sumEncoding = ObjectWithSingleField } ''Vty.Key
|
||||||
|
|
||||||
instance ToJSON Tag where
|
instance ToJSON Tag where
|
||||||
toJSON Latest = String "Latest"
|
toJSON Latest = String "Latest"
|
||||||
toJSON Recommended = String "Recommended"
|
toJSON Recommended = String "Recommended"
|
||||||
toJSON Prerelease = String "Prerelease"
|
toJSON Prerelease = String "Prerelease"
|
||||||
|
toJSON Old = String "old"
|
||||||
toJSON (Base pvp'') = String ("base-" <> prettyPVP pvp'')
|
toJSON (Base pvp'') = String ("base-" <> prettyPVP pvp'')
|
||||||
toJSON (UnknownTag x ) = String (T.pack x)
|
toJSON (UnknownTag x ) = String (T.pack x)
|
||||||
|
|
||||||
@@ -63,6 +73,7 @@ instance FromJSON Tag where
|
|||||||
"Latest" -> pure Latest
|
"Latest" -> pure Latest
|
||||||
"Recommended" -> pure Recommended
|
"Recommended" -> pure Recommended
|
||||||
"Prerelease" -> pure Prerelease
|
"Prerelease" -> pure Prerelease
|
||||||
|
"old" -> pure Old
|
||||||
('b' : 'a' : 's' : 'e' : '-' : ver') -> case pvp (T.pack ver') of
|
('b' : 'a' : 's' : 'e' : '-' : ver') -> case pvp (T.pack ver') of
|
||||||
Right x -> pure $ Base x
|
Right x -> pure $ Base x
|
||||||
Left e -> fail . show $ e
|
Left e -> fail . show $ e
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ import GHCup.Utils.Prelude
|
|||||||
import GHCup.Utils.String.QQ
|
import GHCup.Utils.String.QQ
|
||||||
|
|
||||||
#if !defined(TAR)
|
#if !defined(TAR)
|
||||||
import Codec.Archive
|
import Codec.Archive hiding ( Directory )
|
||||||
#endif
|
#endif
|
||||||
import Control.Applicative
|
import Control.Applicative
|
||||||
import Control.Exception.Safe
|
import Control.Exception.Safe
|
||||||
@@ -50,6 +50,7 @@ import Data.ByteString ( ByteString )
|
|||||||
import Data.Either
|
import Data.Either
|
||||||
import Data.Foldable
|
import Data.Foldable
|
||||||
import Data.List
|
import Data.List
|
||||||
|
import Data.List.NonEmpty ( NonEmpty( (:|) ))
|
||||||
import Data.List.Split
|
import Data.List.Split
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.String.Interpolate
|
import Data.String.Interpolate
|
||||||
@@ -99,21 +100,21 @@ import qualified Text.Megaparsec as MP
|
|||||||
|
|
||||||
|
|
||||||
-- | The symlink destination of a ghc tool.
|
-- | The symlink destination of a ghc tool.
|
||||||
ghcLinkDestination :: (MonadReader Settings m, MonadThrow m, MonadIO m)
|
ghcLinkDestination :: (MonadReader AppState m, MonadThrow m, MonadIO m)
|
||||||
=> ByteString -- ^ the tool, such as 'ghc', 'haddock' etc.
|
=> ByteString -- ^ the tool, such as 'ghc', 'haddock' etc.
|
||||||
-> GHCTargetVersion
|
-> GHCTargetVersion
|
||||||
-> m ByteString
|
-> m ByteString
|
||||||
ghcLinkDestination tool ver = do
|
ghcLinkDestination tool ver = do
|
||||||
Settings {dirs = Dirs {..}} <- ask
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
t <- parseRel tool
|
t <- parseRel tool
|
||||||
ghcd <- ghcupGHCDir ver
|
ghcd <- ghcupGHCDir ver
|
||||||
pure (relativeSymlink binDir (ghcd </> [rel|bin|] </> t))
|
pure (relativeSymlink binDir (ghcd </> [rel|bin|] </> t))
|
||||||
|
|
||||||
|
|
||||||
-- | Removes the minor GHC symlinks, e.g. ghc-8.6.5.
|
-- | Removes the minor GHC symlinks, e.g. ghc-8.6.5.
|
||||||
rmMinorSymlinks :: (MonadReader Settings m, MonadIO m, MonadLogger m) => GHCTargetVersion -> m ()
|
rmMinorSymlinks :: (MonadReader AppState m, MonadIO m, MonadLogger m) => GHCTargetVersion -> m ()
|
||||||
rmMinorSymlinks GHCTargetVersion {..} = do
|
rmMinorSymlinks GHCTargetVersion {..} = do
|
||||||
Settings {dirs = Dirs {..}} <- ask
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
|
|
||||||
files <- liftIO $ findFiles'
|
files <- liftIO $ findFiles'
|
||||||
binDir
|
binDir
|
||||||
@@ -130,11 +131,11 @@ rmMinorSymlinks GHCTargetVersion {..} = do
|
|||||||
|
|
||||||
|
|
||||||
-- | Removes the set ghc version for the given target, if any.
|
-- | Removes the set ghc version for the given target, if any.
|
||||||
rmPlain :: (MonadReader Settings m, MonadLogger m, MonadThrow m, MonadFail m, MonadIO m)
|
rmPlain :: (MonadReader AppState m, MonadLogger m, MonadThrow m, MonadFail m, MonadIO m)
|
||||||
=> Maybe Text -- ^ target
|
=> Maybe Text -- ^ target
|
||||||
-> Excepts '[NotInstalled] m ()
|
-> Excepts '[NotInstalled] m ()
|
||||||
rmPlain target = do
|
rmPlain target = do
|
||||||
Settings {dirs = Dirs {..}} <- lift ask
|
AppState { dirs = Dirs {..} } <- lift ask
|
||||||
mtv <- lift $ ghcSet target
|
mtv <- lift $ ghcSet target
|
||||||
forM_ mtv $ \tv -> do
|
forM_ mtv $ \tv -> do
|
||||||
files <- liftE $ ghcToolFiles tv
|
files <- liftE $ ghcToolFiles tv
|
||||||
@@ -149,11 +150,11 @@ rmPlain target = do
|
|||||||
|
|
||||||
|
|
||||||
-- | Remove the major GHC symlink, e.g. ghc-8.6.
|
-- | Remove the major GHC symlink, e.g. ghc-8.6.
|
||||||
rmMajorSymlinks :: (MonadReader Settings m, MonadThrow m, MonadLogger m, MonadIO m)
|
rmMajorSymlinks :: (MonadReader AppState m, MonadThrow m, MonadLogger m, MonadIO m)
|
||||||
=> GHCTargetVersion
|
=> GHCTargetVersion
|
||||||
-> m ()
|
-> m ()
|
||||||
rmMajorSymlinks GHCTargetVersion {..} = do
|
rmMajorSymlinks GHCTargetVersion {..} = do
|
||||||
Settings {dirs = Dirs {..}} <- ask
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
(mj, mi) <- getMajorMinorV _tvVersion
|
(mj, mi) <- getMajorMinorV _tvVersion
|
||||||
let v' = intToText mj <> "." <> intToText mi
|
let v' = intToText mj <> "." <> intToText mi
|
||||||
|
|
||||||
@@ -179,26 +180,26 @@ rmMajorSymlinks GHCTargetVersion {..} = do
|
|||||||
|
|
||||||
|
|
||||||
-- | Whethe the given GHC versin is installed.
|
-- | Whethe the given GHC versin is installed.
|
||||||
ghcInstalled :: (MonadIO m, MonadReader Settings m, MonadThrow m) => GHCTargetVersion -> m Bool
|
ghcInstalled :: (MonadIO m, MonadReader AppState m, MonadThrow m) => GHCTargetVersion -> m Bool
|
||||||
ghcInstalled ver = do
|
ghcInstalled ver = do
|
||||||
ghcdir <- ghcupGHCDir ver
|
ghcdir <- ghcupGHCDir ver
|
||||||
liftIO $ doesDirectoryExist ghcdir
|
liftIO $ doesDirectoryExist ghcdir
|
||||||
|
|
||||||
|
|
||||||
-- | Whether the given GHC version is installed from source.
|
-- | Whether the given GHC version is installed from source.
|
||||||
ghcSrcInstalled :: (MonadIO m, MonadReader Settings m, MonadThrow m) => GHCTargetVersion -> m Bool
|
ghcSrcInstalled :: (MonadIO m, MonadReader AppState m, MonadThrow m) => GHCTargetVersion -> m Bool
|
||||||
ghcSrcInstalled ver = do
|
ghcSrcInstalled ver = do
|
||||||
ghcdir <- ghcupGHCDir ver
|
ghcdir <- ghcupGHCDir ver
|
||||||
liftIO $ doesFileExist (ghcdir </> ghcUpSrcBuiltFile)
|
liftIO $ doesFileExist (ghcdir </> ghcUpSrcBuiltFile)
|
||||||
|
|
||||||
|
|
||||||
-- | Whether the given GHC version is set as the current.
|
-- | Whether the given GHC version is set as the current.
|
||||||
ghcSet :: (MonadReader Settings m, MonadThrow m, MonadIO m)
|
ghcSet :: (MonadReader AppState m, MonadThrow m, MonadIO m)
|
||||||
=> Maybe Text -- ^ the target of the GHC version, if any
|
=> Maybe Text -- ^ the target of the GHC version, if any
|
||||||
-- (e.g. armv7-unknown-linux-gnueabihf)
|
-- (e.g. armv7-unknown-linux-gnueabihf)
|
||||||
-> m (Maybe GHCTargetVersion)
|
-> m (Maybe GHCTargetVersion)
|
||||||
ghcSet mtarget = do
|
ghcSet mtarget = do
|
||||||
Settings {dirs = Dirs {..}} <- ask
|
AppState {dirs = Dirs {..}} <- ask
|
||||||
ghc <- parseRel $ E.encodeUtf8 (maybe "ghc" (<> "-ghc") mtarget)
|
ghc <- parseRel $ E.encodeUtf8 (maybe "ghc" (<> "-ghc") mtarget)
|
||||||
let ghcBin = binDir </> ghc
|
let ghcBin = binDir </> ghc
|
||||||
|
|
||||||
@@ -231,7 +232,7 @@ ghcLinkVersion bs = do
|
|||||||
|
|
||||||
-- | Get all installed GHCs by reading ~/.ghcup/ghc/<dir>.
|
-- | Get all installed GHCs by reading ~/.ghcup/ghc/<dir>.
|
||||||
-- If a dir cannot be parsed, returns left.
|
-- If a dir cannot be parsed, returns left.
|
||||||
getInstalledGHCs :: (MonadReader Settings m, MonadIO m) => m [Either (Path Rel) GHCTargetVersion]
|
getInstalledGHCs :: (MonadReader AppState m, MonadIO m) => m [Either (Path Rel) GHCTargetVersion]
|
||||||
getInstalledGHCs = do
|
getInstalledGHCs = do
|
||||||
ghcdir <- ghcupGHCBaseDir
|
ghcdir <- ghcupGHCBaseDir
|
||||||
fs <- liftIO $ hideErrorDef [NoSuchThing] [] $ getDirsFiles' ghcdir
|
fs <- liftIO $ hideErrorDef [NoSuchThing] [] $ getDirsFiles' ghcdir
|
||||||
@@ -241,10 +242,10 @@ getInstalledGHCs = do
|
|||||||
|
|
||||||
|
|
||||||
-- | Get all installed cabals, by matching on @~\/.ghcup\/bin/cabal-*@.
|
-- | Get all installed cabals, by matching on @~\/.ghcup\/bin/cabal-*@.
|
||||||
getInstalledCabals :: (MonadReader Settings m, MonadIO m, MonadCatch m)
|
getInstalledCabals :: (MonadReader AppState m, MonadIO m, MonadCatch m)
|
||||||
=> m [Either (Path Rel) Version]
|
=> m [Either (Path Rel) Version]
|
||||||
getInstalledCabals = do
|
getInstalledCabals = do
|
||||||
Settings {dirs = Dirs {..}} <- ask
|
AppState {dirs = Dirs {..}} <- ask
|
||||||
bins <- liftIO $ handleIO (\_ -> pure []) $ findFiles
|
bins <- liftIO $ handleIO (\_ -> pure []) $ findFiles
|
||||||
binDir
|
binDir
|
||||||
(makeRegexOpts compExtended execBlank ([s|^cabal-.*$|] :: ByteString))
|
(makeRegexOpts compExtended execBlank ([s|^cabal-.*$|] :: ByteString))
|
||||||
@@ -257,16 +258,16 @@ getInstalledCabals = do
|
|||||||
|
|
||||||
|
|
||||||
-- | Whether the given cabal version is installed.
|
-- | Whether the given cabal version is installed.
|
||||||
cabalInstalled :: (MonadIO m, MonadReader Settings m, MonadCatch m) => Version -> m Bool
|
cabalInstalled :: (MonadIO m, MonadReader AppState m, MonadCatch m) => Version -> m Bool
|
||||||
cabalInstalled ver = do
|
cabalInstalled ver = do
|
||||||
vers <- fmap rights $ getInstalledCabals
|
vers <- fmap rights $ getInstalledCabals
|
||||||
pure $ elem ver $ vers
|
pure $ elem ver $ vers
|
||||||
|
|
||||||
|
|
||||||
-- Return the currently set cabal version, if any.
|
-- Return the currently set cabal version, if any.
|
||||||
cabalSet :: (MonadReader Settings m, MonadIO m, MonadThrow m, MonadCatch m) => m (Maybe Version)
|
cabalSet :: (MonadReader AppState m, MonadIO m, MonadThrow m, MonadCatch m) => m (Maybe Version)
|
||||||
cabalSet = do
|
cabalSet = do
|
||||||
Settings {dirs = Dirs {..}} <- ask
|
AppState {dirs = Dirs {..}} <- ask
|
||||||
let cabalbin = binDir </> [rel|cabal|]
|
let cabalbin = binDir </> [rel|cabal|]
|
||||||
b <- handleIO (\_ -> pure False) $ fmap (== SymbolicLink) $ liftIO $ getFileType cabalbin
|
b <- handleIO (\_ -> pure False) $ fmap (== SymbolicLink) $ liftIO $ getFileType cabalbin
|
||||||
if
|
if
|
||||||
@@ -301,6 +302,150 @@ cabalSet = do
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get all installed hls, by matching on
|
||||||
|
-- @~\/.ghcup\/bin/haskell-language-server-wrapper-<\hlsver\>@.
|
||||||
|
getInstalledHLSs :: (MonadReader AppState m, MonadIO m, MonadCatch m)
|
||||||
|
=> m [Either (Path Rel) Version]
|
||||||
|
getInstalledHLSs = do
|
||||||
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
|
bins <- liftIO $ handleIO (\_ -> pure []) $ findFiles
|
||||||
|
binDir
|
||||||
|
(makeRegexOpts compExtended
|
||||||
|
execBlank
|
||||||
|
([s|^haskell-language-server-wrapper-.*$|] :: ByteString)
|
||||||
|
)
|
||||||
|
vs <- forM bins $ \f ->
|
||||||
|
case
|
||||||
|
fmap
|
||||||
|
version
|
||||||
|
(fmap decUTF8Safe . B.stripPrefix "haskell-language-server-wrapper-" . toFilePath $ f)
|
||||||
|
of
|
||||||
|
Just (Right r) -> pure $ Right r
|
||||||
|
Just (Left _) -> pure $ Left f
|
||||||
|
Nothing -> pure $ Left f
|
||||||
|
pure $ vs
|
||||||
|
|
||||||
|
|
||||||
|
-- | Whether the given HLS version is installed.
|
||||||
|
hlsInstalled :: (MonadIO m, MonadReader AppState m, MonadCatch m) => Version -> m Bool
|
||||||
|
hlsInstalled ver = do
|
||||||
|
vers <- fmap rights $ getInstalledHLSs
|
||||||
|
pure $ elem ver $ vers
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-- Return the currently set hls version, if any.
|
||||||
|
hlsSet :: (MonadReader AppState m, MonadIO m, MonadThrow m, MonadCatch m) => m (Maybe Version)
|
||||||
|
hlsSet = do
|
||||||
|
AppState {dirs = Dirs {..}} <- ask
|
||||||
|
let hlsBin = binDir </> [rel|haskell-language-server-wrapper|]
|
||||||
|
|
||||||
|
liftIO $ handleIO' NoSuchThing (\_ -> pure $ Nothing) $ do
|
||||||
|
broken <- isBrokenSymlink hlsBin
|
||||||
|
if broken
|
||||||
|
then pure Nothing
|
||||||
|
else do
|
||||||
|
link <- readSymbolicLink $ toFilePath hlsBin
|
||||||
|
Just <$> linkVersion link
|
||||||
|
where
|
||||||
|
linkVersion :: MonadThrow m => ByteString -> m Version
|
||||||
|
linkVersion bs = do
|
||||||
|
t <- throwEither $ E.decodeUtf8' bs
|
||||||
|
throwEither $ MP.parse parser "" t
|
||||||
|
where
|
||||||
|
parser =
|
||||||
|
MP.chunk "haskell-language-server-wrapper-" *> version'
|
||||||
|
|
||||||
|
|
||||||
|
-- | Return the GHC versions the currently selected HLS supports.
|
||||||
|
hlsGHCVersions :: ( MonadReader AppState m
|
||||||
|
, MonadIO m
|
||||||
|
, MonadThrow m
|
||||||
|
, MonadCatch m
|
||||||
|
)
|
||||||
|
=> m [Version]
|
||||||
|
hlsGHCVersions = do
|
||||||
|
h <- hlsSet
|
||||||
|
vers <- forM h $ \h' -> do
|
||||||
|
bins <- hlsServerBinaries h'
|
||||||
|
pure $ fmap
|
||||||
|
(\bin ->
|
||||||
|
version
|
||||||
|
. decUTF8Safe
|
||||||
|
. fromJust
|
||||||
|
. B.stripPrefix "haskell-language-server-"
|
||||||
|
. head
|
||||||
|
. B.split _tilde
|
||||||
|
. toFilePath
|
||||||
|
$ bin
|
||||||
|
)
|
||||||
|
bins
|
||||||
|
pure . rights . concat . maybeToList $ vers
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get all server binaries for an hls version, if any.
|
||||||
|
hlsServerBinaries :: (MonadReader AppState m, MonadIO m)
|
||||||
|
=> Version
|
||||||
|
-> m [Path Rel]
|
||||||
|
hlsServerBinaries ver = do
|
||||||
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
|
liftIO $ handleIO (\_ -> pure []) $ findFiles
|
||||||
|
binDir
|
||||||
|
(makeRegexOpts
|
||||||
|
compExtended
|
||||||
|
execBlank
|
||||||
|
([s|^haskell-language-server-.*~|] <> escapeVerRex ver <> [s|$|] :: ByteString
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get the wrapper binary for an hls version, if any.
|
||||||
|
hlsWrapperBinary :: (MonadReader AppState m, MonadThrow m, MonadIO m)
|
||||||
|
=> Version
|
||||||
|
-> m (Maybe (Path Rel))
|
||||||
|
hlsWrapperBinary ver = do
|
||||||
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
|
wrapper <- liftIO $ handleIO (\_ -> pure []) $ findFiles
|
||||||
|
binDir
|
||||||
|
(makeRegexOpts
|
||||||
|
compExtended
|
||||||
|
execBlank
|
||||||
|
([s|^haskell-language-server-wrapper-|] <> escapeVerRex ver <> [s|$|] :: ByteString
|
||||||
|
)
|
||||||
|
)
|
||||||
|
case wrapper of
|
||||||
|
[] -> pure $ Nothing
|
||||||
|
[x] -> pure $ Just x
|
||||||
|
_ -> throwM $ UnexpectedListLength
|
||||||
|
"There were multiple hls wrapper binaries for a single version"
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get all binaries for an hls version, if any.
|
||||||
|
hlsAllBinaries :: (MonadReader AppState m, MonadIO m, MonadThrow m) => Version -> m [Path Rel]
|
||||||
|
hlsAllBinaries ver = do
|
||||||
|
hls <- hlsServerBinaries ver
|
||||||
|
wrapper <- hlsWrapperBinary ver
|
||||||
|
pure (maybeToList wrapper ++ hls)
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get the active symlinks for hls.
|
||||||
|
hlsSymlinks :: (MonadReader AppState m, MonadIO m, MonadCatch m) => m [Path Rel]
|
||||||
|
hlsSymlinks = do
|
||||||
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
|
oldSyms <- liftIO $ handleIO (\_ -> pure []) $ findFiles
|
||||||
|
binDir
|
||||||
|
(makeRegexOpts compExtended
|
||||||
|
execBlank
|
||||||
|
([s|^haskell-language-server-.*$|] :: ByteString)
|
||||||
|
)
|
||||||
|
filterM
|
||||||
|
( fmap (== SymbolicLink)
|
||||||
|
. liftIO
|
||||||
|
. getFileType
|
||||||
|
. (binDir </>)
|
||||||
|
)
|
||||||
|
oldSyms
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
-----------------------------------------
|
-----------------------------------------
|
||||||
@@ -311,7 +456,7 @@ cabalSet = do
|
|||||||
-- | Extract (major, minor) from any version.
|
-- | Extract (major, minor) from any version.
|
||||||
getMajorMinorV :: MonadThrow m => Version -> m (Int, Int)
|
getMajorMinorV :: MonadThrow m => Version -> m (Int, Int)
|
||||||
getMajorMinorV Version {..} = case _vChunks of
|
getMajorMinorV Version {..} = case _vChunks of
|
||||||
([Digits x] : [Digits y] : _) -> pure (fromIntegral x, fromIntegral y)
|
((Digits x :| []) :| ((Digits y :| []):_)) -> pure (fromIntegral x, fromIntegral y)
|
||||||
_ -> throwM $ ParseError "Could not parse X.Y from version"
|
_ -> throwM $ ParseError "Could not parse X.Y from version"
|
||||||
|
|
||||||
|
|
||||||
@@ -323,7 +468,7 @@ matchMajor v' major' minor' = case getMajorMinorV v' of
|
|||||||
|
|
||||||
-- | Get the latest installed full GHC version that satisfies X.Y.
|
-- | Get the latest installed full GHC version that satisfies X.Y.
|
||||||
-- This reads `ghcupGHCBaseDir`.
|
-- This reads `ghcupGHCBaseDir`.
|
||||||
getGHCForMajor :: (MonadReader Settings m, MonadIO m, MonadThrow m)
|
getGHCForMajor :: (MonadReader AppState m, MonadIO m, MonadThrow m)
|
||||||
=> Int -- ^ major version component
|
=> Int -- ^ major version component
|
||||||
-> Int -- ^ minor version component
|
-> Int -- ^ minor version component
|
||||||
-> Maybe Text -- ^ the target triple
|
-> Maybe Text -- ^ the target triple
|
||||||
@@ -459,16 +604,16 @@ getLatestBaseVersion av pvpVer =
|
|||||||
|
|
||||||
|
|
||||||
-----------------------
|
-----------------------
|
||||||
--[ Settings Getter ]--
|
--[ AppState Getter ]--
|
||||||
-----------------------
|
-----------------------
|
||||||
|
|
||||||
|
|
||||||
getCache :: MonadReader Settings m => m Bool
|
getCache :: MonadReader AppState m => m Bool
|
||||||
getCache = ask <&> cache
|
getCache = ask <&> cache . settings
|
||||||
|
|
||||||
|
|
||||||
getDownloader :: MonadReader Settings m => m Downloader
|
getDownloader :: MonadReader AppState m => m Downloader
|
||||||
getDownloader = ask <&> downloader
|
getDownloader = ask <&> downloader . settings
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -489,7 +634,7 @@ urlBaseName = parseRel . snd . B.breakEnd (== _slash) . urlDecode False
|
|||||||
-- Returns unversioned relative files, e.g.:
|
-- Returns unversioned relative files, e.g.:
|
||||||
--
|
--
|
||||||
-- - @["hsc2hs","haddock","hpc","runhaskell","ghc","ghc-pkg","ghci","runghc","hp2ps"]@
|
-- - @["hsc2hs","haddock","hpc","runhaskell","ghc","ghc-pkg","ghci","runghc","hp2ps"]@
|
||||||
ghcToolFiles :: (MonadReader Settings m, MonadThrow m, MonadFail m, MonadIO m)
|
ghcToolFiles :: (MonadReader AppState m, MonadThrow m, MonadFail m, MonadIO m)
|
||||||
=> GHCTargetVersion
|
=> GHCTargetVersion
|
||||||
-> Excepts '[NotInstalled] m [Path Rel]
|
-> Excepts '[NotInstalled] m [Path Rel]
|
||||||
ghcToolFiles ver = do
|
ghcToolFiles ver = do
|
||||||
@@ -542,7 +687,7 @@ ghcUpSrcBuiltFile = [rel|.ghcup_src_built|]
|
|||||||
|
|
||||||
|
|
||||||
-- | Calls gmake if it exists in PATH, otherwise make.
|
-- | Calls gmake if it exists in PATH, otherwise make.
|
||||||
make :: (MonadThrow m, MonadIO m, MonadReader Settings m)
|
make :: (MonadThrow m, MonadIO m, MonadReader AppState m)
|
||||||
=> [ByteString]
|
=> [ByteString]
|
||||||
-> Maybe (Path Abs)
|
-> Maybe (Path Abs)
|
||||||
-> m (Either ProcessError ())
|
-> m (Either ProcessError ())
|
||||||
@@ -595,13 +740,13 @@ getChangeLog dls tool (Right tag) =
|
|||||||
--
|
--
|
||||||
-- 1. the build directory, depending on the KeepDirs setting
|
-- 1. the build directory, depending on the KeepDirs setting
|
||||||
-- 2. the install destination, depending on whether the build failed
|
-- 2. the install destination, depending on whether the build failed
|
||||||
runBuildAction :: (Show (V e), MonadReader Settings m, MonadIO m, MonadMask m)
|
runBuildAction :: (Show (V e), MonadReader AppState m, MonadIO m, MonadMask m)
|
||||||
=> Path Abs -- ^ build directory
|
=> Path Abs -- ^ build directory (cleaned up depending on Settings)
|
||||||
-> Maybe (Path Abs) -- ^ install location (e.g. for GHC)
|
-> Maybe (Path Abs) -- ^ dir to *always* clean up on exception
|
||||||
-> Excepts e m a
|
-> Excepts e m a
|
||||||
-> Excepts '[BuildFailed] m a
|
-> Excepts '[BuildFailed] m a
|
||||||
runBuildAction bdir instdir action = do
|
runBuildAction bdir instdir action = do
|
||||||
Settings {..} <- lift ask
|
AppState { settings = Settings {..} } <- lift ask
|
||||||
let exAction = do
|
let exAction = do
|
||||||
forM_ instdir $ \dir ->
|
forM_ instdir $ \dir ->
|
||||||
liftIO $ hideError doesNotExistErrorType $ deleteDirRecursive dir
|
liftIO $ hideError doesNotExistErrorType $ deleteDirRecursive dir
|
||||||
@@ -621,3 +766,25 @@ runBuildAction bdir instdir action = do
|
|||||||
when (keepDirs == Never || keepDirs == Errors) $ liftIO $ deleteDirRecursive
|
when (keepDirs == Never || keepDirs == Errors) $ liftIO $ deleteDirRecursive
|
||||||
bdir
|
bdir
|
||||||
pure v
|
pure v
|
||||||
|
|
||||||
|
|
||||||
|
-- | More permissive version of 'createDirRecursive'. This doesn't
|
||||||
|
-- error when the destination is a symlink to a directory.
|
||||||
|
createDirRecursive' :: Path b -> IO ()
|
||||||
|
createDirRecursive' p =
|
||||||
|
handleIO (\e -> if isAlreadyExistsError e then isSymlinkDir e else throwIO e)
|
||||||
|
. createDirRecursive newDirPerms
|
||||||
|
$ p
|
||||||
|
|
||||||
|
where
|
||||||
|
isSymlinkDir e = do
|
||||||
|
ft <- getFileType p
|
||||||
|
case ft of
|
||||||
|
SymbolicLink -> do
|
||||||
|
rp <- canonicalizePath p
|
||||||
|
rft <- getFileType rp
|
||||||
|
case rft of
|
||||||
|
Directory -> pure ()
|
||||||
|
_ -> throwIO e
|
||||||
|
_ -> throwIO e
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
@@ -14,16 +15,18 @@ Portability : POSIX
|
|||||||
-}
|
-}
|
||||||
module GHCup.Utils.Dirs
|
module GHCup.Utils.Dirs
|
||||||
( getDirs
|
( getDirs
|
||||||
|
, ghcupConfigFile
|
||||||
, ghcupGHCBaseDir
|
, ghcupGHCBaseDir
|
||||||
, ghcupGHCDir
|
, ghcupGHCDir
|
||||||
, parseGHCupGHCDir
|
|
||||||
, mkGhcupTmpDir
|
, mkGhcupTmpDir
|
||||||
, withGHCupTmpDir
|
, parseGHCupGHCDir
|
||||||
, relativeSymlink
|
, relativeSymlink
|
||||||
|
, withGHCupTmpDir
|
||||||
)
|
)
|
||||||
where
|
where
|
||||||
|
|
||||||
|
|
||||||
|
import GHCup.Errors
|
||||||
import GHCup.Types
|
import GHCup.Types
|
||||||
import GHCup.Types.JSON ( )
|
import GHCup.Types.JSON ( )
|
||||||
import GHCup.Utils.MegaParsec
|
import GHCup.Utils.MegaParsec
|
||||||
@@ -34,8 +37,11 @@ import Control.Exception.Safe
|
|||||||
import Control.Monad
|
import Control.Monad
|
||||||
import Control.Monad.Reader
|
import Control.Monad.Reader
|
||||||
import Control.Monad.Trans.Resource
|
import Control.Monad.Trans.Resource
|
||||||
|
import Data.Bifunctor
|
||||||
import Data.ByteString ( ByteString )
|
import Data.ByteString ( ByteString )
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
|
import GHC.IO.Exception ( IOErrorType(NoSuchThing) )
|
||||||
|
import Haskus.Utils.Variant.Excepts
|
||||||
import HPath
|
import HPath
|
||||||
import HPath.IO
|
import HPath.IO
|
||||||
import Optics
|
import Optics
|
||||||
@@ -49,8 +55,10 @@ import System.Posix.Env.ByteString ( getEnv
|
|||||||
import System.Posix.FilePath hiding ( (</>) )
|
import System.Posix.FilePath hiding ( (</>) )
|
||||||
import System.Posix.Temp.ByteString ( mkdtemp )
|
import System.Posix.Temp.ByteString ( mkdtemp )
|
||||||
|
|
||||||
|
import qualified Data.ByteString.Lazy as L
|
||||||
import qualified Data.ByteString.UTF8 as UTF8
|
import qualified Data.ByteString.UTF8 as UTF8
|
||||||
import qualified Data.Text.Encoding as E
|
import qualified Data.Text.Encoding as E
|
||||||
|
import qualified Data.Yaml as Y
|
||||||
import qualified System.Posix.FilePath as FP
|
import qualified System.Posix.FilePath as FP
|
||||||
import qualified System.Posix.User as PU
|
import qualified System.Posix.User as PU
|
||||||
import qualified Text.Megaparsec as MP
|
import qualified Text.Megaparsec as MP
|
||||||
@@ -84,6 +92,28 @@ ghcupBaseDir = do
|
|||||||
pure (bdir </> [rel|.ghcup|])
|
pure (bdir </> [rel|.ghcup|])
|
||||||
|
|
||||||
|
|
||||||
|
-- | ~/.ghcup by default
|
||||||
|
--
|
||||||
|
-- If 'GHCUP_USE_XDG_DIRS' is set (to anything),
|
||||||
|
-- then uses 'XDG_CONFIG_HOME/ghcup' as per xdg spec.
|
||||||
|
ghcupConfigDir :: IO (Path Abs)
|
||||||
|
ghcupConfigDir = do
|
||||||
|
xdg <- useXDG
|
||||||
|
if xdg
|
||||||
|
then do
|
||||||
|
bdir <- getEnv "XDG_CONFIG_HOME" >>= \case
|
||||||
|
Just r -> parseAbs r
|
||||||
|
Nothing -> do
|
||||||
|
home <- liftIO getHomeDirectory
|
||||||
|
pure (home </> [rel|.config|])
|
||||||
|
pure (bdir </> [rel|ghcup|])
|
||||||
|
else do
|
||||||
|
bdir <- getEnv "GHCUP_INSTALL_BASE_PREFIX" >>= \case
|
||||||
|
Just r -> parseAbs r
|
||||||
|
Nothing -> liftIO getHomeDirectory
|
||||||
|
pure (bdir </> [rel|.ghcup|])
|
||||||
|
|
||||||
|
|
||||||
-- | If 'GHCUP_USE_XDG_DIRS' is set (to anything),
|
-- | If 'GHCUP_USE_XDG_DIRS' is set (to anything),
|
||||||
-- then uses 'XDG_BIN_HOME' env var or defaults to '~/.local/bin'
|
-- then uses 'XDG_BIN_HOME' env var or defaults to '~/.local/bin'
|
||||||
-- (which, sadly is not strictly xdg spec).
|
-- (which, sadly is not strictly xdg spec).
|
||||||
@@ -142,27 +172,44 @@ getDirs = do
|
|||||||
binDir <- ghcupBinDir
|
binDir <- ghcupBinDir
|
||||||
cacheDir <- ghcupCacheDir
|
cacheDir <- ghcupCacheDir
|
||||||
logsDir <- ghcupLogsDir
|
logsDir <- ghcupLogsDir
|
||||||
|
confDir <- ghcupConfigDir
|
||||||
pure Dirs { .. }
|
pure Dirs { .. }
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-------------------
|
||||||
|
--[ GHCup files ]--
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
|
||||||
|
ghcupConfigFile :: (MonadIO m)
|
||||||
|
=> Excepts '[JSONError] m UserSettings
|
||||||
|
ghcupConfigFile = do
|
||||||
|
confDir <- liftIO $ ghcupConfigDir
|
||||||
|
let file = confDir </> [rel|config.yaml|]
|
||||||
|
bs <- liftIO $ handleIO' NoSuchThing (\_ -> pure $ Nothing) $ fmap Just $ readFile file
|
||||||
|
case bs of
|
||||||
|
Nothing -> pure defaultUserSettings
|
||||||
|
Just bs' -> lE' JSONDecodeError . bimap show id . Y.decodeEither' . L.toStrict $ bs'
|
||||||
|
|
||||||
|
|
||||||
-------------------------
|
-------------------------
|
||||||
--[ GHCup directories ]--
|
--[ GHCup directories ]--
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
|
|
||||||
-- | ~/.ghcup/ghc by default.
|
-- | ~/.ghcup/ghc by default.
|
||||||
ghcupGHCBaseDir :: (MonadReader Settings m) => m (Path Abs)
|
ghcupGHCBaseDir :: (MonadReader AppState m) => m (Path Abs)
|
||||||
ghcupGHCBaseDir = do
|
ghcupGHCBaseDir = do
|
||||||
Settings {..} <- ask
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
pure (baseDir dirs </> [rel|ghc|])
|
pure (baseDir </> [rel|ghc|])
|
||||||
|
|
||||||
|
|
||||||
-- | Gets '~/.ghcup/ghc/<ghcupGHCDir>'.
|
-- | Gets '~/.ghcup/ghc/<ghcupGHCDir>'.
|
||||||
-- The dir may be of the form
|
-- The dir may be of the form
|
||||||
-- * armv7-unknown-linux-gnueabihf-8.8.3
|
-- * armv7-unknown-linux-gnueabihf-8.8.3
|
||||||
-- * 8.8.4
|
-- * 8.8.4
|
||||||
ghcupGHCDir :: (MonadReader Settings m, MonadThrow m)
|
ghcupGHCDir :: (MonadReader AppState m, MonadThrow m)
|
||||||
=> GHCTargetVersion
|
=> GHCTargetVersion
|
||||||
-> m (Path Abs)
|
-> m (Path Abs)
|
||||||
ghcupGHCDir ver = do
|
ghcupGHCDir ver = do
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE TemplateHaskell #-}
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
{-# LANGUAGE ViewPatterns #-}
|
||||||
|
|
||||||
{-|
|
{-|
|
||||||
Module : GHCup.Utils.File
|
Module : GHCup.Utils.File
|
||||||
@@ -25,6 +26,7 @@ import Control.Concurrent.Async
|
|||||||
import Control.Exception ( evaluate )
|
import Control.Exception ( evaluate )
|
||||||
import Control.Exception.Safe
|
import Control.Exception.Safe
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
|
import Control.Monad.Logger
|
||||||
import Control.Monad.Reader
|
import Control.Monad.Reader
|
||||||
import Control.Monad.Trans.State.Strict
|
import Control.Monad.Trans.State.Strict
|
||||||
import Data.ByteString ( ByteString )
|
import Data.ByteString ( ByteString )
|
||||||
@@ -33,6 +35,7 @@ import Data.Functor
|
|||||||
import Data.IORef
|
import Data.IORef
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.Sequence ( Seq, (|>) )
|
import Data.Sequence ( Seq, (|>) )
|
||||||
|
import Data.String.Interpolate
|
||||||
import Data.Text ( Text )
|
import Data.Text ( Text )
|
||||||
import Data.Void
|
import Data.Void
|
||||||
import Data.Word8
|
import Data.Word8
|
||||||
@@ -46,6 +49,7 @@ import System.IO.Error
|
|||||||
import System.Posix.Directory.ByteString
|
import System.Posix.Directory.ByteString
|
||||||
import System.Posix.FD as FD
|
import System.Posix.FD as FD
|
||||||
import System.Posix.FilePath hiding ( (</>) )
|
import System.Posix.FilePath hiding ( (</>) )
|
||||||
|
import System.Posix.Files.ByteString
|
||||||
import System.Posix.Foreign ( oExcl )
|
import System.Posix.Foreign ( oExcl )
|
||||||
import "unix" System.Posix.IO.ByteString
|
import "unix" System.Posix.IO.ByteString
|
||||||
hiding ( openFd )
|
hiding ( openFd )
|
||||||
@@ -113,7 +117,7 @@ executeOut path args chdir = captureOutStreams $ do
|
|||||||
SPPB.executeFile (toFilePath path) True args Nothing
|
SPPB.executeFile (toFilePath path) True args Nothing
|
||||||
|
|
||||||
|
|
||||||
execLogged :: (MonadReader Settings m, MonadIO m, MonadThrow m)
|
execLogged :: (MonadReader AppState m, MonadIO m, MonadThrow m)
|
||||||
=> ByteString -- ^ thing to execute
|
=> ByteString -- ^ thing to execute
|
||||||
-> Bool -- ^ whether to search PATH for the thing
|
-> Bool -- ^ whether to search PATH for the thing
|
||||||
-> [ByteString] -- ^ args for the thing
|
-> [ByteString] -- ^ args for the thing
|
||||||
@@ -122,7 +126,7 @@ execLogged :: (MonadReader Settings m, MonadIO m, MonadThrow m)
|
|||||||
-> Maybe [(ByteString, ByteString)] -- ^ optional environment
|
-> Maybe [(ByteString, ByteString)] -- ^ optional environment
|
||||||
-> m (Either ProcessError ())
|
-> m (Either ProcessError ())
|
||||||
execLogged exe spath args lfile chdir env = do
|
execLogged exe spath args lfile chdir env = do
|
||||||
Settings {dirs = Dirs {..}, ..} <- ask
|
AppState { settings = Settings {..}, dirs = Dirs {..} } <- ask
|
||||||
logfile <- (logsDir </>) <$> parseRel (toFilePath lfile <> ".log")
|
logfile <- (logsDir </>) <$> parseRel (toFilePath lfile <> ".log")
|
||||||
liftIO $ bracket (createFile (toFilePath logfile) newFilePerms)
|
liftIO $ bracket (createFile (toFilePath logfile) newFilePerms)
|
||||||
closeFd
|
closeFd
|
||||||
@@ -375,7 +379,7 @@ toProcessError :: ByteString
|
|||||||
-> Maybe ProcessStatus
|
-> Maybe ProcessStatus
|
||||||
-> Either ProcessError ()
|
-> Either ProcessError ()
|
||||||
toProcessError exe args mps = case mps of
|
toProcessError exe args mps = case mps of
|
||||||
Just (SPPB.Exited (ExitFailure i)) -> Left $ NonZeroExit i exe args
|
Just (SPPB.Exited (ExitFailure xi)) -> Left $ NonZeroExit xi exe args
|
||||||
Just (SPPB.Exited ExitSuccess ) -> Right ()
|
Just (SPPB.Exited ExitSuccess ) -> Right ()
|
||||||
Just (Terminated _ _ ) -> Left $ PTerminated exe args
|
Just (Terminated _ _ ) -> Left $ PTerminated exe args
|
||||||
Just (Stopped _ ) -> Left $ PStopped exe args
|
Just (Stopped _ ) -> Left $ PStopped exe args
|
||||||
@@ -434,3 +438,15 @@ isBrokenSymlink p =
|
|||||||
$ do
|
$ do
|
||||||
_ <- canonicalizePath p
|
_ <- canonicalizePath p
|
||||||
pure False
|
pure False
|
||||||
|
|
||||||
|
|
||||||
|
chmod_777 :: (MonadLogger m, MonadIO m) => Path a -> m ()
|
||||||
|
chmod_777 (toFilePath -> fp) = do
|
||||||
|
let exe_mode =
|
||||||
|
newFilePerms
|
||||||
|
`unionFileModes` ownerExecuteMode
|
||||||
|
`unionFileModes` groupExecuteMode
|
||||||
|
`unionFileModes` otherExecuteMode
|
||||||
|
$(logDebug) [i|chmod 777 #{fp}|]
|
||||||
|
liftIO $ setFileMode fp exe_mode
|
||||||
|
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ Here we define our main logger.
|
|||||||
module GHCup.Utils.Logger where
|
module GHCup.Utils.Logger where
|
||||||
|
|
||||||
import GHCup.Types
|
import GHCup.Types
|
||||||
|
import GHCup.Utils
|
||||||
|
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
import Control.Monad.IO.Class
|
import Control.Monad.IO.Class
|
||||||
@@ -64,12 +65,12 @@ myLoggerT LoggerConfig {..} loggingt = runLoggingT loggingt mylogger
|
|||||||
rawOutter outr
|
rawOutter outr
|
||||||
|
|
||||||
|
|
||||||
initGHCupFileLogging :: (MonadIO m, MonadReader Settings m) => Path Rel -> m (Path Abs)
|
initGHCupFileLogging :: (MonadIO m, MonadReader AppState m) => Path Rel -> m (Path Abs)
|
||||||
initGHCupFileLogging context = do
|
initGHCupFileLogging context = do
|
||||||
Settings {dirs = Dirs {..}} <- ask
|
AppState {dirs = Dirs {..}} <- ask
|
||||||
let logfile = logsDir </> context
|
let logfile = logsDir </> context
|
||||||
liftIO $ do
|
liftIO $ do
|
||||||
createDirRecursive newDirPerms logsDir
|
createDirRecursive' logsDir
|
||||||
hideError doesNotExistErrorType $ deleteFile logfile
|
hideError doesNotExistErrorType $ deleteFile logfile
|
||||||
createRegularFile newFilePerms logfile
|
createRegularFile newFilePerms logfile
|
||||||
pure logfile
|
pure logfile
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ import Data.Text ( Text )
|
|||||||
import Data.Versions
|
import Data.Versions
|
||||||
import Data.Void
|
import Data.Void
|
||||||
|
|
||||||
|
import qualified Data.List.NonEmpty as NE
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
import qualified Text.Megaparsec as MP
|
import qualified Text.Megaparsec as MP
|
||||||
|
|
||||||
@@ -90,6 +91,8 @@ ghcTargetVerP =
|
|||||||
(Digits _) -> True
|
(Digits _) -> True
|
||||||
(Str _) -> False
|
(Str _) -> False
|
||||||
)
|
)
|
||||||
|
. fmap NE.toList
|
||||||
|
. NE.toList
|
||||||
$ (_vChunks v)
|
$ (_vChunks v)
|
||||||
if startsWithDigists && not (isJust (_vEpoch v))
|
if startsWithDigists && not (isJust (_vEpoch v))
|
||||||
then pure $ prettyVer v
|
then pure $ prettyVer v
|
||||||
|
|||||||
@@ -31,11 +31,13 @@ import Data.ByteString ( ByteString )
|
|||||||
import Data.String
|
import Data.String
|
||||||
import Data.Text ( Text )
|
import Data.Text ( Text )
|
||||||
import Data.Versions
|
import Data.Versions
|
||||||
|
import Data.Word8
|
||||||
import Haskus.Utils.Types.List
|
import Haskus.Utils.Types.List
|
||||||
import Haskus.Utils.Variant.Excepts
|
import Haskus.Utils.Variant.Excepts
|
||||||
import System.IO.Error
|
import System.IO.Error
|
||||||
import System.Posix.Env.ByteString ( getEnvironment )
|
import System.Posix.Env.ByteString ( getEnvironment )
|
||||||
|
|
||||||
|
import qualified Data.ByteString as B
|
||||||
import qualified Data.ByteString.Lazy as L
|
import qualified Data.ByteString.Lazy as L
|
||||||
import qualified Data.Strict.Maybe as S
|
import qualified Data.Strict.Maybe as S
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
@@ -275,3 +277,13 @@ decUTF8Safe = E.decodeUtf8With E.lenientDecode
|
|||||||
|
|
||||||
decUTF8Safe' :: L.ByteString -> Text
|
decUTF8Safe' :: L.ByteString -> Text
|
||||||
decUTF8Safe' = TL.toStrict . TLE.decodeUtf8With E.lenientDecode
|
decUTF8Safe' = TL.toStrict . TLE.decodeUtf8With E.lenientDecode
|
||||||
|
|
||||||
|
|
||||||
|
-- | Escape a version for use in regex
|
||||||
|
escapeVerRex :: Version -> ByteString
|
||||||
|
escapeVerRex = B.pack . go . B.unpack . verToBS
|
||||||
|
where
|
||||||
|
go [] = []
|
||||||
|
go (x : xs) | x == _period = [_backslash, _period] ++ go xs
|
||||||
|
| otherwise = x : go xs
|
||||||
|
|
||||||
|
|||||||
@@ -42,6 +42,8 @@ deriving instance Data SemVer
|
|||||||
deriving instance Lift SemVer
|
deriving instance Lift SemVer
|
||||||
deriving instance Data Mess
|
deriving instance Data Mess
|
||||||
deriving instance Lift Mess
|
deriving instance Lift Mess
|
||||||
|
deriving instance Data MChunk
|
||||||
|
deriving instance Lift MChunk
|
||||||
deriving instance Data PVP
|
deriving instance Data PVP
|
||||||
deriving instance Lift PVP
|
deriving instance Lift PVP
|
||||||
deriving instance Lift VSep
|
deriving instance Lift VSep
|
||||||
|
|||||||
@@ -22,11 +22,11 @@ import qualified Data.Text as T
|
|||||||
|
|
||||||
-- | This reflects the API version of the YAML.
|
-- | This reflects the API version of the YAML.
|
||||||
ghcupURL :: URI
|
ghcupURL :: URI
|
||||||
ghcupURL = [uri|https://www.haskell.org/ghcup/data/ghcup-0.0.2.yaml|]
|
ghcupURL = [uri|https://www.haskell.org/ghcup/data/ghcup-0.0.3.yaml|]
|
||||||
|
|
||||||
-- | The current ghcup version.
|
-- | The current ghcup version.
|
||||||
ghcUpVer :: PVP
|
ghcUpVer :: PVP
|
||||||
ghcUpVer = [pver|0.1.9|]
|
ghcUpVer = [pver|0.1.12|]
|
||||||
|
|
||||||
-- | ghcup version as numeric string.
|
-- | ghcup version as numeric string.
|
||||||
numericVer :: String
|
numericVer :: String
|
||||||
|
|||||||
72
stack.yaml
Normal file
72
stack.yaml
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
resolver: lts-16.17
|
||||||
|
|
||||||
|
packages:
|
||||||
|
- .
|
||||||
|
|
||||||
|
extra-deps:
|
||||||
|
- 3rdparty/lzma
|
||||||
|
- 3rdparty/lzma-clib
|
||||||
|
- 3rdparty/zlib
|
||||||
|
|
||||||
|
- git: https://github.com/haskus/packages.git
|
||||||
|
commit: 80a1c5fc07f7226c424250ec17f674cd4d618f42
|
||||||
|
subdirs:
|
||||||
|
- haskus-utils-types
|
||||||
|
|
||||||
|
- git: https://github.com/hasufell/hpath.git
|
||||||
|
commit: bf6d28cf989b70286e12fecc183d5bbf5454a1a2
|
||||||
|
subdirs:
|
||||||
|
- hpath-io
|
||||||
|
- hpath-directory
|
||||||
|
|
||||||
|
- git: https://github.com/hasufell/text-conversions.git
|
||||||
|
commit: 9abf0e5e5664a3178367597c32db19880477a53c
|
||||||
|
|
||||||
|
- IfElse-0.85@sha256:6939b94acc6a55f545f63a168a349dd2fbe4b9a7cca73bf60282db5cc6aa47d2,445
|
||||||
|
- QuickCheck-2.14.1@sha256:01e46d7b0a8d3148288ec977625f62d5516ebb5031a50c63f0453863301b4a79,7736
|
||||||
|
- ascii-string-1.0.1.4@sha256:fa34f1d9ba57e8e89c0d4c9cef5e01ba32cb2d4373d13f92dcc0b531a6c6749b,2582
|
||||||
|
- base64-bytestring-1.1.0.0@sha256:190264fef9e65d9085f00ccda419137096d1dc94777c58272bc96821dc7f37c3,2334
|
||||||
|
- brick-0.55@sha256:f98736eca0cd694837062e06da4655eed969d53b789dfd919716e9b6f5b4c5ce,15858
|
||||||
|
- brotli-0.0.0.0@sha256:2bf383a4cd308745740986be0b18381c5a0784393fe69b91456aacb2d603de46,2964
|
||||||
|
- brotli-streams-0.0.0.0@sha256:1af1e22f67b8bfd6ad0d05e61825e7a178d738f689ebbb21c1aab5f1bbcae176,2331
|
||||||
|
- chs-cabal-0.1.1.0@sha256:20ec6a9fb5ab6991f1a4adf157c537bd5d3b98d08d3c09c387c954c7c50bd011,1153
|
||||||
|
- chs-deps-0.1.0.0@sha256:0cdada6d2c682c41b20331b8c63c2ecfc7e806928585195fd544c9d41f3074fd,2496
|
||||||
|
- composition-prelude-3.0.0.0@sha256:7407835ce8c1e0e2fd6febd25391b12989b216773e685e3cf95bd89072af0ecc,1149
|
||||||
|
- haskus-utils-data-1.3@sha256:f62c4e49021b463185d043f7b69c727b63af641a71d7edd582d9f4f98e80e500,1466
|
||||||
|
- haskus-utils-variant-3.0@sha256:8d51e45d3b664e61ccc25a58b37c0ccc4ee7537138b9fee21cd15c356906dd34,2159
|
||||||
|
- hpath-0.11.0@sha256:12b8405bee13d0007d644a888ef8407069ce7bbbd76970f8746b801447124ade,1440
|
||||||
|
- hpath-filepath-0.10.4@sha256:e9e44fb5fdbade7f30b5b5451257dbee15b6ef1aae4060034d73008bb3b5d878,1269
|
||||||
|
- hpath-posix-0.13.2@sha256:eec4ff2b00dc86be847aca0f409fc8f6212ffd2170ec36a17dc9a52b46562392,1615
|
||||||
|
- http-io-streams-0.1.4.0@sha256:9a74a059daeddf7a41d361919190b9f4d4292f05e0e4bdf156e2098a116a8145,3582
|
||||||
|
- libarchive-3.0.0.0@sha256:e4157b307acf16cca0ec3d398ac5093cc06f092b33a9743be559ef0f6c6ae52f,11204
|
||||||
|
- os-release-1.0.1@sha256:1281c62081f438fc3f0874d3bae6a4887d5964ac25261ba06e29d368ab173467,2716
|
||||||
|
- primitive-extras-0.8@sha256:fca0310150496867f5b9421fe1541ecda87fae17eae44885a29f9c52dd00c8ff,2963
|
||||||
|
- primitive-unlifted-0.1.3.0@sha256:a98f827740f5dcf097d885b3a47c32f4462204449620abc9d51b8c4f8619f9e6,1427
|
||||||
|
- random-1.2.0@sha256:4321209c8faedc034810ea8ed0dbc4a36f1a1df97b75af024219f2f533da57de,6094
|
||||||
|
- splitmix-0.1.0.1@sha256:22f9662e7e8b173421872241edd39350078a9ed4bb9e9f503948c5b483c79276,5253
|
||||||
|
- streamly-posix-0.1.0.1@sha256:5d89b806281035d34020387ed99dde1ddab282c7ed66df3b7cd010b38fd3517b,2138
|
||||||
|
- strict-base-0.4.0.0@sha256:2ff4e43cb95eedf2995558d7fc34d19362846413dd39e6aa6a5b3ea8228fef9f,1248
|
||||||
|
- tar-bytestring-0.6.3.2@sha256:88f29bed56b688c543a4cb3986402d64b360f76b3fd9b88ac618b8344f8da712,5715
|
||||||
|
- versions-4.0.1@sha256:0f644c1587d38f0eb3c3fe364bf1822424db43cbd4d618d0e21473b062c45239,1936
|
||||||
|
- vty-5.30@sha256:4af3938d7b9e6096e222bf52d0ea5d39873bc6fe19febd34106906306af13730,20857
|
||||||
|
- xor-0.0.1.0@sha256:f8362b4a68562b9afbcd727ff64c1a303970df3a032e0033d2f4c094c3501df3,2243
|
||||||
|
|
||||||
|
flags:
|
||||||
|
http-io-streams:
|
||||||
|
brotli: false
|
||||||
|
|
||||||
|
libarchive:
|
||||||
|
system-libarchive: false
|
||||||
|
|
||||||
|
ghcup:
|
||||||
|
tui: true
|
||||||
|
internal-downloader: true
|
||||||
|
|
||||||
|
system-ghc: true
|
||||||
|
compiler: ghc-8.8.4
|
||||||
|
compiler-check: match-exact
|
||||||
|
|
||||||
|
ghc-options:
|
||||||
|
"$locals": -O2
|
||||||
|
streamly: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
|
ghcup: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
193
test/GHCup/ArbitraryTypes.hs
Normal file
193
test/GHCup/ArbitraryTypes.hs
Normal file
@@ -0,0 +1,193 @@
|
|||||||
|
{-# OPTIONS_GHC -Wno-orphans #-}
|
||||||
|
|
||||||
|
{-# LANGUAGE FlexibleInstances #-}
|
||||||
|
{-# LANGUAGE StandaloneDeriving #-}
|
||||||
|
{-# LANGUAGE DeriveGeneric #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE TypeApplications #-}
|
||||||
|
|
||||||
|
module GHCup.ArbitraryTypes where
|
||||||
|
|
||||||
|
|
||||||
|
import GHCup.Types
|
||||||
|
|
||||||
|
import Data.ByteString ( ByteString )
|
||||||
|
import Data.Versions
|
||||||
|
import Data.List.NonEmpty
|
||||||
|
import HPath
|
||||||
|
import Test.QuickCheck
|
||||||
|
import Test.QuickCheck.Arbitrary.ADT ( ToADTArbitrary )
|
||||||
|
import Test.QuickCheck.Arbitrary.Generic
|
||||||
|
import URI.ByteString
|
||||||
|
|
||||||
|
import qualified Data.Map.Strict as M
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Data.Text.Encoding as E
|
||||||
|
import qualified Data.Text.Lazy as T
|
||||||
|
( toStrict )
|
||||||
|
import qualified Data.Text.Lazy.Builder as B
|
||||||
|
import qualified Data.Text.Lazy.Builder.Int as B
|
||||||
|
|
||||||
|
|
||||||
|
-----------------
|
||||||
|
--[ utilities ]--
|
||||||
|
-----------------
|
||||||
|
|
||||||
|
intToText :: Integral a => a -> T.Text
|
||||||
|
intToText = T.toStrict . B.toLazyText . B.decimal
|
||||||
|
|
||||||
|
genVer :: Gen (Int, Int, Int)
|
||||||
|
genVer =
|
||||||
|
(\x y z -> (getPositive x, getPositive y, getPositive z))
|
||||||
|
<$> arbitrary
|
||||||
|
<*> arbitrary
|
||||||
|
<*> arbitrary
|
||||||
|
|
||||||
|
|
||||||
|
instance ToADTArbitrary GHCupInfo
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
----------------------
|
||||||
|
--[ base arbitrary ]--
|
||||||
|
----------------------
|
||||||
|
|
||||||
|
instance Arbitrary T.Text where
|
||||||
|
arbitrary = fmap T.pack $ listOf $ elements ['a' .. 'z']
|
||||||
|
shrink xs = T.pack <$> shrink (T.unpack xs)
|
||||||
|
|
||||||
|
instance Arbitrary (NonEmpty Word) where
|
||||||
|
arbitrary = fmap fromList $ listOf1 $ arbitrary
|
||||||
|
|
||||||
|
-- utf8 encoded bytestring
|
||||||
|
instance Arbitrary ByteString where
|
||||||
|
arbitrary = fmap (E.encodeUtf8 . T.pack) $ listOf $ elements ['a' .. 'z']
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
---------------------
|
||||||
|
--[ uri arbitrary ]--
|
||||||
|
---------------------
|
||||||
|
|
||||||
|
instance Arbitrary Scheme where
|
||||||
|
arbitrary = oneof [ Scheme <$> pure "http", Scheme <$> pure "https" ]
|
||||||
|
|
||||||
|
instance Arbitrary Host where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary Port where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary (URIRef Absolute) where
|
||||||
|
arbitrary =
|
||||||
|
URI <$> arbitrary <*> pure Nothing <*> arbitrary <*> (Query <$> pure []) <*> pure Nothing
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-------------------------
|
||||||
|
--[ version arbitrary ]--
|
||||||
|
-------------------------
|
||||||
|
|
||||||
|
instance Arbitrary Mess where
|
||||||
|
arbitrary = do
|
||||||
|
(x, y, z) <- genVer
|
||||||
|
pure
|
||||||
|
$ either (error . show) id
|
||||||
|
$ mess
|
||||||
|
$ (intToText x <> "." <> intToText y <> "." <> intToText z)
|
||||||
|
|
||||||
|
instance Arbitrary Version where
|
||||||
|
arbitrary = do
|
||||||
|
(x, y, z) <- genVer
|
||||||
|
pure
|
||||||
|
$ either (error . show) id
|
||||||
|
$ version
|
||||||
|
$ (intToText x <> "." <> intToText y <> "." <> intToText z)
|
||||||
|
|
||||||
|
instance Arbitrary SemVer where
|
||||||
|
arbitrary = do
|
||||||
|
(x, y, z) <- genVer
|
||||||
|
pure
|
||||||
|
$ either (error . show) id
|
||||||
|
$ semver
|
||||||
|
$ (intToText x <> "." <> intToText y <> "." <> intToText z)
|
||||||
|
|
||||||
|
instance Arbitrary PVP where
|
||||||
|
arbitrary = do
|
||||||
|
(x, y, z) <- genVer
|
||||||
|
pure
|
||||||
|
$ either (error . show) id
|
||||||
|
$ pvp
|
||||||
|
$ (intToText x <> "." <> intToText y <> "." <> intToText z)
|
||||||
|
|
||||||
|
instance Arbitrary Versioning where
|
||||||
|
arbitrary = Ideal <$> arbitrary
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-----------------------
|
||||||
|
--[ ghcup arbitrary ]--
|
||||||
|
-----------------------
|
||||||
|
|
||||||
|
instance Arbitrary Requirements where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary DownloadInfo where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary LinuxDistro where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary Platform where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary Tag where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary Architecture where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary VersionInfo where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary (Path Rel) where
|
||||||
|
arbitrary =
|
||||||
|
(either (error . show) id . parseRel . E.encodeUtf8 . T.pack)
|
||||||
|
<$> (listOf1 $ elements ['a' .. 'z'])
|
||||||
|
|
||||||
|
instance Arbitrary TarDir where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary Tool where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary GHCupInfo where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
|
||||||
|
-- our maps are nested... the default size easily blows up most ppls ram
|
||||||
|
|
||||||
|
instance {-# OVERLAPS #-} Arbitrary v => Arbitrary (M.Map Tool v) where
|
||||||
|
arbitrary = resize 8 $ M.fromList <$> arbitrary
|
||||||
|
|
||||||
|
instance {-# OVERLAPS #-} Arbitrary v => Arbitrary (M.Map (Maybe Version) v) where
|
||||||
|
arbitrary = resize 8 $ M.fromList <$> arbitrary
|
||||||
|
|
||||||
|
instance {-# OVERLAPS #-} Arbitrary v => Arbitrary (M.Map Platform v) where
|
||||||
|
arbitrary = resize 8 $ M.fromList <$> arbitrary
|
||||||
|
|
||||||
|
instance {-# OVERLAPS #-} Arbitrary v => Arbitrary (M.Map (Maybe Versioning) v) where
|
||||||
|
arbitrary = resize 8 $ M.fromList <$> arbitrary
|
||||||
|
|
||||||
17
test/GHCup/Types/JSONSpec.hs
Normal file
17
test/GHCup/Types/JSONSpec.hs
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
{-# LANGUAGE DeriveGeneric #-}
|
||||||
|
{-# LANGUAGE TypeApplications #-}
|
||||||
|
|
||||||
|
module GHCup.Types.JSONSpec where
|
||||||
|
|
||||||
|
import GHCup.ArbitraryTypes ()
|
||||||
|
import GHCup.Types
|
||||||
|
import GHCup.Types.JSON ()
|
||||||
|
|
||||||
|
import Test.Aeson.GenericSpecs
|
||||||
|
import Test.Hspec
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
spec :: Spec
|
||||||
|
spec = do
|
||||||
|
roundtripAndGoldenSpecs (Proxy @GHCupInfo)
|
||||||
12
test/Main.hs
Normal file
12
test/Main.hs
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
import Test.Hspec.Runner
|
||||||
|
import Test.Hspec.Formatters
|
||||||
|
import qualified Spec
|
||||||
|
|
||||||
|
|
||||||
|
main :: IO ()
|
||||||
|
main =
|
||||||
|
hspecWith
|
||||||
|
defaultConfig { configFormatter = Just progress }
|
||||||
|
$ Spec.spec
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
module Main (main) where
|
|
||||||
|
|
||||||
main :: IO ()
|
|
||||||
main = putStrLn "Test suite not yet implemented."
|
|
||||||
2
test/Spec.hs
Normal file
2
test/Spec.hs
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
-- file test/Spec.hs
|
||||||
|
{-# OPTIONS_GHC -F -pgmF hspec-discover -optF --module-name=Spec #-}
|
||||||
Reference in New Issue
Block a user