Compare commits
680 Commits
0.1.0-pre
...
issue-160-
| Author | SHA1 | Date | |
|---|---|---|---|
|
24880616a0
|
|||
|
b862ca52a9
|
|||
|
59e1eee8ce
|
|||
|
57c8ffda35
|
|||
|
171ebd213d
|
|||
|
2a240cbd09
|
|||
|
068fa3454c
|
|||
|
6b2bcbf2ce
|
|||
|
19e46dac18
|
|||
|
e96c863120
|
|||
|
a30b3c84d7
|
|||
|
0ad5dc4583
|
|||
|
7189998f3b
|
|||
|
b6b24b8e0b
|
|||
|
8e820c6e89
|
|||
|
c74784a37c
|
|||
|
3d940cffcf
|
|||
|
0df044b284
|
|||
|
f576b9fb20
|
|||
|
5e00264119
|
|||
|
|
05eeba32fa | ||
|
|
61019ecd49 | ||
|
|
bed06d1334 | ||
|
|
f09f4bd1b7 | ||
|
|
a3b11f21bb | ||
|
|
69a461d9c3 | ||
|
|
1dfe5cfecf | ||
|
|
8e4550657e | ||
|
|
aee7fa52c3 | ||
|
|
d166cc84a1 | ||
|
|
bb7229d224 | ||
|
|
708cd5ead9 | ||
|
|
f7986cb4da | ||
|
|
395aeb415d | ||
|
|
830fb70492 | ||
|
|
6379a26afb | ||
|
|
2277013c76 | ||
|
|
8934e0e6bd | ||
|
|
59519febbc | ||
|
|
46fcdd356c | ||
|
|
9f343c45e8 | ||
|
|
931904f388 | ||
|
|
a40d0cbb5c | ||
|
|
9f5df9db10 | ||
|
|
d26ddf7015 | ||
|
|
9515065407 | ||
|
|
82a8c61cf6 | ||
|
|
3fae516ce4 | ||
|
|
33eaa765d7 | ||
|
|
3b3dde8413 | ||
|
|
118a2744fe | ||
|
|
2e3dceecf8 | ||
|
|
07fb04bb74 | ||
|
|
8a1dbe9dbb | ||
|
|
4ef3622616 | ||
|
|
82a704ab44 | ||
|
|
0cb22945fe | ||
|
|
d09adf9159 | ||
|
|
0b959c56fb | ||
|
|
ec29332657 | ||
|
|
0f6381e67b | ||
|
|
877b55e21d | ||
|
|
fa11ca665f | ||
|
d9d196439f
|
|||
|
a34fc4ad4f
|
|||
|
b4d52b88c1
|
|||
|
3fc3d27361
|
|||
|
56b86add38
|
|||
|
a608a105e3
|
|||
|
2e3e413f6c
|
|||
|
dfb6c09133
|
|||
|
9636276c17
|
|||
|
41783ff027
|
|||
|
08b0ecd057
|
|||
|
7e31798446
|
|||
|
534afa6e8d
|
|||
|
b56c44a210
|
|||
|
ef0c94fddd
|
|||
|
f14c281841
|
|||
|
b40cefee35
|
|||
|
ff32ccfb50
|
|||
|
581108ab65
|
|||
|
54e8e3efb0
|
|||
|
4dcc63606e
|
|||
|
a396b6044d
|
|||
|
94e5d2e19f
|
|||
|
a0976eee70
|
|||
|
61494d8c4b
|
|||
|
2b34c2dd69
|
|||
|
afc30b87dd
|
|||
|
|
ed0a63eb0c | ||
|
9ba590dd90
|
|||
|
d4bffd2c4a
|
|||
|
650f0a3e4e
|
|||
|
fd6ccf8f0a
|
|||
|
d9fe4b8723
|
|||
|
da2e7e0411
|
|||
|
79d6a50e44
|
|||
|
a13a5e5d20
|
|||
|
82743dda2b
|
|||
|
|
6f80dd1fcc | ||
|
1325dce493
|
|||
|
ac21c19b7e
|
|||
|
2b4088d068
|
|||
|
d86dc9b1d7
|
|||
|
9982311c87
|
|||
|
40c88d0b66
|
|||
|
e0ee1c2f94
|
|||
|
b4fa2780eb
|
|||
|
df86183e97
|
|||
|
f7868dc646
|
|||
|
e742be7693
|
|||
|
924bc8698d
|
|||
|
5214c35b20
|
|||
|
700e04535a
|
|||
|
fedc0bbef6
|
|||
|
468fc5ade9
|
|||
|
2c077db36b
|
|||
|
f80638bba4
|
|||
|
860aa0dafd
|
|||
|
27510b3b51
|
|||
|
96bcbbeeec
|
|||
|
8a632eb3b4
|
|||
|
aa992c0e5d
|
|||
|
810870e3a5
|
|||
|
d584e7b21b
|
|||
|
e93ac62c81
|
|||
|
0d7d6c8382
|
|||
|
5cd9ce8835
|
|||
|
443522d526
|
|||
|
9061e60518
|
|||
|
d65ab434b2
|
|||
|
cff592db82
|
|||
|
97029e8102
|
|||
|
|
828fd9eb10 | ||
|
03800d3b74
|
|||
|
a47304e599
|
|||
|
7b050e9fe2
|
|||
|
687a1d0c88
|
|||
|
e09e3c264d
|
|||
|
b56431b4e3
|
|||
|
70ad50010d
|
|||
|
|
ca3a249bea | ||
|
|
4337cdc38d | ||
|
9f92e0bc86
|
|||
|
98751cf8fb
|
|||
|
2f62067d96
|
|||
|
2cb1554244
|
|||
|
6f3c143228
|
|||
|
9793fc6888
|
|||
|
043cab08ae
|
|||
|
b7c83780da
|
|||
|
cff11135ff
|
|||
|
b94a4123eb
|
|||
|
8ef1c8b5d4
|
|||
|
132d331e7c
|
|||
|
734916728c
|
|||
|
5f6ed1292d
|
|||
|
a7dc03af50
|
|||
|
5a86a28d67
|
|||
|
a905c6322c
|
|||
|
49ccadd470
|
|||
|
9f0ac0ee19
|
|||
|
7e0f839ff8
|
|||
|
1e9ee260e7
|
|||
|
0b7d447aaf
|
|||
|
16a9336d31
|
|||
|
7d13836fea
|
|||
|
b645c4d57e
|
|||
|
5db43cd908
|
|||
|
93cd421ca3
|
|||
|
ec7130dac6
|
|||
|
f2b8cc530c
|
|||
|
de765088d1
|
|||
|
|
e11188aa99 | ||
|
0c6699c3c6
|
|||
|
c5858be6b8
|
|||
|
ffe00c7b1f
|
|||
|
43114959fd
|
|||
|
b1c3ffd729
|
|||
|
4f1a9e95a2
|
|||
|
f6a4f55384
|
|||
|
672b179446
|
|||
|
3111387e5a
|
|||
|
f8fda269f1
|
|||
|
fbac593739
|
|||
|
c80e54dff5
|
|||
|
4a86b2479e
|
|||
|
7bb2552d0f
|
|||
|
ad21640cd9
|
|||
|
e4aad4e645
|
|||
|
adf44ba141
|
|||
|
8707b194fd
|
|||
|
77bb7c4aba
|
|||
|
7383fdd0c0
|
|||
|
f4201d946a
|
|||
|
d5b5f1fddd
|
|||
| 910d660732 | |||
| ab4ad3a1bb | |||
| 13b27101f1 | |||
| 44b5d0dcff | |||
| df54b3711f | |||
| 197288453f | |||
| 00d5691a13 | |||
| 561b6684ce | |||
| a13bb3154d | |||
| fd9d3faf9f | |||
| f6cc467e95 | |||
| ef978c1230 | |||
| 52af598473 | |||
| 2acacd46c2 | |||
| 4b2f9ddb40 | |||
| 0d3dc4eba2 | |||
| 824faa8091 | |||
| d10cdfdf57 | |||
| 5e911793ce | |||
| 8c87c9aeb7 | |||
| 9da5998a5c | |||
| 08943dadca | |||
| 7fac7226c5 | |||
| 9686c8ebf9 | |||
| 4bf96d0ec9 | |||
| 34add82bee | |||
| f46e7e8c4b | |||
| 3baf254251 | |||
| 10ca9ea827 | |||
| 4a50c8ecb7 | |||
| 47d9766c78 | |||
| 45ab69960f | |||
| d3505d4ee6 | |||
| 9297d1a2f8 | |||
| 56feb7c09e | |||
| bede4b8712 | |||
| 95c1c55f22 | |||
| f547a6eb68 | |||
|
|
453a29fdf7 | ||
|
|
1a5f0259f4 | ||
|
|
d6fa61e223 | ||
| eab82b5d63 | |||
| c455b521a9 | |||
| b4f9e12293 | |||
| bbd353ea3a | |||
| 4189c5de69 | |||
| dee3218723 | |||
| 3c803a9f58 | |||
| a9b0c0fbc9 | |||
| 38b6c918f9 | |||
| 6e584c96c4 | |||
| 20338f7d14 | |||
| 1a995a5d63 | |||
| f964382175 | |||
| 0c7f60fae6 | |||
| 413e63d1ca | |||
| 8b000f4e48 | |||
| b0522507be | |||
| d4bcf7021e | |||
| 48cf0b1f67 | |||
| d82e189c01 | |||
| e5a60d1b9a | |||
| ff067351cb | |||
| 345712a617 | |||
| 118dac6907 | |||
| 5ca40caf81 | |||
| d858187fd4 | |||
| bd65517df1 | |||
| b1acad6c95 | |||
| a8333281ac | |||
| 2fdb08ac00 | |||
|
|
bd4e5a2314 | ||
| 34ed317b6b | |||
|
|
14661502ab | ||
| 097754ffdf | |||
|
|
f26ec6d295 | ||
| 858d430845 | |||
| 5134eccbf8 | |||
| 28b4737758 | |||
|
|
5c43ff4c9e | ||
|
|
53db68e39f | ||
| 9e628e34dd | |||
|
|
62d5d53232 | ||
|
|
56569a0698 | ||
|
|
ef44f818d0 | ||
|
|
8944ed6e36 | ||
| 51805b27aa | |||
| 0ec64510b3 | |||
| 20152443da | |||
|
|
7c8929fe9f | ||
| 5617516c93 | |||
| f6fe08367d | |||
| a5f02133e2 | |||
|
|
8ed9b4432d | ||
|
|
db1d05e8ad | ||
| eae58137c8 | |||
| b0f90c096f | |||
| e8361c564a | |||
|
|
54db9c9a92 | ||
|
|
73db341dc8 | ||
| 5fd30b412b | |||
|
|
bbe2e87640 | ||
|
|
67f59f6895 | ||
|
|
3e2df2e111 | ||
| 824d2149c6 | |||
|
|
c86dbe043b | ||
|
|
8043ac7f51 | ||
|
|
ead9d31647 | ||
|
|
a08e624309 | ||
| b20371c3ac | |||
| 0589a7cbcc | |||
| cf48961063 | |||
| 6046582b9c | |||
| 82aa6c70ea | |||
| e829bd8235 | |||
| 66f989e691 | |||
| eebb91fbb0 | |||
| 1d3e88bdfe | |||
| fbb03dee7e | |||
|
|
88e5afb70f | ||
| 67eabfd3af | |||
| cd1dd8c29e | |||
| 08ddb591b7 | |||
| 3e841b3c68 | |||
| 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 | |||
| 90af68b211 | |||
| 80603662b4 | |||
| d2c5d4dfd9 | |||
| 6f1b8b4041 | |||
| f63b2bf744 | |||
| 71cb75c170 | |||
| dac64f5718 | |||
| 27b2d2ac7d | |||
| 47142dd376 | |||
| d071a7e51b | |||
| 5c45884f5f | |||
| cafedd73a2 | |||
| 7163b77837 | |||
| 122c54b51e | |||
| b9d7d7d007 | |||
| 9050c9792a | |||
| aac8f760ad | |||
| 7d334c18f5 | |||
| 86b0e4b31b | |||
| af811f3dbc | |||
| d30d2ac8a5 | |||
| 86a4b10de6 | |||
| cfa7049ab9 | |||
| 391676e90a | |||
| 34e4ece8b5 | |||
| cf6443d83f | |||
| 846cf92fa4 | |||
| ab568901f8 | |||
| bfda95c0d6 | |||
| fb1875ee5b | |||
| 185d4f869b | |||
| 2ac8b61aa8 | |||
| 8739cb4656 | |||
| 826900cc41 | |||
| ec6bbdbf06 | |||
| 15a188c501 | |||
| b5440fc7d2 | |||
| 4b21adadf1 | |||
| 78ae77780b | |||
| ccb95bcbee | |||
| 21ac670bbe | |||
| 8b54dee66c | |||
| dad926f3ba | |||
| a298d949b5 | |||
| e1cf11f9d4 | |||
| a046f16308 | |||
| 97cd43792d | |||
| 08693e6d3a | |||
| e2227da8d2 | |||
| 5f20e4c583 | |||
| f82f1a12dd | |||
| 53148fd1c9 | |||
| 8985101b2a | |||
| d1949c8490 | |||
| b7faae1203 | |||
| b6a9d35c3e | |||
| 6cb6c7a448 | |||
| 22a5ad739e | |||
| 14c91bdd78 | |||
| 2c638cd2e2 | |||
| 9e59f484e3 | |||
| ac8419ecb2 | |||
| 3ecdb63063 | |||
| cfe24428fa | |||
| 4c4266dd8c | |||
| e8336bbc8a | |||
| 0f69c73e0e | |||
| e348de8dc4 | |||
| 55a3ba9be2 | |||
| 51b29b81b0 | |||
| 3c2e0334b7 | |||
| 0679626514 | |||
| 5035051135 | |||
| 63c70ee74b | |||
| 2e0bbca2e0 | |||
| b52fa23ca2 | |||
| ba03b78f23 | |||
| 04ef472c15 | |||
| 75cd8f2341 | |||
| f2e26c1800 | |||
| 0f7dd597d2 | |||
| fb0eba9201 | |||
| 3c80929c38 | |||
| b184ee835f | |||
| ef8e3bd940 | |||
| 1a64527e14 | |||
| 30b4d399b9 | |||
| 50424c2801 | |||
| 7e7c357e47 | |||
| 531b82a406 | |||
| 146ac38549 | |||
| c481956b07 | |||
| 8ef19f0825 | |||
| c1e29a8f16 | |||
| c3611eec6a | |||
| 74b58db7d1 | |||
| 9e4763c640 | |||
| abc4278fc8 | |||
| 8c4cde3d14 | |||
| 3824f6417a | |||
|
|
2be1aa2707 | ||
| da94fa5f92 | |||
| 35bf9b5ff2 | |||
| bed2cca8d2 | |||
| 9717a1c00f | |||
| 3ddc719d8a | |||
| 4b89810892 | |||
| b82367838d | |||
| dd7556ba21 | |||
| f4b6bfc594 | |||
| f9251589cd | |||
| 2de549862a | |||
| c502f70f68 | |||
| cbf076740a | |||
| 86c144b285 | |||
| 7ec6e8604c | |||
| de70f4820f | |||
|
|
febe6fcb35 | ||
|
|
3055529d4c | ||
|
|
d276bfb3ec | ||
| 9db0664465 | |||
| e9c727647a | |||
| 55eef8a3d3 | |||
| d07ad3eb26 | |||
|
|
ad53b141c7 | ||
|
|
23c13a07a9 | ||
|
|
a186b07763 | ||
|
|
1ca628aba1 | ||
| 8f4ef48891 | |||
|
|
d852ab3415 | ||
| a1bcc4b51f | |||
| be93a98bd4 | |||
| 85054d9c76 | |||
| 6c95218daf | |||
| ede6299681 | |||
| bf6e94cfb2 | |||
| 378942cbce | |||
| 1207c7b3dd | |||
| 788883df7b | |||
| 5ab1f6c203 | |||
| 175066780b | |||
| a629cf7b9f | |||
| 684130bfa3 | |||
| ee34d85dbc | |||
| d2b280da2d | |||
| a5593a725f | |||
| 5e35a91ef0 | |||
| 4abd10d9c9 | |||
| 0bbac877bd | |||
| f558bd9932 | |||
| 781c28b03c | |||
| a8be2efd85 | |||
| f46700e1cc | |||
| d7a6935a1a | |||
| a1282b2854 | |||
| 34b9ea7d20 | |||
| 0ff7ebb1fd | |||
| f83dcbc430 | |||
| 56e4a6b15f | |||
| ee9b2ec30d | |||
| 640cf1e2c1 | |||
| 56c439d716 | |||
| 1ed6e49a81 | |||
| fad9f83e6a | |||
| 2e28b0d00f | |||
| ed4ff15f96 | |||
| 1d623723a2 | |||
| 931080244f | |||
| 27e2e7f848 | |||
| 8b638c7ecb | |||
| acd370611f | |||
| e1b5a89cee | |||
| 5edebd57d9 | |||
| bcaccaaf31 | |||
| 818a5d2d85 | |||
| 13acce07d4 | |||
| 4ed5e21b7f | |||
| 86aab6bb59 | |||
| 7f5cb64b18 | |||
| 6c12eb16eb | |||
| e637f90fae | |||
| 5b33c3f491 | |||
| 1842ed464f | |||
| 296bbdd561 | |||
| 27ead1be7c | |||
| 5184609dba | |||
| 5d94d0bf75 | |||
| 72bcfa9270 | |||
| fafff9dadd | |||
| e3c20d53a8 | |||
| 8b7dc68491 | |||
| 7742fe08b5 | |||
| a773da037c | |||
| dfeb814dcc | |||
| 0623c7b1b1 | |||
| 62005f83a4 | |||
| eaafd77a7e | |||
| 9d9e415a09 | |||
| 6c1ae585b7 | |||
| 793aad7b6c | |||
| fd7807a66e | |||
| 879bd061dd | |||
| 75632b2cf1 | |||
| b65b9dc5e1 | |||
| 31d70e34e9 | |||
| 84de282655 | |||
| 997dcadf89 | |||
| b2312629ce | |||
| 3d10f964c6 | |||
| 404038edcb | |||
| ea4f9ceab1 | |||
| 5c481ea94e | |||
| 1ccaf4ba91 | |||
| b532511cd5 | |||
| b3105b439c | |||
| 2b6cb5f1a8 | |||
| f4242b10e7 | |||
| ad4d185ead | |||
| b18aafe2c4 | |||
| 340196bf9d | |||
| 883226aa70 | |||
| 0d393612a7 | |||
| 5635f6cc4e | |||
| a7fd36beeb | |||
| baee1d5b85 | |||
| 68df6b8e50 | |||
| ac73090784 | |||
| faf4f3b7ca | |||
| d888d11d59 | |||
| 28a1077833 | |||
| c40b9dbc0b | |||
| 6bbd262818 | |||
| 78d36bce24 | |||
| aedfc19220 | |||
| 2f34fc7bef | |||
| de66b92631 | |||
| fee3984bf7 | |||
| b953c8fd30 | |||
| 24e4c3a19b | |||
| d2efb504b9 | |||
| df9dd0e785 | |||
| 89c9699158 | |||
| 124ddcdfeb | |||
| 5c0a0fc155 | |||
| b11b74d2b4 | |||
| 5ac8f5b651 | |||
| 9032df97cf | |||
| 14e1077ad1 | |||
| b5648bdd6b | |||
| e7cd952970 | |||
| 1455c2c175 | |||
| c106dd3f65 | |||
| f6725fbf5f | |||
| c706a047ea | |||
| 9602db31ab | |||
| c2c47e1b7e | |||
| 34386680cc | |||
| 16a26d9881 | |||
| 3496f24f6e | |||
| 1a5876a074 | |||
| c782bc44de | |||
| f78e7b1cbc |
13
.gitignore
vendored
13
.gitignore
vendored
@@ -1,2 +1,15 @@
|
|||||||
|
.ghci
|
||||||
|
.vim
|
||||||
|
codex.tags
|
||||||
dist-newstyle/
|
dist-newstyle/
|
||||||
|
cabal.project.local
|
||||||
.stack-work/
|
.stack-work/
|
||||||
|
bin/
|
||||||
|
/*.prof
|
||||||
|
/*.ps
|
||||||
|
/*.hp
|
||||||
|
tags
|
||||||
|
TAGS
|
||||||
|
/tmp/
|
||||||
|
.entangled
|
||||||
|
release/
|
||||||
|
|||||||
500
.gitlab-ci.yml
Normal file
500
.gitlab-ci.yml
Normal file
@@ -0,0 +1,500 @@
|
|||||||
|
stages:
|
||||||
|
- hlint
|
||||||
|
- test
|
||||||
|
- release
|
||||||
|
|
||||||
|
variables:
|
||||||
|
GIT_SSL_NO_VERIFY: "1"
|
||||||
|
|
||||||
|
# Commit of ghc/ci-images repository from which to pull Docker images
|
||||||
|
DOCKER_REV: 8d0224e6b2a08157649651e69302380b2bd24e11
|
||||||
|
|
||||||
|
############################################################
|
||||||
|
# CI Step
|
||||||
|
############################################################
|
||||||
|
|
||||||
|
.debian:
|
||||||
|
image: "registry.gitlab.haskell.org/ghc/ci-images/x86_64-linux-deb9:$DOCKER_REV"
|
||||||
|
tags:
|
||||||
|
- x86_64-linux
|
||||||
|
variables:
|
||||||
|
OS: "LINUX"
|
||||||
|
ARCH: "64"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.alpine:64bit:
|
||||||
|
image: "alpine:3.12"
|
||||||
|
tags:
|
||||||
|
- x86_64-linux
|
||||||
|
variables:
|
||||||
|
OS: "LINUX"
|
||||||
|
ARCH: "64"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.alpine:32bit:
|
||||||
|
image: "i386/alpine:3.12"
|
||||||
|
tags:
|
||||||
|
- x86_64-linux
|
||||||
|
variables:
|
||||||
|
OS: "LINUX"
|
||||||
|
ARCH: "32"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.linux:armv7:
|
||||||
|
image: "registry.gitlab.haskell.org/ghc/ci-images/armv7-linux-deb10:$DOCKER_REV"
|
||||||
|
tags:
|
||||||
|
- armv7-linux
|
||||||
|
variables:
|
||||||
|
OS: "LINUX"
|
||||||
|
ARCH: "ARM"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.linux:aarch64:
|
||||||
|
image: "registry.gitlab.haskell.org/ghc/ci-images/aarch64-linux-deb10:$DOCKER_REV"
|
||||||
|
tags:
|
||||||
|
- aarch64-linux
|
||||||
|
variables:
|
||||||
|
OS: "LINUX"
|
||||||
|
ARCH: "ARM64"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.darwin:
|
||||||
|
tags:
|
||||||
|
- x86_64-darwin
|
||||||
|
variables:
|
||||||
|
OS: "DARWIN"
|
||||||
|
ARCH: "64"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.darwin:aarch64:
|
||||||
|
tags:
|
||||||
|
- aarch64-darwin-m1
|
||||||
|
variables:
|
||||||
|
OS: "DARWIN"
|
||||||
|
ARCH: "ARM64"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.freebsd:
|
||||||
|
tags:
|
||||||
|
- x86_64-freebsd
|
||||||
|
variables:
|
||||||
|
OS: "FREEBSD"
|
||||||
|
ARCH: "64"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.windows:
|
||||||
|
tags:
|
||||||
|
- new-x86_64-windows
|
||||||
|
variables:
|
||||||
|
OS: "WINDOWS"
|
||||||
|
ARCH: "64"
|
||||||
|
CABAL_DIR: "$CI_PROJECT_DIR/cabal"
|
||||||
|
|
||||||
|
.root_cleanup:
|
||||||
|
after_script:
|
||||||
|
- bash ./.gitlab/after_script.sh
|
||||||
|
|
||||||
|
.test_ghcup_version:
|
||||||
|
script:
|
||||||
|
- bash ./.gitlab/script/ghcup_version.sh
|
||||||
|
variables:
|
||||||
|
JSON_VERSION: "0.0.5"
|
||||||
|
artifacts:
|
||||||
|
expire_in: 2 week
|
||||||
|
paths:
|
||||||
|
- golden
|
||||||
|
when: on_failure
|
||||||
|
|
||||||
|
.test_ghcup_version:linux:
|
||||||
|
extends:
|
||||||
|
- .test_ghcup_version
|
||||||
|
- .debian
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps.sh
|
||||||
|
|
||||||
|
.test_ghcup_version:linux32:
|
||||||
|
extends:
|
||||||
|
- .test_ghcup_version
|
||||||
|
- .alpine:32bit
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/alpine/install_deps.sh
|
||||||
|
|
||||||
|
.test_ghcup_version:armv7:
|
||||||
|
extends:
|
||||||
|
- .test_ghcup_version
|
||||||
|
- .linux:armv7
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps.sh
|
||||||
|
|
||||||
|
.test_ghcup_version:aarch64:
|
||||||
|
extends:
|
||||||
|
- .test_ghcup_version
|
||||||
|
- .linux:aarch64
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps.sh
|
||||||
|
|
||||||
|
.test_ghcup_version:darwin:
|
||||||
|
extends:
|
||||||
|
- .test_ghcup_version
|
||||||
|
- .darwin
|
||||||
|
- .root_cleanup
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/darwin/install_deps.sh
|
||||||
|
|
||||||
|
.test_ghcup_version:darwin:aarch64:
|
||||||
|
extends:
|
||||||
|
- .test_ghcup_version
|
||||||
|
- .darwin:aarch64
|
||||||
|
- .root_cleanup
|
||||||
|
script: |
|
||||||
|
set -Eeuo pipefail
|
||||||
|
function runInNixShell() {
|
||||||
|
time nix-shell $CI_PROJECT_DIR/.gitlab/shell.nix \
|
||||||
|
-I nixpkgs=https://github.com/angerman/nixpkgs/archive/75f7281738b.tar.gz \
|
||||||
|
--argstr system "aarch64-darwin" \
|
||||||
|
--pure \
|
||||||
|
--keep CI_PROJECT_DIR \
|
||||||
|
--keep MACOSX_DEPLOYMENT_TARGET \
|
||||||
|
--keep JSON_VERSION \
|
||||||
|
--keep ARTIFACT \
|
||||||
|
--keep OS \
|
||||||
|
--keep ARCH \
|
||||||
|
--keep CABAL_DIR \
|
||||||
|
--keep GHC_VERSION \
|
||||||
|
--keep CABAL_VERSION \
|
||||||
|
--run "$1" 2>&1
|
||||||
|
}
|
||||||
|
runInNixShell ./.gitlab/before_script/darwin/install_deps.sh 2>&1
|
||||||
|
runInNixShell ./.gitlab/script/ghcup_version.sh 2>&1
|
||||||
|
|
||||||
|
.test_ghcup_version:freebsd:
|
||||||
|
extends:
|
||||||
|
- .test_ghcup_version
|
||||||
|
- .freebsd
|
||||||
|
- .root_cleanup
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/freebsd/install_deps.sh
|
||||||
|
|
||||||
|
.test_ghcup_version:windows:
|
||||||
|
extends:
|
||||||
|
- .test_ghcup_version
|
||||||
|
- .windows
|
||||||
|
- .root_cleanup
|
||||||
|
before_script:
|
||||||
|
- set CABAL_DIR="$CI_PROJECT_DIR/cabal"
|
||||||
|
- bash ./.gitlab/before_script/windows/install_deps.sh
|
||||||
|
|
||||||
|
.release_ghcup:
|
||||||
|
script:
|
||||||
|
- bash ./.gitlab/script/ghcup_release.sh
|
||||||
|
artifacts:
|
||||||
|
expire_in: 2 week
|
||||||
|
paths:
|
||||||
|
- out
|
||||||
|
only:
|
||||||
|
- tags
|
||||||
|
variables:
|
||||||
|
JSON_VERSION: "0.0.5"
|
||||||
|
|
||||||
|
######## stack test ########
|
||||||
|
|
||||||
|
test:linux:stack:
|
||||||
|
stage: test
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps_minimal.sh
|
||||||
|
script:
|
||||||
|
- ./.gitlab/script/ghcup_stack.sh
|
||||||
|
extends:
|
||||||
|
- .debian
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
######## bootstrap test ########
|
||||||
|
|
||||||
|
test:linux:bootstrap_script:
|
||||||
|
stage: test
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps_minimal.sh
|
||||||
|
script:
|
||||||
|
- ./.gitlab/script/ghcup_bootstrap.sh
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
extends:
|
||||||
|
- .debian
|
||||||
|
- .root_cleanup
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
test:windows:bootstrap_powershell_script:
|
||||||
|
stage: test
|
||||||
|
script:
|
||||||
|
- ./bootstrap-haskell.ps1 -InstallDir $CI_PROJECT_DIR -BootstrapUrl $CI_PROJECT_DIR/bootstrap-haskell -InBash
|
||||||
|
after_script:
|
||||||
|
- "[Environment]::SetEnvironmentVariable('GHCUP_INSTALL_BASE_PREFIX', $null, [System.EnvironmentVariableTarget]::User)"
|
||||||
|
- "[Environment]::SetEnvironmentVariable('GHCUP_MSYS2', $null, [System.EnvironmentVariableTarget]::User)"
|
||||||
|
- "[Environment]::SetEnvironmentVariable('CABAL_DIR', $null, [System.EnvironmentVariableTarget]::User)"
|
||||||
|
- bash ./.gitlab/after_script.sh
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
extends:
|
||||||
|
- .windows
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
######## linux test ########
|
||||||
|
|
||||||
|
test:linux:recommended:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:linux
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
test:linux:latest:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:linux
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "9.0.1"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
######## linux 32bit test ########
|
||||||
|
|
||||||
|
test:linux:recommended:32bit:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:linux32
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.2.0.0"
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
######## arm tests ########
|
||||||
|
|
||||||
|
test:linux:recommended:armv7:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:armv7
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.4"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
when: manual
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
test:linux:recommended:aarch64:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:aarch64
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.4"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
when: manual
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
######## darwin test ########
|
||||||
|
|
||||||
|
test:mac:recommended:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:darwin
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.4"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
test:mac:latest:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:darwin
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "9.0.1"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
test:mac:recommended:aarch64:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:darwin:aarch64
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
needs: []
|
||||||
|
allow_failure: true
|
||||||
|
|
||||||
|
|
||||||
|
######## freebsd test ########
|
||||||
|
|
||||||
|
test:freebsd:recommended:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:freebsd
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.4"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
allow_failure: true # freebsd runners are unreliable
|
||||||
|
when: manual
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
######## windows test ########
|
||||||
|
|
||||||
|
test:windows:recommended:
|
||||||
|
stage: test
|
||||||
|
extends: .test_ghcup_version:windows
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
needs: []
|
||||||
|
|
||||||
|
######## linux release ########
|
||||||
|
|
||||||
|
release:linux:64bit:
|
||||||
|
stage: release
|
||||||
|
needs: ["test:linux:recommended", "test:linux:latest"]
|
||||||
|
extends:
|
||||||
|
- .alpine:64bit
|
||||||
|
- .release_ghcup
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/alpine/install_deps.sh
|
||||||
|
variables:
|
||||||
|
ARTIFACT: "x86_64-linux-ghcup"
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
|
||||||
|
|
||||||
|
release:linux:32bit:
|
||||||
|
stage: release
|
||||||
|
needs: ["test:linux:recommended:32bit"]
|
||||||
|
extends:
|
||||||
|
- .alpine:32bit
|
||||||
|
- .release_ghcup
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/alpine/install_deps.sh
|
||||||
|
variables:
|
||||||
|
ARTIFACT: "i386-linux-ghcup"
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.2.0.0"
|
||||||
|
|
||||||
|
release:linux:armv7:
|
||||||
|
stage: release
|
||||||
|
needs: ["test:linux:recommended:armv7"]
|
||||||
|
extends:
|
||||||
|
- .linux:armv7
|
||||||
|
- .release_ghcup
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps.sh
|
||||||
|
variables:
|
||||||
|
ARTIFACT: "armv7-linux-ghcup"
|
||||||
|
GHC_VERSION: "8.10.4"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
|
||||||
|
release:linux:aarch64:
|
||||||
|
stage: release
|
||||||
|
needs: ["test:linux:recommended:aarch64"]
|
||||||
|
extends:
|
||||||
|
- .linux:aarch64
|
||||||
|
- .release_ghcup
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/install_deps.sh
|
||||||
|
variables:
|
||||||
|
ARTIFACT: "aarch64-linux-ghcup"
|
||||||
|
GHC_VERSION: "8.10.4"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
|
||||||
|
######## darwin release ########
|
||||||
|
|
||||||
|
release:darwin:
|
||||||
|
stage: release
|
||||||
|
needs: ["test:mac:recommended", "test:mac:latest"]
|
||||||
|
extends:
|
||||||
|
- .darwin
|
||||||
|
- .release_ghcup
|
||||||
|
- .root_cleanup
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/darwin/install_deps.sh
|
||||||
|
variables:
|
||||||
|
ARTIFACT: "x86_64-apple-darwin-ghcup"
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
MACOSX_DEPLOYMENT_TARGET: "10.7"
|
||||||
|
|
||||||
|
release:darwin:aarch64:
|
||||||
|
stage: release
|
||||||
|
needs: ["test:mac:recommended:aarch64"]
|
||||||
|
extends:
|
||||||
|
- .darwin:aarch64
|
||||||
|
- .release_ghcup
|
||||||
|
- .root_cleanup
|
||||||
|
script: |
|
||||||
|
set -Eeuo pipefail
|
||||||
|
function runInNixShell() {
|
||||||
|
time nix-shell .gitlab/shell.nix \
|
||||||
|
-I nixpkgs=https://github.com/angerman/nixpkgs/archive/75f7281738b.tar.gz \
|
||||||
|
--argstr system "aarch64-darwin" \
|
||||||
|
--pure \
|
||||||
|
--keep CI_PROJECT_DIR \
|
||||||
|
--keep MACOSX_DEPLOYMENT_TARGET \
|
||||||
|
--keep JSON_VERSION \
|
||||||
|
--keep ARTIFACT \
|
||||||
|
--keep OS \
|
||||||
|
--keep ARCH \
|
||||||
|
--keep CABAL_DIR \
|
||||||
|
--keep GHC_VERSION \
|
||||||
|
--keep CABAL_VERSION \
|
||||||
|
--run "$1" 2>&1
|
||||||
|
}
|
||||||
|
runInNixShell ./.gitlab/before_script/darwin/install_deps.sh 2>&1
|
||||||
|
runInNixShell ./.gitlab/script/ghcup_release.sh 2>&1
|
||||||
|
variables:
|
||||||
|
ARTIFACT: "aarch64-apple-darwin-ghcup"
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
MACOSX_DEPLOYMENT_TARGET: "10.7"
|
||||||
|
allow_failure: true
|
||||||
|
|
||||||
|
|
||||||
|
######## freebsd release ########
|
||||||
|
|
||||||
|
release:freebsd:
|
||||||
|
stage: release
|
||||||
|
needs: ["test:freebsd:recommended"]
|
||||||
|
extends:
|
||||||
|
- .freebsd
|
||||||
|
- .release_ghcup
|
||||||
|
- .root_cleanup
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/freebsd/install_deps.sh
|
||||||
|
variables:
|
||||||
|
ARTIFACT: "x86_64-portbld-freebsd-ghcup"
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
allow_failure: true
|
||||||
|
|
||||||
|
######## windows release ########
|
||||||
|
|
||||||
|
release:windows:
|
||||||
|
stage: release
|
||||||
|
needs: ["test:windows:recommended"]
|
||||||
|
extends:
|
||||||
|
- .windows
|
||||||
|
- .release_ghcup
|
||||||
|
- .root_cleanup
|
||||||
|
before_script:
|
||||||
|
- bash ./.gitlab/before_script/windows/install_deps.sh
|
||||||
|
variables:
|
||||||
|
ARTIFACT: "x86_64-mingw64-ghcup"
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
|
||||||
|
######## hlint ########
|
||||||
|
|
||||||
|
hlint:
|
||||||
|
stage: hlint
|
||||||
|
extends:
|
||||||
|
- .alpine:64bit
|
||||||
|
before_script:
|
||||||
|
- ./.gitlab/before_script/linux/alpine/install_deps.sh
|
||||||
|
script:
|
||||||
|
- ./.gitlab/script/hlint.sh
|
||||||
|
variables:
|
||||||
|
GHC_VERSION: "8.10.5"
|
||||||
|
CABAL_VERSION: "3.4.0.0"
|
||||||
|
JSON_VERSION: "0.0.4"
|
||||||
|
allow_failure: true
|
||||||
|
artifacts:
|
||||||
|
expire_in: 2 week
|
||||||
|
paths:
|
||||||
|
- report.html
|
||||||
|
when: on_failure
|
||||||
15
.gitlab/after_script.sh
Normal file
15
.gitlab/after_script.sh
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
BUILD_DIR=$CI_PROJECT_DIR
|
||||||
|
echo "Cleaning $BUILD_DIR"
|
||||||
|
cd $HOME
|
||||||
|
test -n "$BUILD_DIR"
|
||||||
|
shopt -s extglob
|
||||||
|
rm -Rf "$BUILD_DIR"/!(out)
|
||||||
|
if [ "${OS}" = "WINDOWS" ] ; then
|
||||||
|
rm -Rf /c/ghcup
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit 0
|
||||||
32
.gitlab/before_script/darwin/install_deps.sh
Executable file
32
.gitlab/before_script/darwin/install_deps.sh
Executable file
@@ -0,0 +1,32 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "${TMPDIR}"
|
||||||
|
|
||||||
|
if [ $ARCH = 'ARM64' ] ; then
|
||||||
|
curl -sSfL https://downloads.haskell.org/~ghcup/0.1.15.1/aarch64-apple-darwin-ghcup-0.1.15.1 > ./ghcup-bin
|
||||||
|
chmod +x ghcup-bin
|
||||||
|
else
|
||||||
|
curl -sSfL https://downloads.haskell.org/~ghcup/x86_64-apple-darwin-ghcup > ./ghcup-bin
|
||||||
|
chmod +x ghcup-bin
|
||||||
|
./ghcup-bin upgrade -i -f
|
||||||
|
fi
|
||||||
|
|
||||||
|
./ghcup-bin install ${GHC_VERSION}
|
||||||
|
./ghcup-bin set ${GHC_VERSION}
|
||||||
|
./ghcup-bin install-cabal ${CABAL_VERSION}
|
||||||
|
|
||||||
|
if [ $ARCH = 'ARM64' ] ; then
|
||||||
|
cabal update
|
||||||
|
mkdir vendored
|
||||||
|
cd vendored
|
||||||
|
cabal unpack network-3.1.2.1
|
||||||
|
cd network*
|
||||||
|
autoreconf -fi
|
||||||
|
cd ../..
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit 0
|
||||||
19
.gitlab/before_script/freebsd/install_deps.sh
Executable file
19
.gitlab/before_script/freebsd/install_deps.sh
Executable file
@@ -0,0 +1,19 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
# pkg install --force --yes --no-repo-update curl gcc gmp gmake ncurses perl5 libffi libiconv
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "${TMPDIR}"
|
||||||
|
|
||||||
|
curl -sSfL https://downloads.haskell.org/~ghcup/x86_64-portbld-freebsd-ghcup > ./ghcup-bin
|
||||||
|
chmod +x ghcup-bin
|
||||||
|
|
||||||
|
./ghcup-bin upgrade -i -f
|
||||||
|
./ghcup-bin install ${GHC_VERSION}
|
||||||
|
./ghcup-bin set ${GHC_VERSION}
|
||||||
|
./ghcup-bin install-cabal ${CABAL_VERSION}
|
||||||
|
|
||||||
|
exit 0
|
||||||
54
.gitlab/before_script/linux/alpine/install_deps.sh
Executable file
54
.gitlab/before_script/linux/alpine/install_deps.sh
Executable file
@@ -0,0 +1,54 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../../../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "${TMPDIR}"
|
||||||
|
|
||||||
|
apk add --no-cache \
|
||||||
|
curl \
|
||||||
|
gcc \
|
||||||
|
g++ \
|
||||||
|
binutils \
|
||||||
|
binutils-gold \
|
||||||
|
bsd-compat-headers \
|
||||||
|
gmp-dev \
|
||||||
|
ncurses-dev \
|
||||||
|
libffi-dev \
|
||||||
|
make \
|
||||||
|
xz \
|
||||||
|
tar \
|
||||||
|
perl
|
||||||
|
|
||||||
|
if [ "${ARCH}" = "32" ] ; then
|
||||||
|
curl -sSfL https://downloads.haskell.org/ghcup/i386-linux-ghcup > ./ghcup-bin
|
||||||
|
else
|
||||||
|
curl -sSfL https://downloads.haskell.org/ghcup/x86_64-linux-ghcup > ./ghcup-bin
|
||||||
|
fi
|
||||||
|
chmod +x ghcup-bin
|
||||||
|
./ghcup-bin upgrade -i -f
|
||||||
|
./ghcup-bin install ${GHC_VERSION}
|
||||||
|
./ghcup-bin install-cabal ${CABAL_VERSION}
|
||||||
|
|
||||||
|
# utils
|
||||||
|
apk add --no-cache \
|
||||||
|
bash \
|
||||||
|
git
|
||||||
|
|
||||||
|
## Package specific
|
||||||
|
apk add --no-cache \
|
||||||
|
zlib \
|
||||||
|
zlib-dev \
|
||||||
|
zlib-static \
|
||||||
|
bzip2 \
|
||||||
|
bzip2-dev \
|
||||||
|
bzip2-static \
|
||||||
|
gmp \
|
||||||
|
gmp-dev \
|
||||||
|
openssl-dev \
|
||||||
|
openssl-libs-static \
|
||||||
|
xz \
|
||||||
|
xz-dev \
|
||||||
|
ncurses-static
|
||||||
|
|
||||||
66
.gitlab/before_script/linux/install_deps.sh
Executable file
66
.gitlab/before_script/linux/install_deps.sh
Executable file
@@ -0,0 +1,66 @@
|
|||||||
|
#!/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 libbz2-dev git wget lsb-release software-properties-common gnupg2 apt-transport-https
|
||||||
|
|
||||||
|
case "${ARCH}" in
|
||||||
|
ARM*)
|
||||||
|
case "${ARCH}" in
|
||||||
|
"ARM")
|
||||||
|
ghc_url=https://downloads.haskell.org/~ghc/${GHC_VERSION}/ghc-${GHC_VERSION}-armv7-deb10-linux.tar.xz
|
||||||
|
cabal_url=home.smart-cactus.org/~ben/cabal-install-${CABAL_VERSION}-armv7-linux-bootstrapped.tar.xz
|
||||||
|
;;
|
||||||
|
"ARM64")
|
||||||
|
ghc_url=https://downloads.haskell.org/~ghc/${GHC_VERSION}/ghc-${GHC_VERSION}-aarch64-deb10-linux.tar.xz
|
||||||
|
cabal_url=https://downloads.haskell.org/~cabal/cabal-install-${CABAL_VERSION}/cabal-install-${CABAL_VERSION}-aarch64-ubuntu-18.04.tar.xz
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
exit 1 ;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
mkdir -p "${GHCUP_INSTALL_BASE_PREFIX}"/.ghcup/bin
|
||||||
|
|
||||||
|
curl -O "${ghc_url}"
|
||||||
|
tar -xf ghc-*.tar.*
|
||||||
|
cd ghc-${GHC_VERSION}
|
||||||
|
./configure --prefix="${GHCUP_INSTALL_BASE_PREFIX}"/.ghcup/ghc/${GHC_VERSION}
|
||||||
|
make install
|
||||||
|
for i in "${GHCUP_INSTALL_BASE_PREFIX}"/.ghcup/ghc/${GHC_VERSION}/bin/*-${GHC_VERSION} ; do
|
||||||
|
ln -s "${i}" "${GHCUP_INSTALL_BASE_PREFIX}"/.ghcup/bin/${i##*/}
|
||||||
|
done
|
||||||
|
for x in "${GHCUP_INSTALL_BASE_PREFIX}"/.ghcup/bin/*-${GHC_VERSION} ; do
|
||||||
|
ln -s ${x##*/} ${x%-${GHC_VERSION}}
|
||||||
|
done
|
||||||
|
cd ..
|
||||||
|
rm -rf ghc-${GHC_VERSION} ghc-*.tar.*
|
||||||
|
unset x i
|
||||||
|
|
||||||
|
mkdir cabal-install
|
||||||
|
cd cabal-install
|
||||||
|
curl -O "${cabal_url}"
|
||||||
|
tar -xf cabal-install-*
|
||||||
|
mv cabal "${GHCUP_INSTALL_BASE_PREFIX}"/.ghcup/bin/cabal
|
||||||
|
cd ..
|
||||||
|
rm -rf cabal-install
|
||||||
|
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
url=https://downloads.haskell.org/~ghcup/x86_64-linux-ghcup
|
||||||
|
|
||||||
|
curl -sSfL "${url}" > ./ghcup-bin
|
||||||
|
chmod +x ghcup-bin
|
||||||
|
|
||||||
|
./ghcup-bin upgrade -i -f
|
||||||
|
./ghcup-bin install ${GHC_VERSION}
|
||||||
|
./ghcup-bin set ${GHC_VERSION}
|
||||||
|
./ghcup-bin install-cabal ${CABAL_VERSION}
|
||||||
|
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
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 libbz2-dev git wget
|
||||||
21
.gitlab/before_script/windows/install_deps.sh
Normal file
21
.gitlab/before_script/windows/install_deps.sh
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "${TMPDIR}" "${CABAL_DIR}"
|
||||||
|
|
||||||
|
mkdir -p "$GHCUP_INSTALL_BASE_PREFIX/ghcup/bin"
|
||||||
|
|
||||||
|
CI_PROJECT_DIR=$(pwd)
|
||||||
|
curl -o ghcup.exe https://downloads.haskell.org/~ghcup/0.1.15.1/x86_64-mingw64-ghcup-0.1.15.1.exe
|
||||||
|
chmod +x ghcup.exe
|
||||||
|
|
||||||
|
./ghcup.exe install ${GHC_VERSION}
|
||||||
|
./ghcup.exe set ${GHC_VERSION}
|
||||||
|
./ghcup.exe install-cabal ${CABAL_VERSION}
|
||||||
|
|
||||||
|
rm ./ghcup.exe
|
||||||
|
|
||||||
|
exit 0
|
||||||
9
.gitlab/ghcup_env
Normal file
9
.gitlab/ghcup_env
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
if [ "${OS}" = "WINDOWS" ] ; then
|
||||||
|
export GHCUP_INSTALL_BASE_PREFIX="$CI_PROJECT_DIR"
|
||||||
|
export PATH="$GHCUP_INSTALL_BASE_PREFIX/ghcup/bin:$CI_PROJECT_DIR/.local/bin:$PATH"
|
||||||
|
export TMPDIR="$CI_PROJECT_DIR/tmp"
|
||||||
|
else
|
||||||
|
export GHCUP_INSTALL_BASE_PREFIX="$CI_PROJECT_DIR"
|
||||||
|
export PATH="$CI_PROJECT_DIR/.ghcup/bin:$CI_PROJECT_DIR/.local/bin:/opt/llvm/bin:$PATH"
|
||||||
|
export TMPDIR="$CI_PROJECT_DIR/tmp"
|
||||||
|
fi
|
||||||
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 "$@"
|
||||||
|
}
|
||||||
|
|
||||||
|
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}" ]
|
||||||
|
|
||||||
48
.gitlab/script/ghcup_release.sh
Executable file
48
.gitlab/script/ghcup_release.sh
Executable file
@@ -0,0 +1,48 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "$CI_PROJECT_DIR"/.local/bin
|
||||||
|
|
||||||
|
ecabal() {
|
||||||
|
cabal "$@"
|
||||||
|
}
|
||||||
|
|
||||||
|
git describe
|
||||||
|
|
||||||
|
# build
|
||||||
|
ecabal update
|
||||||
|
|
||||||
|
(
|
||||||
|
cd /tmp
|
||||||
|
ecabal install -w ghc-${GHC_VERSION} --installdir="$CI_PROJECT_DIR"/.local/bin hspec-discover
|
||||||
|
)
|
||||||
|
|
||||||
|
if [ "${OS}" = "LINUX" ] ; then
|
||||||
|
if [ "${ARCH}" = "32" ] ; then
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} --ghc-options='-split-sections -optl-static' -ftui -ftar
|
||||||
|
elif [ "${ARCH}" = "64" ] ; then
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} --ghc-options='-split-sections -optl-static' -ftui
|
||||||
|
else
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} -ftui
|
||||||
|
fi
|
||||||
|
elif [ "${OS}" = "FREEBSD" ] ; then
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} --ghc-options='-split-sections' --constraint="zlib +bundled-c-zlib" --constraint="zip +disable-zstd" -ftui
|
||||||
|
elif [ "${OS}" = "WINDOWS" ] ; then
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} --constraint="zlib +bundled-c-zlib" --constraint="lzma +static"
|
||||||
|
else
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} --constraint="zlib +bundled-c-zlib" --constraint="lzma +static" -ftui
|
||||||
|
fi
|
||||||
|
|
||||||
|
mkdir out
|
||||||
|
binary=$(ecabal new-exec -w ghc-${GHC_VERSION} --verbose=0 --offline sh -- -c 'command -v ghcup')
|
||||||
|
ver=$("${binary}" --numeric-version)
|
||||||
|
if [ "${OS}" = "DARWIN" ] ; then
|
||||||
|
strip "${binary}"
|
||||||
|
else
|
||||||
|
strip -s "${binary}"
|
||||||
|
fi
|
||||||
|
cp "${binary}" out/${ARTIFACT}-${ver}
|
||||||
|
|
||||||
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
|
||||||
164
.gitlab/script/ghcup_version.sh
Executable file
164
.gitlab/script/ghcup_version.sh
Executable file
@@ -0,0 +1,164 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "$CI_PROJECT_DIR"/.local/bin
|
||||||
|
|
||||||
|
CI_PROJECT_DIR=$(pwd)
|
||||||
|
|
||||||
|
ecabal() {
|
||||||
|
cabal "$@"
|
||||||
|
}
|
||||||
|
|
||||||
|
eghcup() {
|
||||||
|
if [ "${OS}" = "WINDOWS" ] ; then
|
||||||
|
ghcup -v -c -s file:/$CI_PROJECT_DIR/ghcup-${JSON_VERSION}.yaml "$@"
|
||||||
|
else
|
||||||
|
ghcup -v -c -s file://$CI_PROJECT_DIR/ghcup-${JSON_VERSION}.yaml "$@"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
git describe --always
|
||||||
|
|
||||||
|
### build
|
||||||
|
|
||||||
|
ecabal update
|
||||||
|
|
||||||
|
(
|
||||||
|
cd /tmp
|
||||||
|
ecabal install -w ghc-${GHC_VERSION} --installdir="$CI_PROJECT_DIR"/.local/bin hspec-discover
|
||||||
|
)
|
||||||
|
|
||||||
|
if [ "${OS}" = "DARWIN" ] ; then
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} -ftui
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -ftui ghcup-test
|
||||||
|
ecabal haddock -w ghc-${GHC_VERSION} -ftui
|
||||||
|
elif [ "${OS}" = "LINUX" ] ; then
|
||||||
|
if [ "${ARCH}" = "32" ] ; then
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui -ftar
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -finternal-downloader -ftui -ftar ghcup-test
|
||||||
|
ecabal haddock -w ghc-${GHC_VERSION} -finternal-downloader -ftui -ftar
|
||||||
|
else
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -finternal-downloader -ftui ghcup-test
|
||||||
|
ecabal haddock -w ghc-${GHC_VERSION} -finternal-downloader -ftui
|
||||||
|
fi
|
||||||
|
elif [ "${OS}" = "FREEBSD" ] ; then
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui --constraint="zip +disable-zstd"
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -finternal-downloader -ftui --constraint="zip +disable-zstd" ghcup-test
|
||||||
|
ecabal haddock -w ghc-${GHC_VERSION} -finternal-downloader -ftui --constraint="zip +disable-zstd"
|
||||||
|
elif [ "${OS}" = "WINDOWS" ] ; then
|
||||||
|
ecabal build -w ghc-${GHC_VERSION}
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} ghcup-test
|
||||||
|
ecabal haddock -w ghc-${GHC_VERSION}
|
||||||
|
else
|
||||||
|
ecabal build -w ghc-${GHC_VERSION} -finternal-downloader -ftui
|
||||||
|
ecabal test -w ghc-${GHC_VERSION} -finternal-downloader -ftui ghcup-test
|
||||||
|
ecabal haddock -w ghc-${GHC_VERSION} -finternal-downloader -ftui
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
|
if [ "${OS}" = "WINDOWS" ] ; then
|
||||||
|
ext=".exe"
|
||||||
|
else
|
||||||
|
ext=''
|
||||||
|
fi
|
||||||
|
cp "$(ecabal new-exec -w ghc-${GHC_VERSION} --verbose=0 --offline sh -- -c 'command -v ghcup')" "$CI_PROJECT_DIR"/.local/bin/ghcup${ext}
|
||||||
|
cp "$(ecabal new-exec -w ghc-${GHC_VERSION} --verbose=0 --offline sh -- -c 'command -v ghcup-gen')" "$CI_PROJECT_DIR"/.local/bin/ghcup-gen${ext}
|
||||||
|
|
||||||
|
### cleanup
|
||||||
|
|
||||||
|
if [ "${OS}" = "WINDOWS" ] ; then
|
||||||
|
rm -rf "${GHCUP_INSTALL_BASE_PREFIX}"/ghcup
|
||||||
|
else
|
||||||
|
rm -rf "${GHCUP_INSTALL_BASE_PREFIX}"/.ghcup
|
||||||
|
fi
|
||||||
|
|
||||||
|
### manual cli based testing
|
||||||
|
|
||||||
|
|
||||||
|
ghcup-gen check -f ghcup-${JSON_VERSION}.yaml
|
||||||
|
|
||||||
|
eghcup --numeric-version
|
||||||
|
|
||||||
|
eghcup install ${GHC_VERSION}
|
||||||
|
eghcup set ${GHC_VERSION}
|
||||||
|
eghcup install-cabal ${CABAL_VERSION}
|
||||||
|
|
||||||
|
cabal --version
|
||||||
|
|
||||||
|
eghcup debug-info
|
||||||
|
|
||||||
|
eghcup list
|
||||||
|
eghcup list -t ghc
|
||||||
|
eghcup list -t cabal
|
||||||
|
|
||||||
|
ghc_ver=$(ghc --numeric-version)
|
||||||
|
ghc --version
|
||||||
|
ghc-${ghc_ver} --version
|
||||||
|
if [ "${OS}" != "WINDOWS" ] ; then
|
||||||
|
ghci --version
|
||||||
|
ghci-${ghc_ver} --version
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
|
if [ "${OS}" = "DARWIN" ] && [ "${ARCH}" = "ARM64" ] ; then
|
||||||
|
echo
|
||||||
|
else
|
||||||
|
# test installing new ghc doesn't mess with currently set GHC
|
||||||
|
# https://gitlab.haskell.org/haskell/ghcup-hs/issues/7
|
||||||
|
if [ "${OS}" = "LINUX" ] ; then
|
||||||
|
eghcup --downloader=wget install 8.10.3
|
||||||
|
else # test wget a bit
|
||||||
|
eghcup install 8.10.3
|
||||||
|
fi
|
||||||
|
[ "$(ghc --numeric-version)" = "${ghc_ver}" ]
|
||||||
|
eghcup set 8.10.3
|
||||||
|
eghcup set 8.10.3
|
||||||
|
[ "$(ghc --numeric-version)" = "8.10.3" ]
|
||||||
|
eghcup set ${GHC_VERSION}
|
||||||
|
[ "$(ghc --numeric-version)" = "${ghc_ver}" ]
|
||||||
|
eghcup rm 8.10.3
|
||||||
|
[ "$(ghc --numeric-version)" = "${ghc_ver}" ]
|
||||||
|
|
||||||
|
if [ "${OS}" = "DARWIN" ] ; then
|
||||||
|
eghcup install hls
|
||||||
|
haskell-language-server-wrapper --version
|
||||||
|
|
||||||
|
eghcup install stack
|
||||||
|
stack --version
|
||||||
|
elif [ "${OS}" = "LINUX" ] ; then
|
||||||
|
if [ "${ARCH}" = "64" ] ; then
|
||||||
|
eghcup install hls
|
||||||
|
haskell-language-server-wrapper --version
|
||||||
|
|
||||||
|
eghcup install stack
|
||||||
|
stack --version
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
|
eghcup rm $(ghc --numeric-version)
|
||||||
|
|
||||||
|
# https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/116
|
||||||
|
if [ "${OS}" = "LINUX" ] ; then
|
||||||
|
if [ "${ARCH}" = "64" ] ; then
|
||||||
|
eghcup install cabal -u https://oleg.fi/cabal-install-3.4.0.0-rc4/cabal-install-3.4.0.0-x86_64-ubuntu-16.04.tar.xz 3.4.0.0-rc4
|
||||||
|
eghcup rm cabal 3.4.0.0-rc4
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
eghcup upgrade
|
||||||
|
eghcup upgrade -f
|
||||||
|
|
||||||
|
|
||||||
|
# nuke
|
||||||
|
eghcup nuke
|
||||||
|
if [ "${OS}" = "WINDOWS" ] ; then
|
||||||
|
[ ! -e "${GHCUP_INSTALL_BASE_PREFIX}/ghcup" ]
|
||||||
|
else
|
||||||
|
[ ! -e "${GHCUP_INSTALL_BASE_PREFIX}/.ghcup" ]
|
||||||
|
fi
|
||||||
19
.gitlab/script/hlint.sh
Executable file
19
.gitlab/script/hlint.sh
Executable file
@@ -0,0 +1,19 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -eux
|
||||||
|
|
||||||
|
. "$( cd "$(dirname "$0")" ; pwd -P )/../ghcup_env"
|
||||||
|
|
||||||
|
mkdir -p "$CI_PROJECT_DIR"/.local/bin
|
||||||
|
|
||||||
|
ecabal() {
|
||||||
|
cabal --store-dir="$(pwd)"/.store "$@"
|
||||||
|
}
|
||||||
|
|
||||||
|
git describe
|
||||||
|
|
||||||
|
ecabal update
|
||||||
|
ecabal install -w ghc-${GHC_VERSION} --installdir="$CI_PROJECT_DIR"/.local/bin hlint
|
||||||
|
|
||||||
|
hlint -r lib/ test/
|
||||||
|
|
||||||
89
.gitlab/shell.nix
Normal file
89
.gitlab/shell.nix
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
{ system ? "aarch64-darwin"
|
||||||
|
#, nixpkgs ? fetchTarball https://github.com/angerman/nixpkgs/archive/257cb120334.tar.gz #apple-silicon.tar.gz
|
||||||
|
, pkgs ? import <nixpkgs> { inherit system; }
|
||||||
|
, compiler ? if system == "aarch64-darwin" then "ghc8103Binary" else "ghc8103"
|
||||||
|
}: pkgs.mkShell {
|
||||||
|
# this prevents nix from trying to write the env-vars file.
|
||||||
|
# we can't really, as NIX_BUILD_TOP/env-vars is not set.
|
||||||
|
noDumpEnvVars=1;
|
||||||
|
|
||||||
|
# stop polluting LDFLAGS with -liconv
|
||||||
|
dontAddExtraLibs = true;
|
||||||
|
|
||||||
|
# we need to inject ncurses into --with-curses-libraries.
|
||||||
|
# the real fix is to teach terminfo to use libcurses on macOS.
|
||||||
|
# CONFIGURE_ARGS = "--with-intree-gmp --with-curses-libraries=${pkgs.ncurses.out}/lib";
|
||||||
|
CONFIGURE_ARGS = "--with-intree-gmp --with-curses-libraries=/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/lib --with-iconv-includes=/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/include --with-iconv-libraries=/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/lib SH=/bin/bash";
|
||||||
|
|
||||||
|
# magic speedup pony :facepalm:
|
||||||
|
#
|
||||||
|
# nix has the ugly habbit of duplicating ld flags more than necessary. This
|
||||||
|
# somewhat consolidates this.
|
||||||
|
shellHook = ''
|
||||||
|
export NIX_LDFLAGS=$(for a in $NIX_LDFLAGS; do echo $a; done |sort|uniq|xargs)
|
||||||
|
export NIX_LDFLAGS_FOR_TARGET=$(for a in $NIX_LDFLAGS_FOR_TARGET; do echo $a; done |sort|uniq|xargs)
|
||||||
|
export NIX_LDFLAGS_FOR_TARGET=$(comm -3 <(for l in $NIX_LDFLAGS_FOR_TARGET; do echo $l; done) <(for l in $NIX_LDFLAGS; do echo $l; done))
|
||||||
|
|
||||||
|
|
||||||
|
# Impurity hack for GHC releases.
|
||||||
|
#################################
|
||||||
|
# We don't want binary releases to depend on nix, thus we'll need to make sure we don't leak in references.
|
||||||
|
# GHC externally depends only on iconv and curses. However we can't force a specific curses library for
|
||||||
|
# the terminfo package, as such we'll need to make sure we only look in the system path for the curses library
|
||||||
|
# and not pick up the tinfo from the nix provided ncurses package.
|
||||||
|
#
|
||||||
|
# We also need to force us to use the systems COREFOUNDATION, not the one that nix builds. Again this is impure,
|
||||||
|
# but it will allow us to have proper binary distributions.
|
||||||
|
#
|
||||||
|
# do not use nixpkgs provided core foundation
|
||||||
|
export NIX_COREFOUNDATION_RPATH=/System/Library/Frameworks
|
||||||
|
# drop curses from the LDFLAGS, we really want the system ones, not the nix ones.
|
||||||
|
export NIX_LDFLAGS=$(for lib in $NIX_LDFLAGS; do case "$lib" in *curses*);; *) echo -n "$lib ";; esac; done;)
|
||||||
|
export NIX_CFLAGS_COMPILE+=" -Wno-nullability-completeness -Wno-availability -Wno-expansion-to-defined -Wno-builtin-requires-header -Wno-unused-command-line-argument"
|
||||||
|
|
||||||
|
# unconditionally add the MacOSX.sdk and TargetConditional.h
|
||||||
|
export NIX_CFLAGS_COMPILE+=" -isystem /Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/include"
|
||||||
|
|
||||||
|
'';
|
||||||
|
|
||||||
|
nativeBuildInputs = (with pkgs; [
|
||||||
|
# This needs to come *before* ghc,
|
||||||
|
# otherwise we migth end up with the clang from
|
||||||
|
# the bootstrap GHC in PATH with higher priority.
|
||||||
|
clang_11
|
||||||
|
llvm_11
|
||||||
|
|
||||||
|
haskell.compiler.${compiler}
|
||||||
|
haskell.packages.${compiler}.cabal-install
|
||||||
|
haskell.packages.${compiler}.alex
|
||||||
|
haskell.packages.${compiler}.happy # _1_19_12 is needed for older GHCs.
|
||||||
|
|
||||||
|
automake
|
||||||
|
autoconf
|
||||||
|
m4
|
||||||
|
|
||||||
|
gmp
|
||||||
|
zlib.out
|
||||||
|
zlib.dev
|
||||||
|
glibcLocales
|
||||||
|
# locale doesn't build yet :-/
|
||||||
|
# locale
|
||||||
|
|
||||||
|
git
|
||||||
|
|
||||||
|
python3
|
||||||
|
# python3Full
|
||||||
|
# python3Packages.sphinx
|
||||||
|
perl
|
||||||
|
|
||||||
|
which
|
||||||
|
wget
|
||||||
|
curl
|
||||||
|
file
|
||||||
|
|
||||||
|
xz
|
||||||
|
xlibs.lndir
|
||||||
|
|
||||||
|
cacert ])
|
||||||
|
++ (with pkgs.darwin.apple_sdk.frameworks; [ Foundation Security ]);
|
||||||
|
}
|
||||||
83
.hlint.yaml
Normal file
83
.hlint.yaml
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
# HLint configuration file
|
||||||
|
# https://github.com/ndmitchell/hlint
|
||||||
|
##########################
|
||||||
|
|
||||||
|
# This file contains a template configuration file, which is typically
|
||||||
|
# placed as .hlint.yaml in the root of your project
|
||||||
|
|
||||||
|
|
||||||
|
# Warnings currently triggered by your code
|
||||||
|
- ignore: {name: "Redundant bang pattern"}
|
||||||
|
- ignore: {name: "Use camelCase"}
|
||||||
|
- ignore: {name: "Use if"}
|
||||||
|
- ignore: {name: "Use newtype instead of data"}
|
||||||
|
- ignore: {name: "Use <$>"}
|
||||||
|
- ignore: {name: "Use mapMaybe"}
|
||||||
|
- ignore: {name: "Use const"}
|
||||||
|
- ignore: {name: "Use list comprehension"}
|
||||||
|
- ignore: {name: "Redundant multi-way if"}
|
||||||
|
- ignore: {name: "Redundant lambda"}
|
||||||
|
- ignore: {name: "Avoid lambda"}
|
||||||
|
- ignore: {name: "Use uncurry"}
|
||||||
|
- ignore: {name: "Use replicateM"}
|
||||||
|
- ignore: {name: "Redundant irrefutable pattern"}
|
||||||
|
|
||||||
|
|
||||||
|
# Specify additional command line arguments
|
||||||
|
#
|
||||||
|
# - arguments: [--color, --cpp-simple, -XQuasiQuotes]
|
||||||
|
|
||||||
|
|
||||||
|
# Control which extensions/flags/modules/functions can be used
|
||||||
|
#
|
||||||
|
# - extensions:
|
||||||
|
# - default: false # all extension are banned by default
|
||||||
|
# - name: [PatternGuards, ViewPatterns] # only these listed extensions can be used
|
||||||
|
# - {name: CPP, within: CrossPlatform} # CPP can only be used in a given module
|
||||||
|
#
|
||||||
|
# - flags:
|
||||||
|
# - {name: -w, within: []} # -w is allowed nowhere
|
||||||
|
#
|
||||||
|
# - modules:
|
||||||
|
# - {name: [Data.Set, Data.HashSet], as: Set} # if you import Data.Set qualified, it must be as 'Set'
|
||||||
|
# - {name: Control.Arrow, within: []} # Certain modules are banned entirely
|
||||||
|
#
|
||||||
|
# - functions:
|
||||||
|
# - {name: unsafePerformIO, within: []} # unsafePerformIO can only appear in no modules
|
||||||
|
|
||||||
|
|
||||||
|
# Add custom hints for this project
|
||||||
|
#
|
||||||
|
# Will suggest replacing "wibbleMany [myvar]" with "wibbleOne myvar"
|
||||||
|
# - error: {lhs: "wibbleMany [x]", rhs: wibbleOne x}
|
||||||
|
|
||||||
|
# The hints are named by the string they display in warning messages.
|
||||||
|
# For example, if you see a warning starting like
|
||||||
|
#
|
||||||
|
# Main.hs:116:51: Warning: Redundant ==
|
||||||
|
#
|
||||||
|
# You can refer to that hint with `{name: Redundant ==}` (see below).
|
||||||
|
|
||||||
|
# Turn on hints that are off by default
|
||||||
|
#
|
||||||
|
# Ban "module X(module X) where", to require a real export list
|
||||||
|
# - warn: {name: Use explicit module export list}
|
||||||
|
#
|
||||||
|
# Replace a $ b $ c with a . b $ c
|
||||||
|
# - group: {name: dollar, enabled: true}
|
||||||
|
#
|
||||||
|
# Generalise map to fmap, ++ to <>
|
||||||
|
# - group: {name: generalise, enabled: true}
|
||||||
|
|
||||||
|
|
||||||
|
# Ignore some builtin hints
|
||||||
|
# - ignore: {name: Use let}
|
||||||
|
# - ignore: {name: Use const, within: SpecialModule} # Only within certain modules
|
||||||
|
|
||||||
|
|
||||||
|
# Define some custom infix operators
|
||||||
|
# - fixity: infixr 3 ~^#^~
|
||||||
|
|
||||||
|
|
||||||
|
# To generate a suitable file for HLint do:
|
||||||
|
# $ hlint --default > .hlint.yaml
|
||||||
37
.travis.yml
Normal file
37
.travis.yml
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
jobs:
|
||||||
|
include:
|
||||||
|
- os: osx
|
||||||
|
osx_image: xcode8
|
||||||
|
language: generic
|
||||||
|
env: ARTIFACT=x86_64-apple-darwin-10.11-ghcup
|
||||||
|
|
||||||
|
- os: osx
|
||||||
|
osx_image: xcode10.1
|
||||||
|
language: generic
|
||||||
|
env: ARTIFACT=x86_64-apple-darwin-10.13-ghcup
|
||||||
|
|
||||||
|
- os: osx
|
||||||
|
osx_image: xcode11.3
|
||||||
|
language: generic
|
||||||
|
env: ARTIFACT=x86_64-apple-darwin-10.14-ghcup
|
||||||
|
|
||||||
|
allow_failures:
|
||||||
|
- os: osx
|
||||||
|
osx_image: xcode8
|
||||||
|
language: generic
|
||||||
|
env: ARTIFACT=x86_64-apple-darwin-10.11-ghcup
|
||||||
|
|
||||||
|
|
||||||
|
script: ".travis/build.sh"
|
||||||
|
|
||||||
|
deploy:
|
||||||
|
provider: releases
|
||||||
|
api_key:
|
||||||
|
secure: GQESg4TcYf3PQJRRaZV/kWS0hsF+OFnH2+EcwpgnIcfx4+aogMyprdh745KtBXe1FlFN1luKHksFjqceqhcg/xcNyeCJiSnLWMn4D/i4WUperEHseRBi5yZZCB1AvOjIlHrE4DS3a8pyEm1GV3G7CKY5Fu8jBjof2SnyENfd7fofhjtNHWmeFS+jBn8HRDf1YaSRYxzTw6uHLrPLsybfgQZVl7babMu/38Ghin0f5pz5OlNokzDxaubIYQHOZ7st7YndHJtBWWql/KualBWbMILy88dUVQBnbqQLP2P8d1ME8ILUjJVqz33HiRU0JzlEJyWfbvEjcJ6iD8M6n4nXTaxfu3i2UhhGsQ6SSBNKssMP4tji8nkNpMqG59wLQ/zhcetEm71fKkgJNrIMNllkqlWSo5K74IqqP9kiLg/qm8ipOJjui0gPk8tZXKcV+ztX1d1OVCapoLfiDM5l/0LLQXaTOXOV1x3e5LLQ/w2doNiH3eh5CV4II9dRu7owpaiiMBHMssmT0pH99jEeF6giHLKtt3y7l2GWoRLPdhsZZ54gxsaBxZt9GuypmkbNcr97CEnAVaWij5v0CF3w4rAWqy/tAxQpIDJOIOQBgmwG5WrBAKyKrFvEpBL5a8BPcRWJDvqKC83QeWpvPrEVdgJevC6ZN1MKzrb2SiPOwC2Kerc=
|
||||||
|
file: $ARTIFACT
|
||||||
|
on:
|
||||||
|
repo: haskell/ghcup-hs
|
||||||
|
tags: true
|
||||||
|
skip_cleanup: true
|
||||||
|
draft: true
|
||||||
|
|
||||||
28
.travis/build.sh
Executable file
28
.travis/build.sh
Executable file
@@ -0,0 +1,28 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
set -ex
|
||||||
|
|
||||||
|
mkdir -p ~/.ghcup/bin
|
||||||
|
curl -sSfL https://downloads.haskell.org/~ghcup/x86_64-apple-darwin-ghcup > ~/.ghcup/bin/ghcup
|
||||||
|
chmod +x ~/.ghcup/bin/ghcup
|
||||||
|
|
||||||
|
export PATH="$HOME/.ghcup/bin:$PATH"
|
||||||
|
|
||||||
|
ghcup install 8.10.4
|
||||||
|
ghcup install-cabal 3.4.0.0
|
||||||
|
ghcup set 8.10.4
|
||||||
|
|
||||||
|
|
||||||
|
## install ghcup
|
||||||
|
|
||||||
|
cabal update
|
||||||
|
|
||||||
|
(
|
||||||
|
cd /tmp
|
||||||
|
cabal install --installdir="$HOME"/.ghcup/bin hspec-discover
|
||||||
|
)
|
||||||
|
|
||||||
|
cabal build --constraint="zlib +static" --constraint="lzma +static" -ftui
|
||||||
|
cp "$(cabal new-exec --verbose=0 --offline sh -- -c 'command -v ghcup')" .
|
||||||
|
strip ./ghcup
|
||||||
|
cp ghcup "./${ARTIFACT}"
|
||||||
153
CHANGELOG.md
153
CHANGELOG.md
@@ -1,5 +1,156 @@
|
|||||||
# Revision history for ghcup
|
# Revision history for ghcup
|
||||||
|
|
||||||
## 0.1.0.0 -- YYYY-mm-dd
|
## 0.1.16 -- ????-??-??
|
||||||
|
|
||||||
|
* Add 'nuke' subcommand wrt [#135](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/135), implemented by Arjun Kathuria
|
||||||
|
|
||||||
|
## 0.1.15.2 -- 2021-06-13
|
||||||
|
|
||||||
|
* Remove legacy handling of cabal binary and be more graceful about binaries not installed by ghcup (e.g. stack)
|
||||||
|
* Fix GHC compilation from git
|
||||||
|
* Fix 'ghcup upgrade' on windows
|
||||||
|
* Allow to skip update checks via `GHCUP_SKIP_UPDATE_CHECK`
|
||||||
|
* Use libarchive on windows as well, fixing unpack errors wrt [#147](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/147)
|
||||||
|
|
||||||
|
## 0.1.15.1 -- 2021-06-11
|
||||||
|
|
||||||
|
* Add Apple Silicon support
|
||||||
|
* Add windows support wrt [#130](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/130)
|
||||||
|
* Add stack support
|
||||||
|
* Warn when /tmp doesn't have 5GB or more of disk space
|
||||||
|
* Allow to compile GHC from git repo wrt [#126](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/126)
|
||||||
|
* Allow to set custom ghc version when running 'ghcup compile ghc' wrt [#136](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/136)
|
||||||
|
* Add date to GHC bindist names created by ghcup
|
||||||
|
|
||||||
|
## 0.1.14.2 -- 2021-05-12
|
||||||
|
|
||||||
|
* Remove dead dependency on ascii-string
|
||||||
|
|
||||||
|
## 0.1.14.1 -- 2021-04-11
|
||||||
|
|
||||||
|
* Make internal symlink target parser more lax, fixes [#119](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/119)
|
||||||
|
* Prepare for hackage release
|
||||||
|
|
||||||
|
## 0.1.14 -- 2021-03-07
|
||||||
|
|
||||||
|
* Major bugfix: fix handling of stray versions wrt [#116](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/116)
|
||||||
|
* Fix error messages and overhaul pretty printing wrt [#115](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/115)
|
||||||
|
|
||||||
|
## 0.1.13 -- 2021-02-26
|
||||||
|
|
||||||
|
* Support ARMv7/AARCH64
|
||||||
|
* Add command line completions for installed and available versions wrt [MR #70](https://gitlab.haskell.org/haskell/ghcup-hs/-/merge_requests/70)
|
||||||
|
* Allow to cycle through set tools wrt [#114](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/114)
|
||||||
|
* Fix item selection with unavailable versions wrt [#107](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/107)
|
||||||
|
* Allow for dynamic post-install, post-remove and pre-compile msgs wrt [MR #68](https://gitlab.haskell.org/haskell/ghcup-hs/-/merge_requests/68)
|
||||||
|
* Alert user if upgraded ghcup is shadowed by old ghcup wrt [#111](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/111)
|
||||||
|
* Fix to `ghcup` directory creation and placement for the XDG install mode ([MR #49](https://gitlab.haskell.org/haskell/ghcup-hs/-/merge_requests/49))
|
||||||
|
* Do 755 permissions on executables, wrt #97
|
||||||
|
* Add [NO_COLOR](https://no-color.org/) support wrt [MR #47](https://gitlab.haskell.org/haskell/ghcup-hs/-/merge_requests/47)
|
||||||
|
|
||||||
|
## 0.1.12 -- 2020-11-21
|
||||||
|
|
||||||
|
* Fix disappearing HLS symlinks wrt #91
|
||||||
|
* 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 setting TUI hotkeys wrt #41
|
||||||
|
- see https://gitlab.haskell.org/haskell/ghcup-hs#configuration for a more in-depth explanation
|
||||||
|
* add a `--set` switch to `ghcup install ghc` to automatically set as default after install
|
||||||
|
* emit warnings when CC/LD is set wrt #82
|
||||||
|
* add support for version ranges in distro specifiers wrt #84
|
||||||
|
- e.g. `"(>= 19 && <= 20) || ==0.2.2"` is a valid version key for distro
|
||||||
|
|
||||||
|
## 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
|
||||||
|
* Allow pre-release versions of GHC/cabal
|
||||||
|
* Add XDG dirs support (set `GHCUP_USE_XDG_DIRS`) wrt #39
|
||||||
|
* Allow to specify regex for tarball subdir (e.g. `ghc-.*`)
|
||||||
|
* Allow installing arbitrary bindists more seamlessly:
|
||||||
|
- 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
|
||||||
|
|
||||||
|
## 0.1.8 -- 2020-07-21
|
||||||
|
|
||||||
|
* Fix bug in logging thread dying on newlines
|
||||||
|
* Allow to install from arbitrary bindists: `ghcup -n install ghc -u '{"dlHash": "", "dlSubdir": "ghc-8.10.1", "dlUri": "https://github.com/commercialhaskell/ghc/releases/download/ghc-8.10.1-release/ghc-8.10.1-x86_64-deb9-linux.tar.xz"}' 8.10.1`
|
||||||
|
|
||||||
|
## 0.1.7 -- 2020-07-20
|
||||||
|
|
||||||
|
* Fix a bug in libarchive not unpacking some uncleanly packed bindists
|
||||||
|
* Improved fish support in bootstrap-haskell
|
||||||
|
* Only check for upgrades when not upgrading
|
||||||
|
* Fix platform detection for i386 docker images
|
||||||
|
* Improve alpine support
|
||||||
|
- more/proper bindists
|
||||||
|
- don't fall back to glibc based bindists
|
||||||
|
- install bindists with `--disable-ld-override` to avoid ld.gold bugs
|
||||||
|
|
||||||
|
## 0.1.6 -- 2020-07-13
|
||||||
|
|
||||||
|
* Create a new curses (brick) based TUI, accessible via `ghcup tui` #24
|
||||||
|
* Support multiple installed versions of cabal #23
|
||||||
|
* Improvements to `ghcup list` (show unavailable bindists for platform)
|
||||||
|
* Fix redhat downloads #29
|
||||||
|
* Support for hadrian bindists (fixes alpine-8.10.1) #31
|
||||||
|
* Add FreeBSD bindists 8.6.5 and 8.8.3
|
||||||
|
* Fix memory leak during unpack
|
||||||
|
|
||||||
|
## 0.1.5 -- 2020-04-30
|
||||||
|
|
||||||
|
* Fix errors when PATH variable contains path components that are actually files
|
||||||
|
* Add `--version` and `--numeric-version` options
|
||||||
|
* Add `changelog` command
|
||||||
|
* Also check for available GHC and Cabal updates on start
|
||||||
|
* Add base versions as tags for every GHC version (these are "installable" tags and the latest GHC version matching the tag will be picked)
|
||||||
|
* Added `--format-raw` to list subcommand
|
||||||
|
* Allow to install X.Y versions (e.g.: ghcup install 8.8)
|
||||||
|
* Implement `--keep=<always|errors|never>` to control temporary build directories cleanup
|
||||||
|
* Add proper shell completions to the repo
|
||||||
|
* Fix building of documentation
|
||||||
|
* Allow to work in offline mode and use cached files if possible
|
||||||
|
* Allow to set the downloader via `--downloader=<curl|wget>`
|
||||||
|
* Support for compiling and installing a cross GHC (see README). This is experimental.
|
||||||
|
|
||||||
|
## 0.1.4 -- 2020-04-16
|
||||||
|
|
||||||
|
* build on all platforms with curl (as a binary), wrt https://gitlab.haskell.org/haskell/ghcup-hs/issues/6
|
||||||
|
* Fix unlinking of ghc symlinks after new installation, wrt https://gitlab.haskell.org/haskell/ghcup-hs/issues/7
|
||||||
|
|
||||||
|
## 0.1.3 -- 2020-04-15
|
||||||
|
|
||||||
|
* Fix lesser bug when skipping ghcup update
|
||||||
|
|
||||||
|
## 0.1.2 -- 2020-04-15
|
||||||
|
|
||||||
|
* Fix bug when removing the set GHC version
|
||||||
|
* Fix use of undocumented `GHCUP_INSTALL_BASE_PREFIX` variable
|
||||||
|
* skip upgrade if ghcup is already latest version
|
||||||
|
|
||||||
|
## 0.1.1 -- 2020-04-15
|
||||||
|
|
||||||
|
* fix awful fdopendir bug on mac bug by updating hpath-posix
|
||||||
|
|
||||||
|
## 0.1.0
|
||||||
|
|
||||||
* First version. Released on an unsuspecting world.
|
* First version. Released on an unsuspecting world.
|
||||||
|
|||||||
45
Dockerfile
45
Dockerfile
@@ -1,45 +0,0 @@
|
|||||||
FROM alpine:edge
|
|
||||||
|
|
||||||
# ghc and cabal
|
|
||||||
RUN apk add --no-cache \
|
|
||||||
curl \
|
|
||||||
gcc \
|
|
||||||
g++ \
|
|
||||||
gmp-dev \
|
|
||||||
ncurses-dev \
|
|
||||||
libffi-dev \
|
|
||||||
make \
|
|
||||||
xz \
|
|
||||||
tar \
|
|
||||||
perl \
|
|
||||||
\
|
|
||||||
cabal \
|
|
||||||
ghc
|
|
||||||
|
|
||||||
# utils
|
|
||||||
RUN apk add --no-cache \
|
|
||||||
bash
|
|
||||||
|
|
||||||
## Package specific
|
|
||||||
RUN apk add --no-cache \
|
|
||||||
libbz2 \
|
|
||||||
bzip2-dev \
|
|
||||||
bzip2-static \
|
|
||||||
zlib \
|
|
||||||
zlib-dev \
|
|
||||||
zlib-static \
|
|
||||||
gmp \
|
|
||||||
gmp-dev \
|
|
||||||
openssl-dev \
|
|
||||||
openssl-libs-static \
|
|
||||||
xz \
|
|
||||||
xz-dev
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
COPY . /app
|
|
||||||
|
|
||||||
WORKDIR /app
|
|
||||||
|
|
||||||
RUN chmod +x /app/docker/build.sh
|
|
||||||
|
|
||||||
75
HACKING.md
Normal file
75
HACKING.md
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
# HACKING
|
||||||
|
|
||||||
|
## Design decisions
|
||||||
|
|
||||||
|
### Using [Excepts](https://hackage.haskell.org/package/haskus-utils-variant-3.0/docs/Haskus-Utils-Variant-Excepts.html) as a beefed up ExceptT
|
||||||
|
|
||||||
|
This is an open variant, similar to [plucky](https://hackage.haskell.org/package/plucky) or [oops](https://github.com/i-am-tom/oops) and allows us to combine different error types. Maybe it is too much and it's a little bit [unergonomic](https://github.com/haskus/packages/issues/32) at times. If it really hurts maintenance, it will be removed. It was more of an experiment.
|
||||||
|
|
||||||
|
### No use of haskell-TLS
|
||||||
|
|
||||||
|
I consider haskell-TLS an interesting experiment, but not a battle-tested and peer-reviewed crypto implementation. There is little to no research about what the intricacies of using haskell for low-level crypto are and how vulnerable such binaries are. Instead, we use either curl the binary (for FreeBSD and mac) or http-io-streams, which works with OpenSSL bindings.
|
||||||
|
|
||||||
|
### Optics instead of lens
|
||||||
|
|
||||||
|
They're a little safer (less Monoid weirdness with view) and have better error messages. Consider the following wit lens
|
||||||
|
|
||||||
|
```
|
||||||
|
> view (_Just . to (++ "abc")) Nothing
|
||||||
|
""
|
||||||
|
```
|
||||||
|
|
||||||
|
vs optics
|
||||||
|
|
||||||
|
```
|
||||||
|
> view (_Just % to (++ "abc")) Nothing
|
||||||
|
|
||||||
|
<interactive>:2:1: error:
|
||||||
|
• An_AffineFold cannot be used as A_Getter
|
||||||
|
• In the expression: view (_Just % to (++ "abc")) Nothing
|
||||||
|
In an equation for ‘it’: it = view (_Just % to (++ "abc")) Nothing
|
||||||
|
```
|
||||||
|
|
||||||
|
### Strict and StrictData on by default
|
||||||
|
|
||||||
|
Kazu Yamamoto [explained it in his PR](https://github.com/yesodweb/wai/pull/752#issuecomment-501531386) very well. I like to agree with him. The instances where we need non-strict behavior, we annotate it.
|
||||||
|
|
||||||
|
## Code style and formatting
|
||||||
|
|
||||||
|
1. Brittany
|
||||||
|
2. mtl-style preferred
|
||||||
|
3. no overly pointfree style
|
||||||
|
|
||||||
|
## Code structure
|
||||||
|
|
||||||
|
Main functionality is in `GHCup` module. Utility functions are
|
||||||
|
organised tree-ish in `GHCup.Utils` and `GHCup.Utils.*`.
|
||||||
|
|
||||||
|
Anything dealing with ghcup specific directories is in
|
||||||
|
`GHCup.Utils.Dirs`.
|
||||||
|
|
||||||
|
Download information on where to fetch bindists from is in the appropriate
|
||||||
|
yaml files: `ghcup-<yaml-ver>.yaml`.
|
||||||
|
|
||||||
|
## Common Tasks
|
||||||
|
|
||||||
|
### Adding a new GHC version
|
||||||
|
|
||||||
|
1. open the latest `ghcup-<yaml-ver>.yaml`
|
||||||
|
2. find the latest ghc version (in yaml tree e.g. `ghcupDownloads -> GHC -> 8.10.3`)
|
||||||
|
3. copy-paste it
|
||||||
|
4. adjust the version, tags, changelog, source url
|
||||||
|
5. adjust the various bindist urls (make sure to also change the yaml anchors)
|
||||||
|
6. run `cabal run exe:ghcup-gen -- check-tarballs -f ghcup-<yaml-ver>.yaml -u 'ghc-8\.10\.4'`
|
||||||
|
|
||||||
|
## Major refactors
|
||||||
|
|
||||||
|
1. First major refactor included adding cross support. This added
|
||||||
|
`GHCTargetVersion`, which includes the target in addition to the version.
|
||||||
|
Most of the `Version` parameters to functions had to be replaced with
|
||||||
|
that and ensured the logic is consistent for cross and non-cross
|
||||||
|
installs.
|
||||||
|
2. This refactor added windows support wrt [#130](https://gitlab.haskell.org/haskell/ghcup-hs/-/issues/130).
|
||||||
|
The major changes here were switching `hpath` library out for `filepath`/`directory` (sadly) and
|
||||||
|
introducing a non-unix way of handling processes via the `process` library. It also introduced considerable
|
||||||
|
amounts of CPP wrt file handling, installation etc.
|
||||||
119
README.md
119
README.md
@@ -1,16 +1,23 @@
|
|||||||
`ghcup` makes it easy to install specific versions of `ghc` on GNU/Linux,
|
`ghcup` makes it easy to install specific versions of `ghc` on GNU/Linux,
|
||||||
macOS (aka Darwin) and FreeBSD and can also bootstrap a fresh Haskell developer environment from scratch.
|
macOS (aka Darwin), FreeBSD and Windows and can also bootstrap a fresh Haskell developer environment from scratch.
|
||||||
It follows the unix UNIX philosophy of [do one thing and do it well](https://en.wikipedia.org/wiki/Unix_philosophy#Do_One_Thing_and_Do_It_Well).
|
It follows the unix UNIX philosophy of [do one thing and do it well](https://en.wikipedia.org/wiki/Unix_philosophy#Do_One_Thing_and_Do_It_Well).
|
||||||
|
|
||||||
Similar in scope to [rustup](https://github.com/rust-lang-nursery/rustup.rs), [pyenv](https://github.com/pyenv/pyenv) and [jenv](http://www.jenv.be).
|
Similar in scope to [rustup](https://github.com/rust-lang-nursery/rustup.rs), [pyenv](https://github.com/pyenv/pyenv) and [jenv](http://www.jenv.be).
|
||||||
|
|
||||||
*Ubuntu users may prefer [hvr's ppa](https://launchpad.net/~hvr/+archive/ubuntu/ghc).*
|
|
||||||
|
|
||||||
## 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)
|
||||||
|
* [Cross support](#cross-support)
|
||||||
|
* [XDG support](#xdg-support)
|
||||||
|
* [Env variables](#env-variables)
|
||||||
|
* [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)
|
||||||
@@ -25,7 +32,7 @@ Follow the instructions at [https://www.haskell.org/ghcup/](https://www.haskell.
|
|||||||
|
|
||||||
### Manual install
|
### Manual install
|
||||||
|
|
||||||
Download the binary for your platform at [https://github.com/hasufell/ghcup-hs/releases](https://github.com/hasufell/ghcup-hs/releases)
|
Download the binary for your platform at [https://downloads.haskell.org/~ghcup/](https://downloads.haskell.org/~ghcup/)
|
||||||
and place it into your `PATH` anywhere.
|
and place it into your `PATH` anywhere.
|
||||||
|
|
||||||
Then adjust your `PATH` in `~/.bashrc` (or similar, depending on your shell) like so:
|
Then adjust your `PATH` in `~/.bashrc` (or similar, depending on your shell) like so:
|
||||||
@@ -34,11 +41,21 @@ 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`.
|
||||||
|
|
||||||
Common use cases are:
|
For the simple interactive TUI, run:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
ghcup tui
|
||||||
|
```
|
||||||
|
|
||||||
|
For the full functionality via cli:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
# list available ghc/cabal versions
|
# list available ghc/cabal versions
|
||||||
@@ -48,10 +65,10 @@ ghcup list
|
|||||||
ghcup install ghc
|
ghcup install ghc
|
||||||
|
|
||||||
# install a specific GHC version
|
# install a specific GHC version
|
||||||
ghcup install ghc -v 8.2.2
|
ghcup install ghc 8.2.2
|
||||||
|
|
||||||
# set the currently "active" GHC version
|
# set the currently "active" GHC version
|
||||||
ghcup set -v 8.4.4
|
ghcup set ghc 8.4.4
|
||||||
|
|
||||||
# install cabal-install
|
# install cabal-install
|
||||||
ghcup install cabal
|
ghcup install cabal
|
||||||
@@ -60,14 +77,83 @@ ghcup install cabal
|
|||||||
ghcup upgrade
|
ghcup upgrade
|
||||||
```
|
```
|
||||||
|
|
||||||
Generally this is meant to be used with [`cabal-install`](https://hackage.haskell.org/package/cabal-install), which
|
GHCup works very well 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 override 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.
|
||||||
`MANPATH` may be required to be unset.
|
`MANPATH` may be required to be unset.
|
||||||
|
|
||||||
|
### Shell-completion
|
||||||
|
|
||||||
|
Shell completions are in `shell-completions`.
|
||||||
|
|
||||||
|
For bash: install `shell-completions/bash`
|
||||||
|
as e.g. `/etc/bash_completion.d/ghcup` (depending on distro)
|
||||||
|
and make sure your bashrc sources the startup script
|
||||||
|
(`/usr/share/bash-completion/bash_completion` on some distros).
|
||||||
|
|
||||||
|
### Cross support
|
||||||
|
|
||||||
|
ghcup can compile and install a cross GHC for any target. However, this
|
||||||
|
requires that the build host has a complete cross toolchain and various
|
||||||
|
libraries installed for the target platform.
|
||||||
|
|
||||||
|
Consult the GHC documentation on the [prerequisites](https://gitlab.haskell.org/ghc/ghc/-/wikis/building/cross-compiling#tools-to-install).
|
||||||
|
For distributions with non-standard locations of cross toolchain and
|
||||||
|
libraries, this may need some tweaking of `build.mk` or configure args.
|
||||||
|
See `ghcup compile ghc --help` for further information.
|
||||||
|
|
||||||
|
### XDG support
|
||||||
|
|
||||||
|
To enable XDG style directories, set the environment variable `GHCUP_USE_XDG_DIRS` to anything.
|
||||||
|
|
||||||
|
Then you can control the locations via XDG environment variables as such:
|
||||||
|
|
||||||
|
* `XDG_DATA_HOME`: GHCs will be unpacked in `ghcup/ghc` subdir (default: `~/.local/share`)
|
||||||
|
* `XDG_CACHE_HOME`: logs and download files will be stored in `ghcup` subdir (default: `~/.cache`)
|
||||||
|
* `XDG_BIN_HOME`: binaries end up here (default: `~/.local/bin`)
|
||||||
|
* `XDG_CONFIG_HOME`: the config file is stored in `ghcup` subdir as `config.yaml` (default: `~/.config`)
|
||||||
|
|
||||||
|
**Note that `ghcup` makes some assumptions about structure of files in `XDG_BIN_HOME`. So if you have other tools
|
||||||
|
installing e.g. stack/cabal/ghc into it, this will likely clash. In that case consider disabling XDG support.**
|
||||||
|
|
||||||
|
### Env variables
|
||||||
|
|
||||||
|
This is the complete list of env variables that change GHCup behavior:
|
||||||
|
|
||||||
|
* `GHCUP_USE_XDG_DIRS`: see [XDG support](#xdg-support) above
|
||||||
|
* `TMPDIR`: where ghcup does the work (unpacking, building, ...)
|
||||||
|
* `GHCUP_INSTALL_BASE_PREFIX`: the base of ghcup (default: `$HOME`)
|
||||||
|
* `GHCUP_CURL_OPTS`: additional options that can be passed to curl
|
||||||
|
* `GHCUP_WGET_OPTS`: additional options that can be passed to wget
|
||||||
|
* `GHCUP_SKIP_UPDATE_CHECK`: Skip the (possibly annoying) update check when you run a command
|
||||||
|
* `CC`/`LD` etc.: full environment is passed to the build system when compiling GHC via GHCup
|
||||||
|
|
||||||
|
### 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
|
||||||
@@ -96,10 +182,22 @@ In addition this script can also install `cabal-install`.
|
|||||||
|
|
||||||
## Known users
|
## Known users
|
||||||
|
|
||||||
|
* Github action [haskell/actions/setup](https://github.com/haskell/actions/tree/main/setup)
|
||||||
* [vabal](https://github.com/Franciman/vabal)
|
* [vabal](https://github.com/Franciman/vabal)
|
||||||
|
|
||||||
## 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.
|
||||||
@@ -140,5 +238,8 @@ ghcup is not a reimplementation of stack. The only common part is automatic inst
|
|||||||
|
|
||||||
2. Why not support windows?
|
2. Why not support windows?
|
||||||
|
|
||||||
Consider using [Chocolatey](https://chocolatey.org/search?q=ghc) or [ghcups](https://github.com/kakkun61/ghcups).
|
We do.
|
||||||
|
|
||||||
|
3. Why the haskell reimplementation?
|
||||||
|
|
||||||
|
:-)
|
||||||
|
|||||||
19
RELEASING.md
Normal file
19
RELEASING.md
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
# RELEASING
|
||||||
|
|
||||||
|
1. update `GHCup.Version` module. `ghcupURL` must only be updated if we change the `_toolRequirements` type or the YAML representation of it. The version of the YAML represents the change increments. `ghcUpVer` is the current application version.
|
||||||
|
|
||||||
|
2. Update version in ghcup.cabal
|
||||||
|
|
||||||
|
3. Add ChangeLog entry
|
||||||
|
|
||||||
|
4. Add/fix downloads in `ghcup-<ver>.yaml`, then verify with `ghcup-gen check -f ghcup-<ver>.yaml`
|
||||||
|
|
||||||
|
5. Commit and git push with tag. Wait for tests to succeed and release artifacts to build.
|
||||||
|
|
||||||
|
6. Download release artifacts and upload them `downloads.haskell.org/ghcup`
|
||||||
|
|
||||||
|
7. Add release artifacts to yaml file (see point 4.)
|
||||||
|
|
||||||
|
8. Upload the final `ghcup-<ver>.yaml` to `webhost.haskell.org/ghcup/data/`.
|
||||||
|
|
||||||
|
9. Update bootstrap-haskell and symlinks on `downloads.haskell.org/ghcup`
|
||||||
28
TODO.md
28
TODO.md
@@ -2,27 +2,23 @@
|
|||||||
|
|
||||||
## Now
|
## Now
|
||||||
|
|
||||||
* travis
|
* ghcup init?
|
||||||
|
* merge two download files
|
||||||
|
* fetch/unpack functionality
|
||||||
|
* installing multiple versions of the same
|
||||||
|
* post-install
|
||||||
|
|
||||||
* requirements
|
* proper test suite
|
||||||
* for ghcup (bootstrap script)
|
|
||||||
* per tool
|
|
||||||
* mac build: xattr -cr .
|
|
||||||
* static binaries
|
|
||||||
* upgrade plan from old ghcup
|
|
||||||
|
|
||||||
* bootstrap-haskell with new ghcup
|
* !! update of 0.1.5 must go in ghcup-0.0.1.json !!
|
||||||
* add warning to ghcup script about new binary
|
|
||||||
|
|
||||||
* make sure smart-dl is not broken
|
* try to run exe before upgrade (backup to ~/.ghcup/bin/ghcup.old)
|
||||||
|
* stdout flushing?
|
||||||
* handle SIGINT better (remove dirs)
|
* resume support (for make-install only)
|
||||||
|
|
||||||
* review symlink handling (maybe fixed set of tools?)
|
|
||||||
|
|
||||||
## Maybe
|
## Maybe
|
||||||
|
|
||||||
* maybe: changelog Show the changelog of a GHC release (online)
|
* version ranges in json
|
||||||
* sign the JSON? (Or check gpg keys?)
|
* sign the JSON? (Or check gpg keys?)
|
||||||
* testing (especially distro detection -> unit tests)
|
* testing (especially distro detection -> unit tests)
|
||||||
|
|
||||||
@@ -37,7 +33,7 @@
|
|||||||
|
|
||||||
## Questions
|
## Questions
|
||||||
|
|
||||||
* fully static musl builds for linux?
|
* move out GHCup.Version module, bc it's not library-ish?
|
||||||
* mirror support
|
* mirror support
|
||||||
* interactive handling when distro doesn't exist and we know the tarball is incompatible?
|
* interactive handling when distro doesn't exist and we know the tarball is incompatible?
|
||||||
* ghcup-with wrapper to execute a command with a given ghc in PATH?
|
* ghcup-with wrapper to execute a command with a given ghc in PATH?
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,66 +1,40 @@
|
|||||||
|
{-# LANGUAGE CPP #-}
|
||||||
{-# LANGUAGE DataKinds #-}
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE TypeApplications #-}
|
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
|
||||||
{-# LANGUAGE TemplateHaskell #-}
|
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
|
||||||
{-# LANGUAGE DuplicateRecordFields #-}
|
{-# LANGUAGE DuplicateRecordFields #-}
|
||||||
|
{-# LANGUAGE NamedFieldPuns #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
{-# LANGUAGE TypeApplications #-}
|
||||||
|
|
||||||
|
|
||||||
module Main where
|
module Main where
|
||||||
|
|
||||||
|
import GHCup.Types
|
||||||
import GHCup.Types.JSON ( )
|
import GHCup.Types.JSON ( )
|
||||||
import GHCup.Utils.Logger
|
import GHCup.Utils.Logger
|
||||||
import GHCupDownloads
|
|
||||||
|
|
||||||
import Data.Aeson ( eitherDecode )
|
import Data.Char ( toLower )
|
||||||
import Data.Aeson.Encode.Pretty
|
#if !MIN_VERSION_base(4,13,0)
|
||||||
import Data.Semigroup ( (<>) )
|
import Data.Semigroup ( (<>) )
|
||||||
|
#endif
|
||||||
import Options.Applicative hiding ( style )
|
import Options.Applicative hiding ( style )
|
||||||
import System.Console.Pretty
|
import System.Console.Pretty
|
||||||
import System.Exit
|
import System.Exit
|
||||||
import System.IO ( stdout )
|
import System.IO ( stdout )
|
||||||
|
import Text.Regex.Posix
|
||||||
import Validate
|
import Validate
|
||||||
|
|
||||||
import qualified Data.ByteString as B
|
import qualified Data.ByteString as B
|
||||||
import qualified Data.ByteString.Lazy as L
|
import qualified Data.Yaml as Y
|
||||||
|
|
||||||
|
|
||||||
data Options = Options
|
data Options = Options
|
||||||
{ optCommand :: Command
|
{ optCommand :: Command
|
||||||
}
|
}
|
||||||
|
|
||||||
data Command = GenJSON GenJSONOpts
|
data Command = ValidateYAML ValidateYAMLOpts
|
||||||
| ValidateJSON ValidateJSONOpts
|
| ValidateTarballs ValidateYAMLOpts TarballFilter
|
||||||
| ValidateTarballs ValidateJSONOpts
|
|
||||||
|
|
||||||
data Output
|
|
||||||
= FileOutput FilePath -- optsparse-applicative doesn't handle ByteString correctly anyway
|
|
||||||
| StdOutput
|
|
||||||
|
|
||||||
fileOutput :: Parser Output
|
|
||||||
fileOutput =
|
|
||||||
FileOutput
|
|
||||||
<$> (strOption
|
|
||||||
(long "file" <> short 'f' <> metavar "FILENAME" <> help
|
|
||||||
"Output to a file"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
stdOutput :: Parser Output
|
|
||||||
stdOutput = flag'
|
|
||||||
StdOutput
|
|
||||||
(short 'o' <> long "stdout" <> help "Print to stdout (default)")
|
|
||||||
|
|
||||||
outputP :: Parser Output
|
|
||||||
outputP = fileOutput <|> stdOutput
|
|
||||||
|
|
||||||
|
|
||||||
data GenJSONOpts = GenJSONOpts
|
|
||||||
{ output :: Maybe Output
|
|
||||||
}
|
|
||||||
|
|
||||||
genJSONOpts :: Parser GenJSONOpts
|
|
||||||
genJSONOpts = GenJSONOpts <$> optional outputP
|
|
||||||
|
|
||||||
|
|
||||||
data Input
|
data Input
|
||||||
@@ -70,11 +44,10 @@ data Input
|
|||||||
fileInput :: Parser Input
|
fileInput :: Parser Input
|
||||||
fileInput =
|
fileInput =
|
||||||
FileInput
|
FileInput
|
||||||
<$> (strOption
|
<$> strOption
|
||||||
(long "file" <> short 'f' <> metavar "FILENAME" <> help
|
(long "file" <> short 'f' <> metavar "FILENAME" <> help
|
||||||
"Input file to validate"
|
"Input file to validate"
|
||||||
)
|
)
|
||||||
)
|
|
||||||
|
|
||||||
stdInput :: Parser Input
|
stdInput :: Parser Input
|
||||||
stdInput = flag'
|
stdInput = flag'
|
||||||
@@ -84,81 +57,71 @@ stdInput = flag'
|
|||||||
inputP :: Parser Input
|
inputP :: Parser Input
|
||||||
inputP = fileInput <|> stdInput
|
inputP = fileInput <|> stdInput
|
||||||
|
|
||||||
data ValidateJSONOpts = ValidateJSONOpts
|
data ValidateYAMLOpts = ValidateYAMLOpts
|
||||||
{ input :: Maybe Input
|
{ vInput :: Maybe Input
|
||||||
}
|
}
|
||||||
|
|
||||||
validateJSONOpts :: Parser ValidateJSONOpts
|
validateYAMLOpts :: Parser ValidateYAMLOpts
|
||||||
validateJSONOpts = ValidateJSONOpts <$> optional inputP
|
validateYAMLOpts = ValidateYAMLOpts <$> optional inputP
|
||||||
|
|
||||||
|
tarballFilterP :: Parser TarballFilter
|
||||||
|
tarballFilterP = option readm $
|
||||||
|
long "tarball-filter" <> short 'u' <> metavar "<tool>-<version>" <> value def
|
||||||
|
<> help "Only check certain tarballs (format: <tool>-<version>)"
|
||||||
|
where
|
||||||
|
def = TarballFilter (Right Nothing) (makeRegex ("" :: String))
|
||||||
|
readm = do
|
||||||
|
s <- str
|
||||||
|
case span (/= '-') s of
|
||||||
|
(_, []) -> fail "invalid format, missing '-' after the tool name"
|
||||||
|
(t, v) | [tool] <- [ tool | tool <- [minBound..maxBound], low (show tool) == low t ] ->
|
||||||
|
pure (TarballFilter $ Right $ Just tool) <*> makeRegexOptsM compIgnoreCase execBlank (drop 1 v)
|
||||||
|
(t, v) | [tool] <- [ tool | tool <- [minBound..maxBound], low (show tool) == low t ] ->
|
||||||
|
pure (TarballFilter $ Left tool) <*> makeRegexOptsM compIgnoreCase execBlank (drop 1 v)
|
||||||
|
_ -> fail "invalid tool"
|
||||||
|
low = fmap toLower
|
||||||
|
|
||||||
|
|
||||||
opts :: Parser Options
|
opts :: Parser Options
|
||||||
opts = Options <$> com
|
opts = Options <$> com
|
||||||
|
|
||||||
com :: Parser Command
|
com :: Parser Command
|
||||||
com = subparser
|
com = subparser
|
||||||
( (command
|
( command
|
||||||
"gen"
|
|
||||||
( GenJSON
|
|
||||||
<$> (info (genJSONOpts <**> helper)
|
|
||||||
(progDesc "Generate the json downloads file")
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
<> (command
|
|
||||||
"check"
|
"check"
|
||||||
( ValidateJSON
|
( ValidateYAML
|
||||||
<$> (info (validateJSONOpts <**> helper)
|
<$> info (validateYAMLOpts <**> helper)
|
||||||
(progDesc "Validate the JSON")
|
(progDesc "Validate the YAML")
|
||||||
)
|
|
||||||
)
|
)
|
||||||
)
|
<> command
|
||||||
<> (command
|
|
||||||
"check-tarballs"
|
"check-tarballs"
|
||||||
( ValidateTarballs
|
(info
|
||||||
<$> (info
|
((ValidateTarballs <$> validateYAMLOpts <*> tarballFilterP) <**> helper)
|
||||||
(validateJSONOpts <**> helper)
|
(progDesc "Validate all tarballs (download and checksum)")
|
||||||
(progDesc "Validate all tarballs (download and checksum)")
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
main :: IO ()
|
main :: IO ()
|
||||||
main = do
|
main = do
|
||||||
customExecParser (prefs showHelpOnError) (info (opts <**> helper) idm)
|
_ <- customExecParser (prefs showHelpOnError) (info (opts <**> helper) idm)
|
||||||
>>= \Options {..} -> case optCommand of
|
>>= \Options {..} -> case optCommand of
|
||||||
GenJSON gopts -> do
|
ValidateYAML vopts -> withValidateYamlOpts vopts validate
|
||||||
let
|
ValidateTarballs vopts tarballFilter -> withValidateYamlOpts vopts (validateTarballs tarballFilter)
|
||||||
bs = encodePretty' (defConfig { confIndent = Spaces 2 })
|
|
||||||
ghcupDownloads
|
|
||||||
case gopts of
|
|
||||||
GenJSONOpts { output = Nothing } -> L.hPutStr stdout bs
|
|
||||||
GenJSONOpts { output = Just StdOutput } -> L.hPutStr stdout bs
|
|
||||||
GenJSONOpts { output = Just (FileOutput file) } ->
|
|
||||||
L.writeFile file bs
|
|
||||||
ValidateJSON vopts -> case vopts of
|
|
||||||
ValidateJSONOpts { input = Nothing } ->
|
|
||||||
L.getContents >>= valAndExit validate
|
|
||||||
ValidateJSONOpts { input = Just StdInput } ->
|
|
||||||
L.getContents >>= valAndExit validate
|
|
||||||
ValidateJSONOpts { input = Just (FileInput file) } ->
|
|
||||||
L.readFile file >>= valAndExit validate
|
|
||||||
ValidateTarballs vopts -> case vopts of
|
|
||||||
ValidateJSONOpts { input = Nothing } ->
|
|
||||||
L.getContents >>= valAndExit validateTarballs
|
|
||||||
ValidateJSONOpts { input = Just StdInput } ->
|
|
||||||
L.getContents >>= valAndExit validateTarballs
|
|
||||||
ValidateJSONOpts { input = Just (FileInput file) } ->
|
|
||||||
L.readFile file >>= valAndExit validateTarballs
|
|
||||||
pure ()
|
pure ()
|
||||||
|
|
||||||
where
|
where
|
||||||
|
withValidateYamlOpts vopts f = case vopts of
|
||||||
|
ValidateYAMLOpts { vInput = Nothing } ->
|
||||||
|
B.getContents >>= valAndExit f
|
||||||
|
ValidateYAMLOpts { vInput = Just StdInput } ->
|
||||||
|
B.getContents >>= valAndExit f
|
||||||
|
ValidateYAMLOpts { vInput = Just (FileInput file) } ->
|
||||||
|
B.readFile file >>= valAndExit f
|
||||||
valAndExit f contents = do
|
valAndExit f contents = do
|
||||||
av <- case eitherDecode contents of
|
(GHCupInfo _ av gt) <- case Y.decodeEither' contents of
|
||||||
Right r -> pure r
|
Right r -> pure r
|
||||||
Left e -> die (color Red $ show e)
|
Left e -> die (color Red $ show e)
|
||||||
myLoggerT (LoggerConfig True (B.hPut stdout) (\_ -> pure ())) (f av)
|
myLoggerT (LoggerConfig True (B.hPut stdout) (\_ -> pure ())) (f av gt)
|
||||||
>>= exitWith
|
>>= exitWith
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,29 @@
|
|||||||
{-# LANGUAGE TemplateHaskell #-}
|
{-# LANGUAGE CPP #-}
|
||||||
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
{-# LANGUAGE TypeApplications #-}
|
||||||
|
{-# LANGUAGE ViewPatterns #-}
|
||||||
|
|
||||||
module Validate where
|
module Validate where
|
||||||
|
|
||||||
import GHCup
|
import GHCup
|
||||||
import GHCup.Download
|
import GHCup.Download
|
||||||
|
import GHCup.Errors
|
||||||
|
import GHCup.Platform
|
||||||
import GHCup.Types
|
import GHCup.Types
|
||||||
|
import GHCup.Types.Optics
|
||||||
|
import GHCup.Utils
|
||||||
import GHCup.Utils.Logger
|
import GHCup.Utils.Logger
|
||||||
|
import GHCup.Utils.Version.QQ
|
||||||
|
|
||||||
|
#if defined(TAR)
|
||||||
|
import qualified Codec.Archive.Tar as Tar
|
||||||
|
#else
|
||||||
|
import Codec.Archive
|
||||||
|
#endif
|
||||||
|
import Control.Applicative
|
||||||
import Control.Exception.Safe
|
import Control.Exception.Safe
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
import Control.Monad.IO.Class
|
import Control.Monad.IO.Class
|
||||||
@@ -19,17 +34,24 @@ import Control.Monad.Trans.Reader ( runReaderT )
|
|||||||
import Control.Monad.Trans.Resource ( runResourceT
|
import Control.Monad.Trans.Resource ( runResourceT
|
||||||
, MonadUnliftIO
|
, MonadUnliftIO
|
||||||
)
|
)
|
||||||
|
import Data.Containers.ListUtils ( nubOrd )
|
||||||
import Data.IORef
|
import Data.IORef
|
||||||
import Data.List
|
import Data.List
|
||||||
import Data.String.Interpolate
|
import Data.String.Interpolate
|
||||||
import Data.Versions
|
import Data.Versions
|
||||||
import Haskus.Utils.Variant.Excepts
|
import Haskus.Utils.Variant.Excepts
|
||||||
import Optics
|
import Optics
|
||||||
|
import System.FilePath
|
||||||
import System.Exit
|
import System.Exit
|
||||||
import System.IO
|
import System.IO
|
||||||
|
import Text.ParserCombinators.ReadP
|
||||||
|
import Text.PrettyPrint.HughesPJClass ( prettyShow )
|
||||||
|
import Text.Regex.Posix
|
||||||
|
|
||||||
import qualified Data.ByteString as B
|
import qualified Data.ByteString as B
|
||||||
import qualified Data.Map.Strict as M
|
import qualified Data.Map.Strict as M
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Data.Version as V
|
||||||
|
|
||||||
|
|
||||||
data ValidationError = InternalError String
|
data ValidationError = InternalError String
|
||||||
@@ -46,11 +68,12 @@ addError = do
|
|||||||
|
|
||||||
validate :: (Monad m, MonadLogger m, MonadThrow m, MonadIO m, MonadUnliftIO m)
|
validate :: (Monad m, MonadLogger m, MonadThrow m, MonadIO m, MonadUnliftIO m)
|
||||||
=> GHCupDownloads
|
=> GHCupDownloads
|
||||||
|
-> M.Map GlobalTool DownloadInfo
|
||||||
-> m ExitCode
|
-> m ExitCode
|
||||||
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
|
||||||
@@ -59,10 +82,11 @@ validate dls = do
|
|||||||
forM_ (M.toList dls) $ \(t, versions) ->
|
forM_ (M.toList dls) $ \(t, versions) ->
|
||||||
forM_ (M.toList versions) $ \(v, vi) ->
|
forM_ (M.toList versions) $ \(v, vi) ->
|
||||||
forM_ (M.toList $ _viArch vi) $ \(arch, pspecs) -> do
|
forM_ (M.toList $ _viArch vi) $ \(arch, pspecs) -> do
|
||||||
checkHasRequiredPlatforms t v arch (M.keys pspecs)
|
checkHasRequiredPlatforms t v (_viTags vi) arch (M.keys pspecs)
|
||||||
|
|
||||||
checkGHCisSemver
|
checkGHCVerIsValid
|
||||||
forM_ (M.toList dls) $ \(t, _) -> checkMandatoryTags t
|
forM_ (M.toList dls) $ \(t, _) -> checkMandatoryTags t
|
||||||
|
_ <- checkGHCHasBaseVersion
|
||||||
|
|
||||||
-- exit
|
-- exit
|
||||||
e <- liftIO $ readIORef ref
|
e <- liftIO $ readIORef ref
|
||||||
@@ -72,20 +96,37 @@ validate dls = do
|
|||||||
lift $ $(logInfo) [i|All good|]
|
lift $ $(logInfo) [i|All good|]
|
||||||
pure ExitSuccess
|
pure ExitSuccess
|
||||||
where
|
where
|
||||||
checkHasRequiredPlatforms t v arch pspecs = do
|
checkHasRequiredPlatforms t v tags arch pspecs = do
|
||||||
let v' = prettyVer v
|
let v' = prettyVer v
|
||||||
when (not $ any (== Linux UnknownLinux) pspecs) $ do
|
arch' = prettyShow arch
|
||||||
|
when (notElem (Linux UnknownLinux) pspecs) $ do
|
||||||
lift $ $(logError)
|
lift $ $(logError)
|
||||||
[i|Linux UnknownLinux missing for for #{t} #{v'} #{arch}|]
|
[i|Linux UnknownLinux missing for for #{t} #{v'} #{arch'}|]
|
||||||
addError
|
addError
|
||||||
when ((not $ any (== Darwin) pspecs) && arch == A_64) $ do
|
when ((notElem Darwin pspecs) && arch == A_64) $ do
|
||||||
lift $ $(logError) [i|Darwin missing for #{t} #{v'} #{arch}|]
|
lift $ $(logError) [i|Darwin missing for #{t} #{v'} #{arch'}|]
|
||||||
addError
|
addError
|
||||||
when ((not $ any (== FreeBSD) pspecs) && arch == A_64) $ lift $ $(logWarn)
|
when ((notElem FreeBSD pspecs) && arch == A_64) $ lift $ $(logWarn)
|
||||||
[i|FreeBSD missing for #{t} #{v'} #{arch}|]
|
[i|FreeBSD missing for #{t} #{v'} #{arch'}|]
|
||||||
|
when (notElem Windows pspecs && arch == A_64) $ do
|
||||||
|
lift $ $(logError)
|
||||||
|
[i|Windows missing for for #{t} #{v'} #{arch'}|]
|
||||||
|
addError
|
||||||
|
|
||||||
|
-- alpine needs to be set explicitly, because
|
||||||
|
-- we cannot assume that "Linux UnknownLinux" runs on Alpine
|
||||||
|
-- (although it could be static)
|
||||||
|
when (notElem (Linux Alpine) pspecs) $
|
||||||
|
case t of
|
||||||
|
GHCup | arch `elem` [A_64, A_32] -> lift ($(logError) [i|Linux Alpine missing for #{t} #{v'} #{arch}|]) >> addError
|
||||||
|
Cabal | v > [vver|2.4.1.0|]
|
||||||
|
, arch `elem` [A_64, A_32] -> lift ($(logError) [i|Linux Alpine missing for #{t} #{v'} #{arch'}|]) >> addError
|
||||||
|
GHC | Latest `elem` tags || Recommended `elem` tags
|
||||||
|
, arch `elem` [A_64, A_32] -> lift ($(logError) [i|Linux Alpine missing for #{t} #{v'} #{arch'}|])
|
||||||
|
_ -> lift $ $(logWarn) [i|Linux Alpine missing for #{t} #{v'} #{arch'}|]
|
||||||
|
|
||||||
checkUniqueTags tool = do
|
checkUniqueTags tool = do
|
||||||
let allTags = join $ fmap snd $ availableToolVersions dls tool
|
let allTags = join $ M.elems $ availableToolVersions dls tool
|
||||||
let nonUnique =
|
let nonUnique =
|
||||||
fmap fst
|
fmap fst
|
||||||
. filter (\(_, b) -> not b)
|
. filter (\(_, b) -> not b)
|
||||||
@@ -93,7 +134,7 @@ validate dls = do
|
|||||||
(\case
|
(\case
|
||||||
[] -> throwM $ InternalError "empty inner list"
|
[] -> throwM $ InternalError "empty inner list"
|
||||||
(t : ts) ->
|
(t : ts) ->
|
||||||
pure $ (t, ) $ if isUniqueTag t then ts == [] else True
|
pure $ (t, ) (not (isUniqueTag t) || null ts)
|
||||||
)
|
)
|
||||||
. group
|
. group
|
||||||
. sort
|
. sort
|
||||||
@@ -105,26 +146,47 @@ validate dls = do
|
|||||||
lift $ $(logError) [i|Tags not unique for #{tool}: #{xs}|]
|
lift $ $(logError) [i|Tags not unique for #{tool}: #{xs}|]
|
||||||
addError
|
addError
|
||||||
where
|
where
|
||||||
isUniqueTag Latest = True
|
isUniqueTag Latest = True
|
||||||
isUniqueTag Recommended = True
|
isUniqueTag Recommended = True
|
||||||
|
isUniqueTag Old = False
|
||||||
|
isUniqueTag Prerelease = False
|
||||||
|
isUniqueTag (Base _) = False
|
||||||
|
isUniqueTag (UnknownTag _) = False
|
||||||
|
|
||||||
checkGHCisSemver = do
|
checkGHCVerIsValid = do
|
||||||
let ghcVers = toListOf (ix GHC % to M.keys % folded) dls
|
let ghcVers = toListOf (ix GHC % to M.keys % folded) dls
|
||||||
forM_ ghcVers $ \v -> case semver (prettyVer v) of
|
forM_ ghcVers $ \v ->
|
||||||
Left _ -> do
|
case [ x | (x,"") <- readP_to_S V.parseVersion (T.unpack . prettyVer $ v) ] of
|
||||||
lift $ $(logError) [i|GHC version #{v} is not valid semver|]
|
[_] -> pure ()
|
||||||
addError
|
_ -> do
|
||||||
Right _ -> pure ()
|
lift $ $(logError) [i|GHC version #{v} is not valid |]
|
||||||
|
addError
|
||||||
|
|
||||||
-- a tool must have at least one of each mandatory tags
|
-- a tool must have at least one of each mandatory tags
|
||||||
checkMandatoryTags tool = do
|
checkMandatoryTags tool = do
|
||||||
let allTags = join $ fmap snd $ availableToolVersions dls tool
|
let allTags = join $ M.elems $ availableToolVersions dls tool
|
||||||
forM_ [Latest, Recommended] $ \t -> case elem t allTags of
|
forM_ [Latest, Recommended] $ \t -> case elem t allTags of
|
||||||
False -> do
|
False -> do
|
||||||
lift $ $(logError) [i|Tag #{t} missing from #{tool}|]
|
lift $ $(logError) [i|Tag #{t} missing from #{tool}|]
|
||||||
addError
|
addError
|
||||||
True -> pure ()
|
True -> pure ()
|
||||||
|
|
||||||
|
-- all GHC versions must have a base tag
|
||||||
|
checkGHCHasBaseVersion = do
|
||||||
|
let allTags = M.toList $ availableToolVersions dls GHC
|
||||||
|
forM allTags $ \(ver, tags) -> case any isBase tags of
|
||||||
|
False -> do
|
||||||
|
lift $ $(logError) [i|Base tag missing from GHC ver #{ver}|]
|
||||||
|
addError
|
||||||
|
True -> pure ()
|
||||||
|
|
||||||
|
isBase (Base _) = True
|
||||||
|
isBase _ = False
|
||||||
|
|
||||||
|
data TarballFilter = TarballFilter
|
||||||
|
{ tfTool :: Either GlobalTool (Maybe Tool)
|
||||||
|
, tfVersion :: Regex
|
||||||
|
}
|
||||||
|
|
||||||
validateTarballs :: ( Monad m
|
validateTarballs :: ( Monad m
|
||||||
, MonadLogger m
|
, MonadLogger m
|
||||||
@@ -132,24 +194,23 @@ validateTarballs :: ( Monad m
|
|||||||
, MonadIO m
|
, MonadIO m
|
||||||
, MonadUnliftIO m
|
, MonadUnliftIO m
|
||||||
, MonadMask m
|
, MonadMask m
|
||||||
|
, Alternative m
|
||||||
|
, MonadFail m
|
||||||
)
|
)
|
||||||
=> GHCupDownloads
|
=> TarballFilter
|
||||||
|
-> GHCupDownloads
|
||||||
|
-> M.Map GlobalTool DownloadInfo
|
||||||
-> m ExitCode
|
-> m ExitCode
|
||||||
validateTarballs dls = do
|
validateTarballs (TarballFilter etool versionRegex) dls gt = do
|
||||||
ref <- liftIO $ newIORef 0
|
ref <- liftIO $ newIORef 0
|
||||||
|
|
||||||
flip runReaderT ref $ do
|
flip runReaderT ref $ do
|
||||||
-- download/verify all binary tarballs
|
-- download/verify all tarballs
|
||||||
let
|
let dlis = either (const []) (\tool -> nubOrd $ dls ^.. each %& indices (maybe (const True) (==) tool) %> each %& indices (matchTest versionRegex . T.unpack . prettyVer) % (viSourceDL % _Just `summing` viArch % each % each % each)) etool
|
||||||
dlbis = nub $ join $ (M.elems dls) <&> \versions ->
|
let gdlis = nubOrd $ gt ^.. each
|
||||||
join $ (M.elems versions) <&> \vi ->
|
let allDls = either (const gdlis) (const dlis) etool
|
||||||
join $ (M.elems $ _viArch vi) <&> \pspecs ->
|
when (null allDls) $ $(logError) [i|no tarballs selected by filter|] *> addError
|
||||||
join $ (M.elems pspecs) <&> \pverspecs -> (M.elems pverspecs)
|
forM_ allDls downloadAll
|
||||||
forM_ dlbis $ downloadAll
|
|
||||||
|
|
||||||
let dlsrc = nub $ join $ (M.elems dls) <&> \versions ->
|
|
||||||
join $ (M.elems versions) <&> maybe [] (: []) . _viSourceDL
|
|
||||||
forM_ dlsrc $ downloadAll
|
|
||||||
|
|
||||||
-- exit
|
-- exit
|
||||||
e <- liftIO $ readIORef ref
|
e <- liftIO $ readIORef ref
|
||||||
@@ -160,22 +221,77 @@ validateTarballs dls = do
|
|||||||
pure ExitSuccess
|
pure ExitSuccess
|
||||||
|
|
||||||
where
|
where
|
||||||
|
runLogger = myLoggerT LoggerConfig { lcPrintDebug = True
|
||||||
|
, colorOutter = B.hPut stderr
|
||||||
|
, rawOutter = \_ -> pure ()
|
||||||
|
}
|
||||||
downloadAll dli = do
|
downloadAll dli = do
|
||||||
let settings = Settings True False
|
dirs <- liftIO getDirs
|
||||||
let runLogger = myLoggerT LoggerConfig { lcPrintDebug = True
|
|
||||||
, colorOutter = B.hPut stderr
|
pfreq <- (
|
||||||
, rawOutter = (\_ -> pure ())
|
runLogger . runE @'[NoCompatiblePlatform, NoCompatibleArch, DistroNotFound] . liftE $ platformRequest
|
||||||
}
|
) >>= \case
|
||||||
|
VRight r -> pure r
|
||||||
|
VLeft e -> do
|
||||||
|
lift $ runLogger
|
||||||
|
($(logError) $ T.pack $ prettyShow e)
|
||||||
|
liftIO $ exitWith (ExitFailure 2)
|
||||||
|
|
||||||
|
let appstate = AppState (Settings True False Never Curl False GHCupURL) dirs defaultKeyBindings (GHCupInfo mempty mempty mempty) pfreq
|
||||||
|
|
||||||
r <-
|
r <-
|
||||||
runLogger
|
runLogger
|
||||||
. flip runReaderT settings
|
. flip runReaderT appstate
|
||||||
. runResourceT
|
. runResourceT
|
||||||
. runE
|
. runE @'[DigestError
|
||||||
$ downloadCached dli Nothing
|
, DownloadFailed
|
||||||
|
, UnknownArchive
|
||||||
|
#if defined(TAR)
|
||||||
|
, Tar.FormatError
|
||||||
|
#else
|
||||||
|
, ArchiveResult
|
||||||
|
#endif
|
||||||
|
]
|
||||||
|
$ do
|
||||||
|
case etool of
|
||||||
|
Right (Just GHCup) -> do
|
||||||
|
tmpUnpack <- lift mkGhcupTmpDir
|
||||||
|
_ <- liftE $ download (settings appstate) dli tmpUnpack Nothing
|
||||||
|
pure Nothing
|
||||||
|
Right _ -> do
|
||||||
|
p <- liftE $ downloadCached (settings appstate) dirs dli Nothing
|
||||||
|
fmap (Just . head . splitDirectories . head)
|
||||||
|
. liftE
|
||||||
|
. getArchiveFiles
|
||||||
|
$ p
|
||||||
|
Left ShimGen -> do
|
||||||
|
tmpUnpack <- lift mkGhcupTmpDir
|
||||||
|
_ <- liftE $ download (settings appstate) dli tmpUnpack Nothing
|
||||||
|
pure Nothing
|
||||||
case r of
|
case r of
|
||||||
VRight _ -> pure ()
|
VRight (Just basePath) -> do
|
||||||
|
case _dlSubdir dli of
|
||||||
|
Just (RealDir prel) -> do
|
||||||
|
lift $ $(logInfo)
|
||||||
|
[i|verifying subdir: #{prel}|]
|
||||||
|
when (basePath /= prel) $ do
|
||||||
|
lift $ $(logError)
|
||||||
|
[i|Subdir doesn't match: expected "#{prel}", got "#{basePath}"|]
|
||||||
|
addError
|
||||||
|
Just (RegexDir regexString) -> do
|
||||||
|
lift $ $(logInfo)
|
||||||
|
[i|verifying subdir (regex): #{regexString}|]
|
||||||
|
let regex = makeRegexOpts
|
||||||
|
compIgnoreCase
|
||||||
|
execBlank
|
||||||
|
regexString
|
||||||
|
when (not (match regex basePath)) $ do
|
||||||
|
lift $ $(logError)
|
||||||
|
[i|Subdir doesn't match: expected regex "#{regexString}", got "#{basePath}"|]
|
||||||
|
addError
|
||||||
|
Nothing -> pure ()
|
||||||
|
VRight Nothing -> pure ()
|
||||||
VLeft e -> do
|
VLeft e -> do
|
||||||
lift $ $(logError)
|
lift $ $(logError)
|
||||||
[i|Could not download (or verify hash) of #{dli}, Error was: #{e}|]
|
[i|Could not download (or verify hash) of #{dli}, Error was: #{prettyShow e}|]
|
||||||
addError
|
addError
|
||||||
|
|||||||
643
app/ghcup/BrickMain.hs
Normal file
643
app/ghcup/BrickMain.hs
Normal file
@@ -0,0 +1,643 @@
|
|||||||
|
{-# LANGUAGE CPP #-}
|
||||||
|
{-# LANGUAGE DataKinds #-}
|
||||||
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
{-# LANGUAGE TypeApplications #-}
|
||||||
|
{-# LANGUAGE ViewPatterns #-}
|
||||||
|
{-# LANGUAGE RankNTypes #-}
|
||||||
|
|
||||||
|
module BrickMain where
|
||||||
|
|
||||||
|
import GHCup
|
||||||
|
import GHCup.Download
|
||||||
|
import GHCup.Errors
|
||||||
|
import GHCup.Types
|
||||||
|
import GHCup.Utils
|
||||||
|
import GHCup.Utils.Prelude ( decUTF8Safe )
|
||||||
|
import GHCup.Utils.File
|
||||||
|
import GHCup.Utils.Logger
|
||||||
|
|
||||||
|
import Brick
|
||||||
|
import Brick.Widgets.Border
|
||||||
|
import Brick.Widgets.Border.Style
|
||||||
|
import Brick.Widgets.Center
|
||||||
|
import Brick.Widgets.List ( listSelectedFocusedAttr
|
||||||
|
, listSelectedAttr
|
||||||
|
, listAttr
|
||||||
|
)
|
||||||
|
#if !defined(TAR)
|
||||||
|
import Codec.Archive
|
||||||
|
#endif
|
||||||
|
import Control.Exception.Safe
|
||||||
|
import Control.Monad.Logger
|
||||||
|
import Control.Monad.Reader
|
||||||
|
import Control.Monad.Trans.Except
|
||||||
|
import Control.Monad.Trans.Resource
|
||||||
|
import Data.Bool
|
||||||
|
import Data.Functor
|
||||||
|
import Data.List
|
||||||
|
import Data.Maybe
|
||||||
|
import Data.IORef
|
||||||
|
import Data.String.Interpolate
|
||||||
|
import Data.Vector ( Vector
|
||||||
|
, (!?)
|
||||||
|
)
|
||||||
|
import Data.Versions hiding ( str )
|
||||||
|
import Haskus.Utils.Variant.Excepts
|
||||||
|
import Prelude hiding ( appendFile )
|
||||||
|
import System.Environment
|
||||||
|
import System.Exit
|
||||||
|
import System.IO.Unsafe
|
||||||
|
import Text.PrettyPrint.HughesPJClass ( prettyShow )
|
||||||
|
import URI.ByteString
|
||||||
|
|
||||||
|
import qualified GHCup.Types as GT
|
||||||
|
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Graphics.Vty as Vty
|
||||||
|
import qualified Data.Vector as V
|
||||||
|
|
||||||
|
|
||||||
|
hiddenTools :: [Tool]
|
||||||
|
hiddenTools = [Stack]
|
||||||
|
|
||||||
|
|
||||||
|
data BrickData = BrickData
|
||||||
|
{ lr :: [ListResult]
|
||||||
|
}
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
data BrickSettings = BrickSettings
|
||||||
|
{ showAllVersions :: Bool
|
||||||
|
, showAllTools :: 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 :: KeyBindings
|
||||||
|
-> [ ( Vty.Key
|
||||||
|
, BrickSettings -> String
|
||||||
|
, BrickState -> EventM n (Next BrickState)
|
||||||
|
)
|
||||||
|
]
|
||||||
|
keyHandlers KeyBindings {..} =
|
||||||
|
[ (bQuit, const "Quit" , halt)
|
||||||
|
, (bInstall, const "Install" , withIOAction install')
|
||||||
|
, (bUninstall, const "Uninstall", withIOAction del')
|
||||||
|
, (bSet, const "Set" , withIOAction ((liftIO .) . set'))
|
||||||
|
, (bChangelog, const "ChangeLog", withIOAction changelog')
|
||||||
|
, ( bShowAllVersions
|
||||||
|
, \BrickSettings {..} ->
|
||||||
|
if showAllVersions then "Don't show all versions" else "Show all versions"
|
||||||
|
, hideShowHandler (not . showAllVersions) showAllTools
|
||||||
|
)
|
||||||
|
, ( bShowAllTools
|
||||||
|
, \BrickSettings {..} ->
|
||||||
|
if showAllTools then "Don't show all tools" else "Show all tools"
|
||||||
|
, hideShowHandler showAllVersions (not . showAllTools)
|
||||||
|
)
|
||||||
|
, (bUp, const "Up", \BrickState {..} -> continue BrickState{ appState = moveCursor 1 appState Up, .. })
|
||||||
|
, (bDown, const "Down", \BrickState {..} -> continue BrickState{ appState = moveCursor 1 appState Down, .. })
|
||||||
|
]
|
||||||
|
where
|
||||||
|
hideShowHandler f p BrickState{..} =
|
||||||
|
let newAppSettings = appSettings { showAllVersions = f appSettings , showAllTools = p appSettings }
|
||||||
|
newInternalState = constructList appData newAppSettings (Just appState)
|
||||||
|
in continue (BrickState appData newAppSettings newInternalState appKeys)
|
||||||
|
|
||||||
|
|
||||||
|
showKey :: Vty.Key -> String
|
||||||
|
showKey (Vty.KChar c) = [c]
|
||||||
|
showKey Vty.KUp = "↑"
|
||||||
|
showKey Vty.KDown = "↓"
|
||||||
|
showKey key = tail (show key)
|
||||||
|
|
||||||
|
|
||||||
|
ui :: AttrMap -> BrickState -> Widget String
|
||||||
|
ui dimAttrs BrickState{ appSettings = as@BrickSettings{}, ..}
|
||||||
|
= padBottom Max
|
||||||
|
( withBorderStyle unicode
|
||||||
|
$ borderWithLabel (str "GHCup")
|
||||||
|
(center (header <=> hBorder <=> renderList' appState))
|
||||||
|
)
|
||||||
|
<=> footer
|
||||||
|
|
||||||
|
where
|
||||||
|
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
|
||||||
|
| lSet -> (withAttr "set" $ str "✔✔")
|
||||||
|
| lInstalled -> (withAttr "installed" $ str "✓ ")
|
||||||
|
| otherwise -> (withAttr "not-installed" $ str "✗ ")
|
||||||
|
ver = case lCross of
|
||||||
|
Nothing -> T.unpack . prettyVer $ lVer
|
||||||
|
Just c -> T.unpack (c <> "-" <> prettyVer lVer)
|
||||||
|
dim
|
||||||
|
| lNoBindist && not lInstalled
|
||||||
|
&& not b -- TODO: overloading dim and active ignores active
|
||||||
|
-- so we hack around it here
|
||||||
|
= updateAttrMap (const dimAttrs) . withAttr "no-bindist"
|
||||||
|
| otherwise = id
|
||||||
|
hooray
|
||||||
|
| elem Latest lTag && not lInstalled =
|
||||||
|
withAttr "hooray"
|
||||||
|
| otherwise = id
|
||||||
|
active = if b then forceAttr "active" else id
|
||||||
|
in hooray $ active $ dim
|
||||||
|
( marks
|
||||||
|
<+> padLeft (Pad 2)
|
||||||
|
( minHSize 6
|
||||||
|
(printTool lTool)
|
||||||
|
)
|
||||||
|
<+> minHSize 15 (str ver)
|
||||||
|
<+> (let l = catMaybes . fmap printTag $ sort lTag
|
||||||
|
in padLeft (Pad 1) $ minHSize 25 $ if null l
|
||||||
|
then emptyWidget
|
||||||
|
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 = Just $ withAttr "recommended" $ str "recommended"
|
||||||
|
printTag Latest = Just $ withAttr "latest" $ str "latest"
|
||||||
|
printTag Prerelease = Just $ withAttr "prerelease" $ str "prerelease"
|
||||||
|
printTag (Base pvp'') = Just $ str ("base-" ++ T.unpack (prettyPVP pvp''))
|
||||||
|
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"
|
||||||
|
printTool Stack = str "Stack"
|
||||||
|
|
||||||
|
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 s' = hLimit s' . vLimit 1 . (<+> fill ' ')
|
||||||
|
|
||||||
|
|
||||||
|
app :: AttrMap -> AttrMap -> App BrickState e String
|
||||||
|
app attrs dimAttrs =
|
||||||
|
App { appDraw = \st -> [ui dimAttrs st]
|
||||||
|
, appHandleEvent = eventHandler
|
||||||
|
, appStartEvent = return
|
||||||
|
, appAttrMap = const attrs
|
||||||
|
, appChooseCursor = neverShowCursor
|
||||||
|
}
|
||||||
|
|
||||||
|
defaultAttributes :: Bool -> AttrMap
|
||||||
|
defaultAttributes no_color = attrMap
|
||||||
|
Vty.defAttr
|
||||||
|
[ ("active" , Vty.defAttr `withBackColor` Vty.blue)
|
||||||
|
, ("not-installed", Vty.defAttr `withForeColor` Vty.red)
|
||||||
|
, ("set" , Vty.defAttr `withForeColor` Vty.green)
|
||||||
|
, ("installed" , Vty.defAttr `withForeColor` Vty.green)
|
||||||
|
, ("recommended" , Vty.defAttr `withForeColor` Vty.green)
|
||||||
|
, ("hls-powered" , Vty.defAttr `withForeColor` Vty.green)
|
||||||
|
, ("latest" , Vty.defAttr `withForeColor` Vty.yellow)
|
||||||
|
, ("prerelease" , Vty.defAttr `withForeColor` Vty.red)
|
||||||
|
, ("compiled" , Vty.defAttr `withForeColor` Vty.blue)
|
||||||
|
, ("stray" , Vty.defAttr `withForeColor` Vty.blue)
|
||||||
|
, ("help" , Vty.defAttr `withStyle` Vty.italic)
|
||||||
|
, ("hooray" , Vty.defAttr `withForeColor` Vty.brightWhite)
|
||||||
|
]
|
||||||
|
where
|
||||||
|
withForeColor | no_color = const
|
||||||
|
| otherwise = Vty.withForeColor
|
||||||
|
|
||||||
|
withBackColor | no_color = \attr _ -> attr `Vty.withStyle` Vty.reverseVideo
|
||||||
|
| otherwise = Vty.withBackColor
|
||||||
|
|
||||||
|
withStyle = Vty.withStyle
|
||||||
|
|
||||||
|
dimAttributes :: Bool -> AttrMap
|
||||||
|
dimAttributes no_color = attrMap
|
||||||
|
(Vty.defAttr `Vty.withStyle` Vty.dim)
|
||||||
|
[ ("active" , Vty.defAttr `withBackColor` Vty.blue) -- has no effect ??
|
||||||
|
, ("no-bindist", Vty.defAttr `Vty.withStyle` Vty.dim)
|
||||||
|
]
|
||||||
|
where
|
||||||
|
withBackColor | no_color = \attr _ -> attr `Vty.withStyle` Vty.reverseVideo
|
||||||
|
| otherwise = Vty.withBackColor
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
|
||||||
|
moveCursor :: Int -> BrickInternalState -> Direction -> BrickInternalState
|
||||||
|
moveCursor steps ais@BrickInternalState{..} direction =
|
||||||
|
let newIx = if direction == Down then ix + steps else ix - steps
|
||||||
|
in case clr !? newIx of
|
||||||
|
Just _ -> BrickInternalState { ix = newIx, .. }
|
||||||
|
Nothing -> ais
|
||||||
|
|
||||||
|
|
||||||
|
-- | 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.
|
||||||
|
withIOAction :: (BrickState
|
||||||
|
-> (Int, ListResult)
|
||||||
|
-> ReaderT AppState IO (Either String a))
|
||||||
|
-> BrickState
|
||||||
|
-> EventM n (Next BrickState)
|
||||||
|
withIOAction action as = case listSelectedElement' (appState as) of
|
||||||
|
Nothing -> continue as
|
||||||
|
Just (ix, e) -> do
|
||||||
|
suspendAndResume $ do
|
||||||
|
settings <- readIORef settings'
|
||||||
|
flip runReaderT settings $ action as (ix, e) >>= \case
|
||||||
|
Left err -> liftIO $ putStrLn ("Error: " <> err)
|
||||||
|
Right _ -> liftIO $ putStrLn "Success"
|
||||||
|
getAppData Nothing >>= \case
|
||||||
|
Right data' -> do
|
||||||
|
putStrLn "Press enter to continue"
|
||||||
|
_ <- getLine
|
||||||
|
pure (updateList data' as)
|
||||||
|
Left err -> throwIO $ userError err
|
||||||
|
|
||||||
|
|
||||||
|
-- | Update app data and list internal state based on new evidence.
|
||||||
|
-- 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 =
|
||||||
|
replaceLR (filterVisible (showAllVersions appSettings)
|
||||||
|
(showAllTools appSettings))
|
||||||
|
(lr appD)
|
||||||
|
|
||||||
|
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 -> Bool -> ListResult -> Bool
|
||||||
|
filterVisible v t e | lInstalled e = True
|
||||||
|
| v
|
||||||
|
, not t
|
||||||
|
, not (elem (lTool e) hiddenTools) = True
|
||||||
|
| not v
|
||||||
|
, t
|
||||||
|
, not (elem Old (lTag e)) = True
|
||||||
|
| v
|
||||||
|
, t = True
|
||||||
|
| otherwise = not (elem Old (lTag e)) &&
|
||||||
|
not (elem (lTool e) hiddenTools)
|
||||||
|
|
||||||
|
|
||||||
|
install' :: (MonadReader AppState m, MonadIO m, MonadThrow m, MonadFail m, MonadMask m, MonadUnliftIO m)
|
||||||
|
=> BrickState
|
||||||
|
-> (Int, ListResult)
|
||||||
|
-> m (Either String ())
|
||||||
|
install' _ (_, ListResult {..}) = do
|
||||||
|
AppState { ghcupInfo = GHCupInfo { _ghcupDownloads = dls }} <- ask
|
||||||
|
|
||||||
|
l <- liftIO $ readIORef logger'
|
||||||
|
let runLogger = myLoggerT l
|
||||||
|
|
||||||
|
let run =
|
||||||
|
runLogger
|
||||||
|
. runResourceT
|
||||||
|
. runE
|
||||||
|
@'[ AlreadyInstalled
|
||||||
|
#if !defined(TAR)
|
||||||
|
, ArchiveResult
|
||||||
|
#endif
|
||||||
|
, UnknownArchive
|
||||||
|
, FileDoesNotExistError
|
||||||
|
, CopyError
|
||||||
|
, NoDownload
|
||||||
|
, NotInstalled
|
||||||
|
, BuildFailed
|
||||||
|
, TagNotFound
|
||||||
|
, DigestError
|
||||||
|
, DownloadFailed
|
||||||
|
, NoUpdate
|
||||||
|
, TarDirDoesNotExist
|
||||||
|
]
|
||||||
|
|
||||||
|
run (do
|
||||||
|
case lTool of
|
||||||
|
GHC -> do
|
||||||
|
let vi = getVersionInfo lVer GHC dls
|
||||||
|
liftE $ installGHCBin lVer $> vi
|
||||||
|
Cabal -> do
|
||||||
|
let vi = getVersionInfo lVer Cabal dls
|
||||||
|
liftE $ installCabalBin lVer $> vi
|
||||||
|
GHCup -> do
|
||||||
|
let vi = snd <$> getLatest dls GHCup
|
||||||
|
liftE $ upgradeGHCup Nothing False $> vi
|
||||||
|
HLS -> do
|
||||||
|
let vi = getVersionInfo lVer HLS dls
|
||||||
|
liftE $ installHLSBin lVer $> vi
|
||||||
|
Stack -> do
|
||||||
|
let vi = getVersionInfo lVer Stack dls
|
||||||
|
liftE $ installStackBin lVer $> vi
|
||||||
|
)
|
||||||
|
>>= \case
|
||||||
|
VRight vi -> do
|
||||||
|
forM_ (_viPostInstall =<< vi) $ \msg ->
|
||||||
|
myLoggerT l $ $(logInfo) msg
|
||||||
|
pure $ Right ()
|
||||||
|
VLeft (V (AlreadyInstalled _ _)) -> pure $ Right ()
|
||||||
|
VLeft (V NoUpdate) -> pure $ Right ()
|
||||||
|
VLeft e -> pure $ Left [i|#{prettyShow e}
|
||||||
|
Also check the logs in ~/.ghcup/logs|]
|
||||||
|
|
||||||
|
|
||||||
|
set' :: BrickState -> (Int, ListResult) -> IO (Either String ())
|
||||||
|
set' _ (_, ListResult {..}) = do
|
||||||
|
settings <- readIORef settings'
|
||||||
|
l <- readIORef logger'
|
||||||
|
let runLogger = myLoggerT l
|
||||||
|
|
||||||
|
let run =
|
||||||
|
runLogger
|
||||||
|
. flip runReaderT settings
|
||||||
|
. runE @'[FileDoesNotExistError , NotInstalled , TagNotFound]
|
||||||
|
|
||||||
|
run (do
|
||||||
|
case lTool of
|
||||||
|
GHC -> liftE $ setGHC (GHCTargetVersion lCross lVer) SetGHCOnly $> ()
|
||||||
|
Cabal -> liftE $ setCabal lVer $> ()
|
||||||
|
HLS -> liftE $ setHLS lVer $> ()
|
||||||
|
Stack -> liftE $ setStack lVer $> ()
|
||||||
|
GHCup -> pure ()
|
||||||
|
)
|
||||||
|
>>= \case
|
||||||
|
VRight _ -> pure $ Right ()
|
||||||
|
VLeft e -> pure $ Left (prettyShow e)
|
||||||
|
|
||||||
|
|
||||||
|
del' :: (MonadReader AppState m, MonadIO m, MonadFail m, MonadMask m, MonadUnliftIO m)
|
||||||
|
=> BrickState
|
||||||
|
-> (Int, ListResult)
|
||||||
|
-> m (Either String ())
|
||||||
|
del' _ (_, ListResult {..}) = do
|
||||||
|
AppState { ghcupInfo = GHCupInfo { _ghcupDownloads = dls }} <- ask
|
||||||
|
|
||||||
|
l <- liftIO $ readIORef logger'
|
||||||
|
let runLogger = myLoggerT l
|
||||||
|
let run = myLoggerT l . runE @'[NotInstalled]
|
||||||
|
|
||||||
|
run (do
|
||||||
|
let vi = getVersionInfo lVer lTool dls
|
||||||
|
case lTool of
|
||||||
|
GHC -> liftE $ rmGHCVer (GHCTargetVersion lCross lVer) $> vi
|
||||||
|
Cabal -> liftE $ rmCabalVer lVer $> vi
|
||||||
|
HLS -> liftE $ rmHLSVer lVer $> vi
|
||||||
|
Stack -> liftE $ rmStackVer lVer $> vi
|
||||||
|
GHCup -> pure Nothing
|
||||||
|
)
|
||||||
|
>>= \case
|
||||||
|
VRight vi -> do
|
||||||
|
forM_ (join $ fmap _viPostRemove vi) $ \msg ->
|
||||||
|
runLogger $ $(logInfo) msg
|
||||||
|
pure $ Right ()
|
||||||
|
VLeft e -> pure $ Left (prettyShow e)
|
||||||
|
|
||||||
|
|
||||||
|
changelog' :: (MonadReader AppState m, MonadIO m)
|
||||||
|
=> BrickState
|
||||||
|
-> (Int, ListResult)
|
||||||
|
-> m (Either String ())
|
||||||
|
changelog' _ (_, ListResult {..}) = do
|
||||||
|
AppState { pfreq, ghcupInfo = GHCupInfo { _ghcupDownloads = dls }} <- ask
|
||||||
|
case getChangeLog dls lTool (Left lVer) of
|
||||||
|
Nothing -> pure $ Left
|
||||||
|
[i|Could not find ChangeLog for #{lTool}, version #{prettyVer lVer}|]
|
||||||
|
Just uri -> do
|
||||||
|
let cmd = case _rPlatform pfreq of
|
||||||
|
Darwin -> "open"
|
||||||
|
Linux _ -> "xdg-open"
|
||||||
|
FreeBSD -> "xdg-open"
|
||||||
|
Windows -> "start"
|
||||||
|
exec cmd [T.unpack $ decUTF8Safe $ serializeURIRef' uri] Nothing Nothing >>= \case
|
||||||
|
Right _ -> pure $ Right ()
|
||||||
|
Left e -> pure $ Left $ prettyShow e
|
||||||
|
|
||||||
|
|
||||||
|
settings' :: IORef AppState
|
||||||
|
{-# NOINLINE settings' #-}
|
||||||
|
settings' = unsafePerformIO $ do
|
||||||
|
dirs <- getDirs
|
||||||
|
newIORef $ AppState (Settings { cache = True
|
||||||
|
, noVerify = False
|
||||||
|
, keepDirs = Never
|
||||||
|
, downloader = Curl
|
||||||
|
, verbose = False
|
||||||
|
, urlSource = GHCupURL
|
||||||
|
, ..
|
||||||
|
})
|
||||||
|
dirs
|
||||||
|
defaultKeyBindings
|
||||||
|
(GHCupInfo mempty mempty mempty)
|
||||||
|
(PlatformRequest A_64 Darwin Nothing)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
logger' :: IORef LoggerConfig
|
||||||
|
{-# NOINLINE logger' #-}
|
||||||
|
logger' = unsafePerformIO
|
||||||
|
(newIORef $ LoggerConfig { lcPrintDebug = False
|
||||||
|
, colorOutter = \_ -> pure ()
|
||||||
|
, rawOutter = \_ -> pure ()
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
brickMain :: AppState
|
||||||
|
-> LoggerConfig
|
||||||
|
-> GHCupInfo
|
||||||
|
-> IO ()
|
||||||
|
brickMain s l gi = do
|
||||||
|
writeIORef settings' s
|
||||||
|
-- logger interpreter
|
||||||
|
writeIORef logger' l
|
||||||
|
let runLogger = myLoggerT l
|
||||||
|
|
||||||
|
no_color <- isJust <$> lookupEnv "NO_COLOR"
|
||||||
|
|
||||||
|
eAppData <- getAppData (Just gi)
|
||||||
|
case eAppData of
|
||||||
|
Right ad ->
|
||||||
|
defaultMain
|
||||||
|
(app (defaultAttributes no_color) (dimAttributes no_color))
|
||||||
|
(BrickState ad
|
||||||
|
defaultAppSettings
|
||||||
|
(constructList ad defaultAppSettings Nothing)
|
||||||
|
(keyBindings s)
|
||||||
|
|
||||||
|
)
|
||||||
|
$> ()
|
||||||
|
Left e -> do
|
||||||
|
runLogger ($(logError) [i|Error building app state: #{show e}|])
|
||||||
|
exitWith $ ExitFailure 2
|
||||||
|
|
||||||
|
|
||||||
|
defaultAppSettings :: BrickSettings
|
||||||
|
defaultAppSettings = BrickSettings { showAllVersions = False, showAllTools = False }
|
||||||
|
|
||||||
|
|
||||||
|
getGHCupInfo :: IO (Either String GHCupInfo)
|
||||||
|
getGHCupInfo = do
|
||||||
|
settings <- readIORef settings'
|
||||||
|
l <- readIORef logger'
|
||||||
|
let runLogger = myLoggerT l
|
||||||
|
|
||||||
|
r <-
|
||||||
|
runLogger
|
||||||
|
. flip runReaderT settings
|
||||||
|
. runE @'[JSONError , DownloadFailed , FileDoesNotExistError]
|
||||||
|
$ liftE
|
||||||
|
$ getDownloadsF (GT.settings settings) (GT.dirs settings)
|
||||||
|
|
||||||
|
case r of
|
||||||
|
VRight a -> pure $ Right a
|
||||||
|
VLeft e -> pure $ Left (prettyShow e)
|
||||||
|
|
||||||
|
|
||||||
|
getAppData :: Maybe GHCupInfo
|
||||||
|
-> IO (Either String BrickData)
|
||||||
|
getAppData mgi = runExceptT $ do
|
||||||
|
l <- liftIO $ readIORef logger'
|
||||||
|
let runLogger = myLoggerT l
|
||||||
|
|
||||||
|
r <- ExceptT $ maybe getGHCupInfo (pure . Right) mgi
|
||||||
|
liftIO $ modifyIORef settings' (\s -> s { ghcupInfo = r })
|
||||||
|
settings <- liftIO $ readIORef settings'
|
||||||
|
|
||||||
|
runLogger . flip runReaderT settings $ do
|
||||||
|
lV <- listVersions Nothing Nothing
|
||||||
|
pure $ BrickData (reverse lV)
|
||||||
|
|
||||||
2071
app/ghcup/Main.hs
2071
app/ghcup/Main.hs
File diff suppressed because it is too large
Load Diff
484
bootstrap-haskell
Executable file
484
bootstrap-haskell
Executable file
@@ -0,0 +1,484 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Main settings:
|
||||||
|
# * BOOTSTRAP_HASKELL_NONINTERACTIVE - any nonzero value for noninteractive installation
|
||||||
|
# * BOOTSTRAP_HASKELL_NO_UPGRADE - any nonzero value to not trigger the upgrade
|
||||||
|
# * GHCUP_USE_XDG_DIRS - any nonzero value to respect The XDG Base Directory Specification
|
||||||
|
# * BOOTSTRAP_HASKELL_VERBOSE - any nonzero value for more verbose installation
|
||||||
|
# * BOOTSTRAP_HASKELL_GHC_VERSION
|
||||||
|
# * BOOTSTRAP_HASKELL_CABAL_VERSION
|
||||||
|
|
||||||
|
# License: LGPL-3.0
|
||||||
|
|
||||||
|
|
||||||
|
# safety subshell to avoid executing anything in case this script is not downloaded properly
|
||||||
|
(
|
||||||
|
|
||||||
|
plat="$(uname -s)"
|
||||||
|
arch=$(uname -m)
|
||||||
|
ghver="0.1.15.2"
|
||||||
|
base_url="https://downloads.haskell.org/~ghcup"
|
||||||
|
|
||||||
|
case "${plat}" in
|
||||||
|
MSYS*|MINGW*)
|
||||||
|
: "${GHCUP_INSTALL_BASE_PREFIX:=/c}"
|
||||||
|
GHCUP_DIR=$(cygpath -u "${GHCUP_INSTALL_BASE_PREFIX}/ghcup")
|
||||||
|
GHCUP_BIN=$(cygpath -u "${GHCUP_INSTALL_BASE_PREFIX}/ghcup/bin")
|
||||||
|
: "${GHCUP_MSYS2:=${GHCUP_DIR}/msys64}"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
: "${GHCUP_INSTALL_BASE_PREFIX:=$HOME}"
|
||||||
|
export GHCUP_USE_XDG_DIRS
|
||||||
|
|
||||||
|
if [ -n "${GHCUP_USE_XDG_DIRS}" ] ; then
|
||||||
|
GHCUP_DIR=${XDG_DATA_HOME:=$HOME/.local/share}/ghcup
|
||||||
|
GHCUP_BIN=${XDG_BIN_HOME:=$HOME/.local/bin}
|
||||||
|
else
|
||||||
|
GHCUP_DIR=${GHCUP_INSTALL_BASE_PREFIX}/.ghcup
|
||||||
|
GHCUP_BIN=${GHCUP_INSTALL_BASE_PREFIX}/.ghcup/bin
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
: "${BOOTSTRAP_HASKELL_GHC_VERSION:=recommended}"
|
||||||
|
: "${BOOTSTRAP_HASKELL_CABAL_VERSION:=recommended}"
|
||||||
|
|
||||||
|
|
||||||
|
die() {
|
||||||
|
(>&2 printf "\\033[0;31m%s\\033[0m\\n" "$1")
|
||||||
|
exit 2
|
||||||
|
}
|
||||||
|
|
||||||
|
warn() {
|
||||||
|
case "${plat}" in
|
||||||
|
MSYS*|MINGW*)
|
||||||
|
echo -e "\\033[0;35m$1\\033[0m"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
printf "\\033[0;35m%s\\033[0m\\n" "$1"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
}
|
||||||
|
|
||||||
|
edo() {
|
||||||
|
"$@" || die "\"$*\" failed!"
|
||||||
|
}
|
||||||
|
|
||||||
|
eghcup() {
|
||||||
|
edo _eghcup "$@"
|
||||||
|
}
|
||||||
|
|
||||||
|
_eghcup() {
|
||||||
|
if [ -n "${BOOTSTRAP_HASKELL_YAML}" ] ; then
|
||||||
|
args="-s ${BOOTSTRAP_HASKELL_YAML}"
|
||||||
|
fi
|
||||||
|
if [ -z "${BOOTSTRAP_HASKELL_VERBOSE}" ] ; then
|
||||||
|
"${GHCUP_BIN}/ghcup" ${args} "$@"
|
||||||
|
else
|
||||||
|
"${GHCUP_BIN}/ghcup" ${args} --verbose "$@"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
_done() {
|
||||||
|
case "${plat}" in
|
||||||
|
MSYS*|MINGW*)
|
||||||
|
echo
|
||||||
|
echo "All done!"
|
||||||
|
echo
|
||||||
|
echo "In a new powershell or cmd.exe session, now you can..."
|
||||||
|
echo
|
||||||
|
echo "Start a simple repl via:"
|
||||||
|
echo " ghci"
|
||||||
|
echo
|
||||||
|
echo "Start a new haskell project in the current directory via:"
|
||||||
|
echo " cabal init --interactive"
|
||||||
|
echo
|
||||||
|
echo "Install other GHC versions and tools via:"
|
||||||
|
echo " ghcup list"
|
||||||
|
echo " ghcup install <tool> <version>"
|
||||||
|
echo
|
||||||
|
echo "To install system libraries and update msys2/mingw64,"
|
||||||
|
echo "open the \"Mingw haskell shell\""
|
||||||
|
echo "and the \"Mingw package management docs\""
|
||||||
|
echo "desktop shortcuts."
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo
|
||||||
|
echo "All done!"
|
||||||
|
echo
|
||||||
|
echo "To start a simple repl, run:"
|
||||||
|
echo " ghci"
|
||||||
|
echo
|
||||||
|
echo "To start a new haskell project in the current directory, run:"
|
||||||
|
echo " cabal init --interactive"
|
||||||
|
echo
|
||||||
|
echo "To install other GHC versions and tools, run:"
|
||||||
|
echo " ghcup tui"
|
||||||
|
;;
|
||||||
|
|
||||||
|
esac
|
||||||
|
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
}
|
||||||
|
|
||||||
|
download_ghcup() {
|
||||||
|
|
||||||
|
case "${plat}" in
|
||||||
|
"linux"|"Linux")
|
||||||
|
case "${arch}" in
|
||||||
|
x86_64|amd64)
|
||||||
|
# we could be in a 32bit docker container, in which
|
||||||
|
# case uname doesn't give us what we want
|
||||||
|
if [ "$(getconf LONG_BIT)" = "32" ] ; then
|
||||||
|
_url=${base_url}/${ghver}/i386-linux-ghcup-${ghver}
|
||||||
|
elif [ "$(getconf LONG_BIT)" = "64" ] ; then
|
||||||
|
_url=${base_url}/${ghver}/x86_64-linux-ghcup-${ghver}
|
||||||
|
else
|
||||||
|
die "Unknown long bit size: $(getconf LONG_BIT)"
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
i*86)
|
||||||
|
_url=${base_url}/${ghver}/i386-linux-ghcup-${ghver}
|
||||||
|
;;
|
||||||
|
armv7*)
|
||||||
|
_url=${base_url}/${ghver}/armv7-linux-ghcup-${ghver}
|
||||||
|
;;
|
||||||
|
aarch64|arm64|armv8l)
|
||||||
|
_url=${base_url}/${ghver}/aarch64-linux-ghcup-${ghver}
|
||||||
|
;;
|
||||||
|
*) die "Unknown architecture: ${arch}"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
;;
|
||||||
|
"FreeBSD"|"freebsd")
|
||||||
|
case "${arch}" in
|
||||||
|
x86_64|amd64)
|
||||||
|
;;
|
||||||
|
i*86)
|
||||||
|
die "i386 currently not supported!"
|
||||||
|
;;
|
||||||
|
*) die "Unknown architecture: ${arch}"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
_url=${base_url}/${ghver}/x86_64-portbld-freebsd-ghcup-${ghver}
|
||||||
|
;;
|
||||||
|
"Darwin"|"darwin")
|
||||||
|
case "${arch}" in
|
||||||
|
x86_64|amd64)
|
||||||
|
_url=${base_url}/${ghver}/x86_64-apple-darwin-ghcup-${ghver}
|
||||||
|
;;
|
||||||
|
aarch64|arm64|armv8l)
|
||||||
|
_url=${base_url}/${ghver}/aarch64-apple-darwin-ghcup-${ghver}
|
||||||
|
;;
|
||||||
|
i*86)
|
||||||
|
die "i386 currently not supported!"
|
||||||
|
;;
|
||||||
|
*) die "Unknown architecture: ${arch}"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
;;
|
||||||
|
MSYS*|MINGW*)
|
||||||
|
case "${arch}" in
|
||||||
|
x86_64|amd64)
|
||||||
|
_url=${base_url}/${ghver}/x86_64-mingw64-ghcup-${ghver}.exe
|
||||||
|
;;
|
||||||
|
*) die "Unknown architecture: ${arch}"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
;;
|
||||||
|
*) die "Unknown platform: ${plat}"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
case "${plat}" in
|
||||||
|
MSYS*|MINGW*)
|
||||||
|
edo curl -Lf "${_url}" > "${GHCUP_BIN}"/ghcup.exe
|
||||||
|
edo chmod +x "${GHCUP_BIN}"/ghcup.exe
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
edo curl -Lf "${_url}" > "${GHCUP_BIN}"/ghcup
|
||||||
|
edo chmod +x "${GHCUP_BIN}"/ghcup
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
edo mkdir -p "${GHCUP_DIR}"
|
||||||
|
cat <<-EOF > "${GHCUP_DIR}"/env || die "Failed to create env file"
|
||||||
|
export PATH="\$HOME/.cabal/bin:${GHCUP_BIN}:\$PATH"
|
||||||
|
EOF
|
||||||
|
# shellcheck disable=SC1090
|
||||||
|
edo . "${GHCUP_DIR}"/env
|
||||||
|
eghcup upgrade
|
||||||
|
}
|
||||||
|
|
||||||
|
adjust_bashrc() {
|
||||||
|
case $SHELL in
|
||||||
|
*/zsh) # login shell is zsh
|
||||||
|
GHCUP_PROFILE_FILE="$HOME/.zshrc"
|
||||||
|
MY_SHELL="zsh" ;;
|
||||||
|
*/bash) # login shell is bash
|
||||||
|
GHCUP_PROFILE_FILE="$HOME/.bashrc"
|
||||||
|
MY_SHELL="bash" ;;
|
||||||
|
*/sh) # login shell is sh, but might be a symlink to bash or zsh
|
||||||
|
if [ -n "${BASH}" ] ; then
|
||||||
|
GHCUP_PROFILE_FILE="$HOME/.bashrc"
|
||||||
|
MY_SHELL="bash"
|
||||||
|
elif [ -n "${ZSH_VERSION}" ] ; then
|
||||||
|
GHCUP_PROFILE_FILE="$HOME/.zshrc"
|
||||||
|
MY_SHELL="zsh"
|
||||||
|
else
|
||||||
|
return
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
*/fish) # login shell is fish
|
||||||
|
GHCUP_PROFILE_FILE="$HOME/.config/fish/config.fish"
|
||||||
|
MY_SHELL="fish" ;;
|
||||||
|
*) return ;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
|
||||||
|
warn ""
|
||||||
|
warn "Detected ${MY_SHELL} shell on your system..."
|
||||||
|
warn "If you want ghcup to automatically add the required PATH variable to \"${GHCUP_PROFILE_FILE}\""
|
||||||
|
warn ""
|
||||||
|
warn "[Y] Yes [N] No [?] Help (default is \"Y\")."
|
||||||
|
warn ""
|
||||||
|
|
||||||
|
while true; do
|
||||||
|
if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
||||||
|
read -r next_answer </dev/tty
|
||||||
|
else
|
||||||
|
next_answer="yes"
|
||||||
|
fi
|
||||||
|
|
||||||
|
case $next_answer in
|
||||||
|
[Nn]*)
|
||||||
|
return ;;
|
||||||
|
[Yy]* | "")
|
||||||
|
case $MY_SHELL in
|
||||||
|
"") break ;;
|
||||||
|
fish)
|
||||||
|
mkdir -p "${GHCUP_PROFILE_FILE%/*}"
|
||||||
|
sed -i -e '/# ghcup-env$/ s/^#*/#/' "${GHCUP_PROFILE_FILE}"
|
||||||
|
echo "set -q GHCUP_INSTALL_BASE_PREFIX[1]; or set GHCUP_INSTALL_BASE_PREFIX \$HOME ; test -f $GHCUP_DIR/env ; and set -gx PATH \$HOME/.cabal/bin $GHCUP_BIN \$PATH # ghcup-env" >> "${GHCUP_PROFILE_FILE}"
|
||||||
|
break ;;
|
||||||
|
bash)
|
||||||
|
sed -i -e '/# ghcup-env$/ s/^#*/#/' "${GHCUP_PROFILE_FILE}"
|
||||||
|
echo "[ -f \"${GHCUP_DIR}/env\" ] && source \"${GHCUP_DIR}/env\" # ghcup-env" >> "${GHCUP_PROFILE_FILE}"
|
||||||
|
case "${plat}" 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)
|
||||||
|
sed -i -e '/# ghcup-env$/ s/^#*/#/' "${GHCUP_PROFILE_FILE}"
|
||||||
|
echo "[ -f \"${GHCUP_DIR}/env\" ] && source \"${GHCUP_DIR}/env\" # ghcup-env" >> "${GHCUP_PROFILE_FILE}"
|
||||||
|
break ;;
|
||||||
|
esac
|
||||||
|
warn "OK! ${GHCUP_PROFILE_FILE} has been modified. Restart your terminal for the changes to take effect,"
|
||||||
|
warn "or type \"source ${GHCUP_DIR}/env\" to apply them in your current terminal session."
|
||||||
|
return
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "Possible choices are:"
|
||||||
|
echo
|
||||||
|
echo "Y - Yes, update my \"${GHCUP_PROFILE_FILE}\" (default)"
|
||||||
|
echo "N - No, don't mess with my configuration"
|
||||||
|
echo
|
||||||
|
echo "Please make your choice and press ENTER."
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo "Welcome to Haskell!"
|
||||||
|
echo
|
||||||
|
echo "This script will download and install the following binaries:"
|
||||||
|
echo " * ghcup - The Haskell toolchain installer"
|
||||||
|
echo " * ghc - The Glasgow Haskell Compiler"
|
||||||
|
echo " * cabal - The Cabal build tool for managing Haskell software"
|
||||||
|
echo " * stack - (optional) A cross-platform program for developing Haskell projects"
|
||||||
|
echo " * hls - (optional) A language server for developers to integrate with their editor/IDE"
|
||||||
|
echo
|
||||||
|
if [ -z "${GHCUP_USE_XDG_DIRS}" ] ; then
|
||||||
|
echo "ghcup installs only into the following directory,"
|
||||||
|
echo "which can be removed anytime:"
|
||||||
|
case "${plat}" in
|
||||||
|
MSYS*|MINGW*)
|
||||||
|
echo " $(cygpath -w "$GHCUP_DIR")"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo " $GHCUP_DIR"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
else
|
||||||
|
echo "ghcup installs into XDG directories as long as"
|
||||||
|
echo "'GHCUP_USE_XDG_DIRS' is set."
|
||||||
|
fi
|
||||||
|
echo
|
||||||
|
|
||||||
|
if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
||||||
|
warn "Press ENTER to proceed or ctrl-c to abort."
|
||||||
|
warn "Note that this script can be re-run at any given time."
|
||||||
|
echo
|
||||||
|
# Wait for user input to continue.
|
||||||
|
# shellcheck disable=SC2034
|
||||||
|
read -r answer </dev/tty
|
||||||
|
fi
|
||||||
|
|
||||||
|
edo mkdir -p "${GHCUP_BIN}"
|
||||||
|
|
||||||
|
if command -V "ghcup" >/dev/null 2>&1 ; then
|
||||||
|
if [ -z "${BOOTSTRAP_HASKELL_NO_UPGRADE}" ] ; then
|
||||||
|
_eghcup upgrade || download_ghcup
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
download_ghcup
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo "$(if [ -n "${BOOTSTRAP_HASKELL_YAML}" ] ; then ghcup -s "${BOOTSTRAP_HASKELL_YAML}" tool-requirements ; else ghcup tool-requirements ; fi)"
|
||||||
|
echo
|
||||||
|
|
||||||
|
if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
||||||
|
warn "Press ENTER to proceed or ctrl-c to abort."
|
||||||
|
warn "Installation may take a while."
|
||||||
|
echo
|
||||||
|
|
||||||
|
# Wait for user input to continue.
|
||||||
|
# shellcheck disable=SC2034
|
||||||
|
read -r answer </dev/tty
|
||||||
|
fi
|
||||||
|
|
||||||
|
eghcup --cache install ghc "${BOOTSTRAP_HASKELL_GHC_VERSION}"
|
||||||
|
|
||||||
|
eghcup set ghc "${BOOTSTRAP_HASKELL_GHC_VERSION}"
|
||||||
|
eghcup --cache install cabal "${BOOTSTRAP_HASKELL_CABAL_VERSION}"
|
||||||
|
|
||||||
|
adjust_cabal_config() {
|
||||||
|
edo cabal user-config -a "extra-prog-path: $(cygpath -w "$GHCUP_BIN"), $(cygpath -w "$HOME"/AppData/Roaming/cabal/bin), $(cygpath -w "$GHCUP_MSYS2"/usr/bin), $(cygpath -w "$GHCUP_MSYS2"/mingw64/bin)" -a "extra-include-dirs: $(cygpath -w "$GHCUP_MSYS2"/mingw64/include)" -a "extra-lib-dirs: $(cygpath -w "$GHCUP_MSYS2"/mingw64/lib)" -f init
|
||||||
|
}
|
||||||
|
|
||||||
|
case "${plat}" in
|
||||||
|
MSYS*|MINGW*)
|
||||||
|
if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
||||||
|
warn "Create an initial cabal.config including relevant msys2 paths (recommended)?"
|
||||||
|
warn "[Y] Yes [N] No [?] Help (default is \"Y\")."
|
||||||
|
echo
|
||||||
|
while true; do
|
||||||
|
read -r mingw_answer </dev/tty
|
||||||
|
|
||||||
|
case $mingw_answer in
|
||||||
|
[Yy]* | "")
|
||||||
|
adjust_cabal_config
|
||||||
|
break ;;
|
||||||
|
[Nn]*)
|
||||||
|
echo "Make sure that your global cabal.config references the correct mingw64 paths (extra-prog-path, extra-include-dirs and extra-lib-dirs)."
|
||||||
|
echo "And set the environment variable GHCUP_MSYS2 to the root path of your msys2 installation."
|
||||||
|
sleep 5
|
||||||
|
break ;;
|
||||||
|
*)
|
||||||
|
echo "Possible choices are:"
|
||||||
|
echo
|
||||||
|
echo "Y - Yes, create a cabal.config with pre-set paths to msys2/mingw64 (default)"
|
||||||
|
echo "N - No, leave the current/default cabal config untouched"
|
||||||
|
echo
|
||||||
|
echo "Please make your choice and press ENTER."
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
else
|
||||||
|
adjust_cabal_config
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
|
||||||
|
edo cabal new-update
|
||||||
|
|
||||||
|
|
||||||
|
if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
||||||
|
warn "Do you want to install haskell-language-server (HLS) now?"
|
||||||
|
warn "HLS is a language-server that provides IDE-like functionality"
|
||||||
|
warn "and can integrate with different editors, such as Vim, Emacs, VS Code, Atom, ..."
|
||||||
|
warn "Also see https://github.com/haskell/haskell-language-server/blob/master/README.md"
|
||||||
|
warn ""
|
||||||
|
warn "[Y] Yes [N] No [?] Help (default is \"N\")."
|
||||||
|
warn ""
|
||||||
|
|
||||||
|
while true; do
|
||||||
|
read -r hls_answer </dev/tty
|
||||||
|
|
||||||
|
case $hls_answer in
|
||||||
|
[Yy]*)
|
||||||
|
_eghcup --cache install hls || warn "HLS installation failed, continuing anyway"
|
||||||
|
break ;;
|
||||||
|
[Nn]* | "")
|
||||||
|
break ;;
|
||||||
|
*)
|
||||||
|
echo "Possible choices are:"
|
||||||
|
echo
|
||||||
|
echo "Y - Yes, install the haskell-langauge-server"
|
||||||
|
echo "N - No, don't install anything more (default)"
|
||||||
|
echo
|
||||||
|
echo "Please make your choice and press ENTER."
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
warn "Do you want to install stack now?"
|
||||||
|
warn "Stack is a haskell build tool similar to cabal that is used by some projects."
|
||||||
|
warn "Also see https://docs.haskellstack.org/"
|
||||||
|
warn ""
|
||||||
|
warn "[Y] Yes [N] No [?] Help (default is \"N\")."
|
||||||
|
warn ""
|
||||||
|
|
||||||
|
while true; do
|
||||||
|
read -r stack_answer </dev/tty
|
||||||
|
|
||||||
|
case $stack_answer in
|
||||||
|
[Yy]*)
|
||||||
|
_eghcup --cache install stack || warn "Stack installation failed, continuing anyway"
|
||||||
|
break ;;
|
||||||
|
[Nn]* | "")
|
||||||
|
break ;;
|
||||||
|
*)
|
||||||
|
echo "Possible choices are:"
|
||||||
|
echo
|
||||||
|
echo "Y - Yes, install stack"
|
||||||
|
echo "N - No, don't install anything more (default)"
|
||||||
|
echo
|
||||||
|
echo "Please make your choice and press ENTER."
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
|
# short-circuit script based on platform
|
||||||
|
case "${plat}" in
|
||||||
|
MSYS*|MINGW*)
|
||||||
|
# For windows we always adjust bashrc, since it's inside msys2
|
||||||
|
adjust_bashrc
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
if [ -z "${BOOTSTRAP_HASKELL_NONINTERACTIVE}" ] ; then
|
||||||
|
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 "configuration to do so (e.g. ~/.bashrc)."
|
||||||
|
|
||||||
|
adjust_bashrc
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
_done
|
||||||
|
|
||||||
|
)
|
||||||
|
|
||||||
|
# vim: tabstop=4 shiftwidth=4 expandtab
|
||||||
|
|
||||||
422
bootstrap-haskell.ps1
Normal file
422
bootstrap-haskell.ps1
Normal file
@@ -0,0 +1,422 @@
|
|||||||
|
<#
|
||||||
|
.SYNOPSIS
|
||||||
|
Script to bootstrap a Haskell environment
|
||||||
|
|
||||||
|
.DESCRIPTION
|
||||||
|
This is the windows GHCup installer, installing:
|
||||||
|
|
||||||
|
* ghcup - The Haskell toolchain installer"
|
||||||
|
* ghc - The Glasgow Haskell Compiler"
|
||||||
|
* msys2 - Unix-style toolchain needed for dependencies and tools
|
||||||
|
* cabal - The Cabal build tool for managing Haskell software"
|
||||||
|
* stack - (optional) A cross-platform program for developing Haskell projects"
|
||||||
|
* hls - (optional) A language server for developers to integrate with their editor/IDE"
|
||||||
|
#>
|
||||||
|
param (
|
||||||
|
# Run an interactive installation
|
||||||
|
[switch]$Interactive,
|
||||||
|
# Specify the install root (default: 'C:\')
|
||||||
|
[string]$InstallDir,
|
||||||
|
# Instead of installing a new MSys2, use an existing installation
|
||||||
|
[string]$ExistingMsys2Dir,
|
||||||
|
# Specify the cabal root directory (default: '$InstallDir\cabal')
|
||||||
|
[string]$CabalDir,
|
||||||
|
# Overwrite (or rather backup) a previous install
|
||||||
|
[switch]$Overwrite,
|
||||||
|
# Specify the bootstrap url (default: 'https://www.haskell.org/ghcup/sh/bootstrap-haskell')
|
||||||
|
[string]$BootstrapUrl,
|
||||||
|
# Run the final bootstrap script via 'bash' instead of a full newly spawned msys2 shell
|
||||||
|
[switch]$InBash
|
||||||
|
)
|
||||||
|
|
||||||
|
$Silent = !$Interactive
|
||||||
|
|
||||||
|
function Print-Msg {
|
||||||
|
param ( [Parameter(Mandatory=$true, HelpMessage='String to output')][string]$msg, [string]$color = "Green" )
|
||||||
|
Write-Host ('{0}' -f $msg) -ForegroundColor $color
|
||||||
|
}
|
||||||
|
|
||||||
|
function Create-Shortcut {
|
||||||
|
param ( [Parameter(Mandatory=$true,HelpMessage='Target path')][string]$SourceExe, [Parameter(Mandatory=$true,HelpMessage='Arguments to the path/exe')][AllowEmptyString()]$ArgumentsToSourceExe, [Parameter(Mandatory=$true,HelpMessage='The destination of the desktop link')][string]$DestinationPath )
|
||||||
|
$WshShell = New-Object -comObject WScript.Shell
|
||||||
|
$Shortcut = $WshShell.CreateShortcut($DestinationPath)
|
||||||
|
$Shortcut.TargetPath = $SourceExe
|
||||||
|
if($ArgumentsToSourceExe) {
|
||||||
|
$Shortcut.Arguments = $ArgumentsToSourceExe
|
||||||
|
}
|
||||||
|
$Shortcut.Save()
|
||||||
|
}
|
||||||
|
|
||||||
|
function Add-EnvPath {
|
||||||
|
param(
|
||||||
|
[Parameter(Mandatory=$true,HelpMessage='The Path to add to Users environment')]
|
||||||
|
[string] $Path,
|
||||||
|
|
||||||
|
[ValidateSet('Machine', 'User', 'Session')]
|
||||||
|
[string] $Container = 'Session'
|
||||||
|
)
|
||||||
|
|
||||||
|
if ($Container -eq 'Session') {
|
||||||
|
$envPaths = [Collections.Generic.List[String]]($env:Path -split ([IO.Path]::PathSeparator))
|
||||||
|
if ($envPaths -notcontains $Path) {
|
||||||
|
$envPaths.Add($Path)
|
||||||
|
$env:PATH = $envPaths -join ([IO.Path]::PathSeparator)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
[Microsoft.Win32.RegistryHive]$hive, $keyPath = switch ($Container) {
|
||||||
|
'Machine' { 'LocalMachine', 'SYSTEM\CurrentControlSet\Control\Session Manager\Environment' }
|
||||||
|
'User' { 'CurrentUser', 'Environment' }
|
||||||
|
}
|
||||||
|
|
||||||
|
$hiveKey = $envKey = $null
|
||||||
|
try {
|
||||||
|
$hiveKey = [Microsoft.Win32.RegistryKey]::OpenRemoteBaseKey($hive, '')
|
||||||
|
$envKey = $hiveKey.OpenSubKey($keyPath, $true)
|
||||||
|
$rawPath = $envKey.GetValue('PATH', '', 'DoNotExpandEnvironmentNames')
|
||||||
|
|
||||||
|
$envPaths = [Collections.Generic.List[String]]($rawPath -split ([IO.Path]::PathSeparator))
|
||||||
|
if ($envPaths -notcontains $Path) {
|
||||||
|
$envPaths.Add($Path)
|
||||||
|
$envKey.SetValue('PATH', ($envPaths -join ([IO.Path]::PathSeparator)), 'ExpandString')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
if ($envKey) { $envKey.Close() }
|
||||||
|
if ($hiveKey) { $hiveKey.Close() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
filter Get-FileSize {
|
||||||
|
'{0:N2} {1}' -f $(
|
||||||
|
if ($_ -lt 1kb) { $_, 'Bytes' }
|
||||||
|
elseif ($_ -lt 1mb) { ($_/1kb), 'KB' }
|
||||||
|
elseif ($_ -lt 1gb) { ($_/1mb), 'MB' }
|
||||||
|
elseif ($_ -lt 1tb) { ($_/1gb), 'GB' }
|
||||||
|
elseif ($_ -lt 1pb) { ($_/1tb), 'TB' }
|
||||||
|
else { ($_/1pb), 'PB' }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function Get-FileWCSynchronous{
|
||||||
|
param(
|
||||||
|
[Parameter(Mandatory=$true)]
|
||||||
|
[string]$url,
|
||||||
|
[string]$destinationFolder="$env:USERPROFILE\Downloads",
|
||||||
|
[switch]$includeStats
|
||||||
|
)
|
||||||
|
$wc = New-Object -TypeName Net.WebClient
|
||||||
|
$wc.UseDefaultCredentials = $true
|
||||||
|
$destination = Join-Path -Path $destinationFolder -ChildPath ($url | Split-Path -Leaf)
|
||||||
|
$start = Get-Date
|
||||||
|
$wc.DownloadFile($url, $destination)
|
||||||
|
$elapsed = ((Get-Date) - $start).ToString('hh\:mm\:ss')
|
||||||
|
$totalSize = (Get-Item -Path $destination).Length | Get-FileSize
|
||||||
|
if ($includeStats.IsPresent){
|
||||||
|
[PSCustomObject]@{Name=$MyInvocation.MyCommand;TotalSize=$totalSize;Time=$elapsed}
|
||||||
|
}
|
||||||
|
Get-Item -Path $destination | Unblock-File
|
||||||
|
}
|
||||||
|
|
||||||
|
function Test-AbsolutePath {
|
||||||
|
Param (
|
||||||
|
[Parameter(Mandatory=$True)]
|
||||||
|
[ValidateScript({[System.IO.Path]::IsPathRooted($_)})]
|
||||||
|
[String]$Path
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function Exec
|
||||||
|
{
|
||||||
|
[CmdletBinding()]
|
||||||
|
param(
|
||||||
|
[Parameter(Position = 0, Mandatory = 1)][string]$cmd,
|
||||||
|
[Parameter()][string]$errorMessage,
|
||||||
|
[parameter(ValueFromRemainingArguments = $true)]
|
||||||
|
[string[]]$Passthrough
|
||||||
|
)
|
||||||
|
& $cmd @Passthrough
|
||||||
|
if ($lastexitcode -ne 0) {
|
||||||
|
if (!($errorMessage)) {
|
||||||
|
throw ('Exec: Error executing command {0} with arguments ''{1}''' -f $cmd, "$Passthrough")
|
||||||
|
} else {
|
||||||
|
throw ('Exec: ' + $errorMessage)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$ErrorActionPreference = 'Stop'
|
||||||
|
|
||||||
|
$GhcupBasePrefixEnv = [System.Environment]::GetEnvironmentVariable('GHCUP_INSTALL_BASE_PREFIX', 'user')
|
||||||
|
|
||||||
|
if ($GhcupBasePrefixEnv) {
|
||||||
|
$defaultGhcupBasePrefix = $GhcupBasePrefixEnv
|
||||||
|
} else {
|
||||||
|
$partitions = Get-CimInstance win32_logicaldisk
|
||||||
|
$defaultGhcupBasePrefix = $null
|
||||||
|
foreach ($p in $partitions){
|
||||||
|
try {
|
||||||
|
if ($p."FreeSpace" -lt 5368709120) { # at least 5 GB are needed
|
||||||
|
throw ("Not enough free space on {0}" -f $p."DeviceId")
|
||||||
|
}
|
||||||
|
$null = New-Item -Path ('{0}\' -f $p."DeviceId") -Name "ghcup.test" -ItemType "directory" -Force
|
||||||
|
$defaultGhcupBasePrefix = ('{0}\' -f $p."DeviceId")
|
||||||
|
Remove-Item -LiteralPath ('{0}\ghcup.test' -f $p."DeviceId")
|
||||||
|
break
|
||||||
|
} catch {
|
||||||
|
Print-Msg -color Yellow -msg ("{0} not writable or not enough disk space, trying next device" -f $p."DeviceId")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($defaultGhcupBasePrefix) {
|
||||||
|
Print-Msg -color Green -msg ("Picked {0} as default Install prefix!" -f $defaultGhcupBasePrefix)
|
||||||
|
} else {
|
||||||
|
Print-Msg -color Red -msg "Couldn't find a writable partition with at least 5GB free disk space!"
|
||||||
|
Exit 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($Silent -and !($InstallDir)) {
|
||||||
|
$GhcupBasePrefix = $defaultGhcupBasePrefix
|
||||||
|
} elseif ($InstallDir) {
|
||||||
|
if (!(Test-Path -LiteralPath ('{0}' -f $InstallDir) -IsValid)) {
|
||||||
|
Print-Msg -color Red -msg "Not a valid directory!"
|
||||||
|
Exit 1
|
||||||
|
} elseif (!(Split-Path -IsAbsolute -Path "$InstallDir")) {
|
||||||
|
Print-Msg -color Red -msg "Non-absolute Path specified!"
|
||||||
|
Exit 1
|
||||||
|
} else {
|
||||||
|
$GhcupBasePrefix = $InstallDir
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
while ($true) {
|
||||||
|
Print-Msg -color Magenta -msg ('Where to install to (this should be a short Path, preferably a Drive like ''C:\''){1}Press enter to accept the default [{0}]:' -f $defaultGhcupBasePrefix, "`n")
|
||||||
|
$basePrefixPrompt = Read-Host
|
||||||
|
$GhcupBasePrefix = ($defaultGhcupBasePrefix,$basePrefixPrompt)[[bool]$basePrefixPrompt]
|
||||||
|
if (!($GhcupBasePrefix.EndsWith('\'))) {
|
||||||
|
$GhcupBasePrefix = ('{0}\' -f $GhcupBasePrefix)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!($GhcupBasePrefix)) {
|
||||||
|
Print-Msg -color Red -msg "No directory specified!"
|
||||||
|
} elseif (!(Test-Path -LiteralPath ('{0}' -f $GhcupBasePrefix))) {
|
||||||
|
Print-Msg -color Red -msg "Directory does not exist, need to specify an existing Drive/Directory"
|
||||||
|
} elseif (!(Split-Path -IsAbsolute -Path "$GhcupBasePrefix")) {
|
||||||
|
Print-Msg -color Red -msg "Invalid/Non-absolute Path specified"
|
||||||
|
} else {
|
||||||
|
Break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Print-Msg -msg ('Setting env variable GHCUP_INSTALL_BASE_PREFIX to ''{0}''' -f $GhcupBasePrefix)
|
||||||
|
$null = [Environment]::SetEnvironmentVariable("GHCUP_INSTALL_BASE_PREFIX", $GhcupBasePrefix, [System.EnvironmentVariableTarget]::User)
|
||||||
|
|
||||||
|
|
||||||
|
$GhcupDir = ('{0}\ghcup' -f $GhcupBasePrefix)
|
||||||
|
$MsysDir = ('{0}\msys64' -f $GhcupDir)
|
||||||
|
$Bash = ('{0}\usr\bin\bash' -f $MsysDir)
|
||||||
|
if (!($BootstrapUrl)) {
|
||||||
|
$BootstrapUrl = 'https://www.haskell.org/ghcup/sh/bootstrap-haskell'
|
||||||
|
}
|
||||||
|
$GhcupMsys2 = [System.Environment]::GetEnvironmentVariable('GHCUP_MSYS2', 'user')
|
||||||
|
|
||||||
|
Print-Msg -msg 'Preparing for GHCup installation...'
|
||||||
|
|
||||||
|
if (Test-Path -LiteralPath ('{0}' -f $GhcupDir)) {
|
||||||
|
Print-Msg -msg ('GHCup already installed at ''{0}''...' -f $GhcupDir)
|
||||||
|
if ($Overwrite) {
|
||||||
|
$decision = 0
|
||||||
|
} elseif (!($Silent)) {
|
||||||
|
$decision = $Host.UI.PromptForChoice('Install GHCup'
|
||||||
|
, 'GHCup is already installed, what do you want to do?'
|
||||||
|
, [System.Management.Automation.Host.ChoiceDescription[]] @('&Reinstall'
|
||||||
|
'&Continue'
|
||||||
|
'&Abort'), 1)
|
||||||
|
} else {
|
||||||
|
$decision = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($decision -eq 0) {
|
||||||
|
$suffix = [IO.Path]::GetRandomFileName()
|
||||||
|
Print-Msg -msg ('Backing up {0} to {0}-{1} ...' -f $GhcupDir, $suffix)
|
||||||
|
Rename-Item -Path ('{0}' -f $GhcupDir) -NewName ('{0}-{1}' -f $GhcupDir, $suffix)
|
||||||
|
} elseif ($decision -eq 1) {
|
||||||
|
Print-Msg -msg 'Continuing installation...'
|
||||||
|
} elseif ($decision -eq 2) {
|
||||||
|
Exit 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
$null = New-Item -Path ('{0}' -f $GhcupDir) -ItemType 'directory' -ErrorAction SilentlyContinue
|
||||||
|
$null = New-Item -Path ('{0}' -f $GhcupDir) -Name 'bin' -ItemType 'directory' -ErrorAction SilentlyContinue
|
||||||
|
|
||||||
|
Print-Msg -msg 'First checking for Msys2...'
|
||||||
|
|
||||||
|
if (!(Test-Path -Path ('{0}' -f $MsysDir))) {
|
||||||
|
if ($Silent) {
|
||||||
|
$msys2Decision = 0
|
||||||
|
} else {
|
||||||
|
$msys2Decision = $Host.UI.PromptForChoice('Install MSys2'
|
||||||
|
, 'Do you want GHCup to install a default MSys2 toolchain (recommended)?'
|
||||||
|
, [System.Management.Automation.Host.ChoiceDescription[]] @('&Yes'
|
||||||
|
'&No'), 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($msys2Decision -eq 0) {
|
||||||
|
Print-Msg -msg ('...Msys2 doesn''t exist, installing into {0} ...this may take a while' -f $MsysDir)
|
||||||
|
|
||||||
|
# Download the archive
|
||||||
|
Print-Msg -msg 'Downloading Msys2 archive...'
|
||||||
|
$archive = 'msys2-x86_64-latest.sfx.exe'
|
||||||
|
|
||||||
|
if (Get-Command -Name 'curl.exe' -ErrorAction SilentlyContinue) {
|
||||||
|
Exec "curl.exe" '-o' ('{0}\{1}' -f $env:TEMP, $archive) ('https://repo.msys2.org/distrib/{0}' -f $archive)
|
||||||
|
} else {
|
||||||
|
Get-FileWCSynchronous -url ('https://repo.msys2.org/distrib/{0}' -f $archive) -destinationFolder "$env:TEMP" -includeStats
|
||||||
|
}
|
||||||
|
|
||||||
|
Print-Msg -msg 'Extracting Msys2 archive...'
|
||||||
|
$null = & "$env:TEMP\$archive" '-y' ('-o{0}' -f $GhcupDir) # Extract
|
||||||
|
Remove-Item -Path ('{0}/{1}' -f $env:TEMP, $archive)
|
||||||
|
|
||||||
|
Print-Msg -msg 'Processing MSYS2 bash for first time use...'
|
||||||
|
Exec "$Bash" '-lc' 'exit'
|
||||||
|
|
||||||
|
Exec "$env:windir\system32\taskkill.exe" /F /FI `"MODULES eq msys-2.0.dll`"
|
||||||
|
|
||||||
|
Print-Msg -msg 'Upgrading full system...'
|
||||||
|
Exec "$Bash" '-lc' 'pacman --noconfirm -Syuu'
|
||||||
|
|
||||||
|
Print-Msg -msg 'Upgrading full system twice...'
|
||||||
|
Exec "$Bash" '-lc' 'pacman --noconfirm -Syuu'
|
||||||
|
|
||||||
|
Print-Msg -msg 'Installing Dependencies...'
|
||||||
|
Exec "$Bash" '-lc' 'pacman --noconfirm -S --needed curl mingw-w64-x86_64-pkgconf'
|
||||||
|
|
||||||
|
Print-Msg -msg 'Updating SSL root certificate authorities...'
|
||||||
|
Exec "$Bash" '-lc' 'pacman --noconfirm -S ca-certificates'
|
||||||
|
|
||||||
|
Print-Msg -msg 'Setting default home directory...'
|
||||||
|
Exec "$Bash" '-lc' "sed -i -e 's/db_home:.*$/db_home: windows/' /etc/nsswitch.conf"
|
||||||
|
} elseif ($msys2Decision -eq 1) {
|
||||||
|
Print-Msg -color Yellow -msg 'Skipping MSys2 installation.'
|
||||||
|
while ($true) {
|
||||||
|
if ($GhcupMsys2) {
|
||||||
|
$defaultMsys2Dir = $GhcupMsys2
|
||||||
|
Print-Msg -color Magenta -msg ('Input existing MSys2 toolchain directory. Press enter to accept the default [{0}]:' -f $defaultMsys2Dir)
|
||||||
|
$MsysDirPrompt = Read-Host
|
||||||
|
$MsysDir = ($defaultMsys2Dir,$MsysDirPrompt)[[bool]$MsysDirPrompt]
|
||||||
|
} else {
|
||||||
|
Print-Msg -color Magenta -msg 'Input existing MSys2 toolchain directory:'
|
||||||
|
$MsysDir = Read-Host
|
||||||
|
}
|
||||||
|
if (!($MsysDir)) {
|
||||||
|
Print-Msg -color Red -msg "No directory specified!"
|
||||||
|
} elseif (!(Test-Path -LiteralPath ('{0}' -f $MsysDir))) {
|
||||||
|
Print-Msg -color Red -msg ('MSys2 installation at ''{0}'' could not be found!' -f $MsysDir)
|
||||||
|
} elseif (!(Split-Path -IsAbsolute -Path "$MsysDir")) {
|
||||||
|
Print-Msg -color Red -msg "Invalid/Non-absolute Path specified"
|
||||||
|
} else {
|
||||||
|
Break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Print-Msg -msg ('Setting GHCUP_MSYS2 env var to ''{0}''' -f $MsysDir)
|
||||||
|
$null = [Environment]::SetEnvironmentVariable("GHCUP_MSYS2", $MsysDir, [System.EnvironmentVariableTarget]::User)
|
||||||
|
$Bash = ('{0}\usr\bin\bash' -f $MsysDir)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Print-Msg -msg ('...Msys2 found in {0} ...skipping Msys2 installation.' -f $MsysDir)
|
||||||
|
}
|
||||||
|
|
||||||
|
Print-Msg -msg 'Creating shortcuts...'
|
||||||
|
$DesktopDir = [Environment]::GetFolderPath("Desktop")
|
||||||
|
$GhcInstArgs = '-mingw64 -mintty -c "pacman --noconfirm -S --needed base-devel gettext autoconf make libtool automake python p7zip patch unzip"'
|
||||||
|
Create-Shortcut -SourceExe ('{0}\msys2_shell.cmd' -f $MsysDir) -ArgumentsToSourceExe $GhcInstArgs -DestinationPath ('{0}\Install GHC dev dependencies.lnk' -f $DesktopDir)
|
||||||
|
Create-Shortcut -SourceExe ('{0}\msys2_shell.cmd' -f $MsysDir) -ArgumentsToSourceExe '-mingw64' -DestinationPath ('{0}\Mingw haskell shell.lnk' -f $DesktopDir)
|
||||||
|
Create-Shortcut -SourceExe 'https://www.msys2.org/docs/package-management' -ArgumentsToSourceExe '' -DestinationPath ('{0}\Mingw package management docs.url' -f $DesktopDir)
|
||||||
|
|
||||||
|
Print-Msg -msg ('Adding {0}\bin to Users Path...' -f $GhcupDir)
|
||||||
|
Add-EnvPath -Path ('{0}\bin' -f ([System.IO.Path]::GetFullPath("$GhcupDir"))) -Container 'User'
|
||||||
|
|
||||||
|
if ($CabalDir) {
|
||||||
|
$CabDirEnv = $CabalDir
|
||||||
|
if (!($CabDirEnv)) {
|
||||||
|
Print-Msg -color Red -msg "No directory specified!"
|
||||||
|
Exit 1
|
||||||
|
} elseif (!(Split-Path -IsAbsolute -Path "$CabDirEnv")) {
|
||||||
|
Print-Msg -color Red -msg "Invalid/Non-absolute Path specified"
|
||||||
|
Exit 1
|
||||||
|
}
|
||||||
|
} elseif (!($Silent)) {
|
||||||
|
while ($true) {
|
||||||
|
|
||||||
|
$defaultCabalDir = ('{0}\cabal' -f $GhcupBasePrefix)
|
||||||
|
Print-Msg -color Magenta -msg ('Specify Cabal directory (this is where haskell packages end up). Press enter to accept the default [{0}]:' -f $defaultCabalDir)
|
||||||
|
$CabalDirPrompt = Read-Host
|
||||||
|
$CabDirEnv = ($defaultCabalDir,$CabalDirPrompt)[[bool]$CabalDirPrompt]
|
||||||
|
|
||||||
|
if (!($CabDirEnv)) {
|
||||||
|
Print-Msg -color Red -msg "No directory specified!"
|
||||||
|
} elseif (!(Split-Path -IsAbsolute -Path "$CabDirEnv")) {
|
||||||
|
Print-Msg -color Red -msg "Invalid/Non-absolute Path specified"
|
||||||
|
} else {
|
||||||
|
Break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$CabDirEnv = ('{0}\cabal' -f $GhcupBasePrefix)
|
||||||
|
}
|
||||||
|
|
||||||
|
$CabalDirFull = [System.IO.Path]::GetFullPath("$CabDirEnv")
|
||||||
|
Print-Msg -msg ('Setting CABAL_DIR to ''{0}''' -f $CabalDirFull)
|
||||||
|
$null = [Environment]::SetEnvironmentVariable("CABAL_DIR", $CabalDirFull, [System.EnvironmentVariableTarget]::User)
|
||||||
|
|
||||||
|
Print-Msg -msg 'Starting GHCup installer...'
|
||||||
|
|
||||||
|
$Msys2Shell = ('{0}\msys2_shell.cmd' -f $MsysDir)
|
||||||
|
|
||||||
|
if ($Silent) {
|
||||||
|
$SilentExport = 'export BOOTSTRAP_HASKELL_NONINTERACTIVE=1 ;'
|
||||||
|
} else {
|
||||||
|
$SilentExport = ''
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((Get-Process -ID $PID).ProcessName.StartsWith("bootstrap-haskell") -Or $InBash) {
|
||||||
|
Exec "$Bash" '-lc' ('{4} [ -n ''{1}'' ] && export GHCUP_MSYS2=$(cygpath -m ''{1}'') ; [ -n ''{2}'' ] && export GHCUP_INSTALL_BASE_PREFIX=$(cygpath -m ''{2}/'') ; export PATH=$(cygpath -u ''{3}/bin''):$PATH ; export CABAL_DIR=''{5}'' ; [[ ''{0}'' = https* ]] && curl --proto ''=https'' --tlsv1.2 -sSf {0} | bash || cat $(cygpath -m ''{0}'') | bash' -f $BootstrapUrl, $MsysDir, $GhcupBasePrefix, $GhcupDir, $SilentExport, $CabalDirFull)
|
||||||
|
} else {
|
||||||
|
Exec "$Msys2Shell" '-mingw64' '-mintty' '-c' ('{4} [ -n ''{1}'' ] && export GHCUP_MSYS2=$(cygpath -m ''{1}'') ; [ -n ''{2}'' ] && export GHCUP_INSTALL_BASE_PREFIX=$(cygpath -m ''{2}/'') ; export PATH=$(cygpath -u ''{3}/bin''):$PATH ; export CABAL_DIR=''{5}'' ; trap ''echo Press any key to exit && read -n 1 && exit'' 2 ; [[ ''{0}'' = https* ]] && curl --proto ''=https'' --tlsv1.2 -sSf {0} | bash || cat $(cygpath -m ''{0}'') | bash ; echo ''Press any key to exit'' && read -n 1' -f $BootstrapUrl, $MsysDir, $GhcupBasePrefix, $GhcupDir, $SilentExport, $CabalDirFull)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# SIG # Begin signature block
|
||||||
|
# MIID4QYJKoZIhvcNAQcCoIID0jCCA84CAQExCzAJBgUrDgMCGgUAMGkGCisGAQQB
|
||||||
|
# gjcCAQSgWzBZMDQGCisGAQQBgjcCAR4wJgIDAQAABBAfzDtgWUsITrck0sYpfvNR
|
||||||
|
# AgEAAgEAAgEAAgEAAgEAMCEwCQYFKw4DAhoFAAQUVqKek181kF/Jx/P7z176herc
|
||||||
|
# ZyCgggH/MIIB+zCCAWSgAwIBAgIQGOezhGS1A5tHh9VubW0liDANBgkqhkiG9w0B
|
||||||
|
# AQUFADAYMRYwFAYDVQQDDA1KdWxpYW4gT3NwYWxkMB4XDTIxMDUzMDE4Mzk1OVoX
|
||||||
|
# DTI1MDUzMDAwMDAwMFowGDEWMBQGA1UEAwwNSnVsaWFuIE9zcGFsZDCBnzANBgkq
|
||||||
|
# hkiG9w0BAQEFAAOBjQAwgYkCgYEAs76XCXYPM14buR1RkVKhOB8pyM4Df6kPaz75
|
||||||
|
# nkbA0nq1VmMhBfCYFWyYHd7jniqTH0LoAKGGquN1bniREaCP9j2pFWpMIgLpQH3H
|
||||||
|
# +jpsfmxV2BTG8q+Jok88gTXS1FlAk72E85zO/Jhr6Fja1aFYAdibBRsRxcVMTVh7
|
||||||
|
# 4AGLNGUCAwEAAaNGMEQwEwYDVR0lBAwwCgYIKwYBBQUHAwMwHQYDVR0OBBYEFC+R
|
||||||
|
# hdhPo0Ty5HnzHyo1pN35IfZQMA4GA1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQUF
|
||||||
|
# AAOBgQAl3IdBVIwbJJDp7BksMYPeM4ivB3UyNvlw8aVxGwAzNgdSaezYIdMFtKXV
|
||||||
|
# CSv5bd4VnFRAPDJW9dhW0h3SkeJUoklUxMjKXhR3qygQhSxPDjIatAuOCffGACba
|
||||||
|
# ZZ7Om40b+pKXc6i/HnlApk9DGbXJ59bFcLGGcZ9QjoUae6Ex1DGCAUwwggFIAgEB
|
||||||
|
# MCwwGDEWMBQGA1UEAwwNSnVsaWFuIE9zcGFsZAIQGOezhGS1A5tHh9VubW0liDAJ
|
||||||
|
# BgUrDgMCGgUAoHgwGAYKKwYBBAGCNwIBDDEKMAigAoAAoQKAADAZBgkqhkiG9w0B
|
||||||
|
# CQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAj
|
||||||
|
# BgkqhkiG9w0BCQQxFgQUosm9nN1JgajqSBa1cUwxxhLrAsYwDQYJKoZIhvcNAQEB
|
||||||
|
# BQAEgYCnKzfsH1aDjS6xkC/uymjaBowHSnh6nFu2AkjcKu8RgcBZzP5SLBXgU9wm
|
||||||
|
# aED5Ujwyq3Qre+TGVRUqwkEauDhQiX2A008G00fRO6+di6yJRCRn5eaRAbdU3Xww
|
||||||
|
# E5VhEwLBnwzWrvLKtdEclhgUCo5Tq87QMXVdgX4aRmunl4ZE+Q==
|
||||||
|
# SIG # End signature block
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
20
cabal.ghc8104.project
Normal file
20
cabal.ghc8104.project
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
packages: ./ghcup.cabal
|
||||||
|
|
||||||
|
with-compiler: ghc-8.10.4
|
||||||
|
|
||||||
|
optional-packages: ./3rdparty/*/*.cabal
|
||||||
|
|
||||||
|
optimization: 2
|
||||||
|
|
||||||
|
package streamly
|
||||||
|
ghc-options: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
|
|
||||||
|
package ghcup
|
||||||
|
ghc-options: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
|
|
||||||
|
constraints: http-io-streams -brotli
|
||||||
|
|
||||||
|
package libarchive
|
||||||
|
flags: -system-libarchive
|
||||||
|
|
||||||
|
allow-newer: base, ghc-prim, template-haskell
|
||||||
261
cabal.ghc8104.project.freeze
Normal file
261
cabal.ghc8104.project.freeze
Normal file
@@ -0,0 +1,261 @@
|
|||||||
|
active-repositories: hackage.haskell.org:merge
|
||||||
|
constraints: any.Cabal ==3.2.1.0,
|
||||||
|
any.HUnit ==1.6.2.0,
|
||||||
|
any.IfElse ==0.85,
|
||||||
|
any.QuickCheck ==2.14.2,
|
||||||
|
QuickCheck -old-random +templatehaskell,
|
||||||
|
any.StateVar ==1.2.1,
|
||||||
|
any.abstract-deque ==0.3,
|
||||||
|
abstract-deque -usecas,
|
||||||
|
any.aeson ==1.5.6.0,
|
||||||
|
aeson -bytestring-builder -cffi -developer -fast,
|
||||||
|
any.aeson-pretty ==0.8.8,
|
||||||
|
aeson-pretty -lib-only,
|
||||||
|
any.alex ==3.2.6,
|
||||||
|
alex +small_base,
|
||||||
|
any.ansi-terminal ==0.11,
|
||||||
|
ansi-terminal -example,
|
||||||
|
any.ansi-wl-pprint ==0.6.9,
|
||||||
|
ansi-wl-pprint -example,
|
||||||
|
any.array ==0.5.4.0,
|
||||||
|
any.ascii-string ==1.0.1.4,
|
||||||
|
any.assoc ==1.0.2,
|
||||||
|
any.async ==2.2.3,
|
||||||
|
async -bench,
|
||||||
|
any.atomic-primops ==0.8.4,
|
||||||
|
atomic-primops -debug,
|
||||||
|
any.attoparsec ==0.13.2.5,
|
||||||
|
attoparsec -developer,
|
||||||
|
any.auto-update ==0.1.6,
|
||||||
|
any.base ==4.14.1.0,
|
||||||
|
any.base-compat ==0.11.2,
|
||||||
|
any.base-compat-batteries ==0.11.2,
|
||||||
|
any.base-orphans ==0.8.4,
|
||||||
|
any.base16-bytestring ==1.0.1.0,
|
||||||
|
any.base64-bytestring ==1.2.0.1,
|
||||||
|
any.bifunctors ==5.5.10,
|
||||||
|
bifunctors +semigroups +tagged,
|
||||||
|
any.binary ==0.8.8.0,
|
||||||
|
any.blaze-builder ==0.4.2.1,
|
||||||
|
any.bytestring ==0.10.12.0,
|
||||||
|
any.bz2 ==1.0.1.0,
|
||||||
|
bz2 -cross +with-bzlib,
|
||||||
|
any.c2hs ==0.28.7,
|
||||||
|
c2hs +base3 -regression,
|
||||||
|
any.call-stack ==0.3.0,
|
||||||
|
any.case-insensitive ==1.2.1.0,
|
||||||
|
any.casing ==0.1.4.1,
|
||||||
|
any.cereal ==0.5.8.1,
|
||||||
|
cereal -bytestring-builder,
|
||||||
|
any.chs-cabal ==0.1.1.0,
|
||||||
|
any.chs-deps ==0.1.0.0,
|
||||||
|
chs-deps -cross,
|
||||||
|
any.clock ==0.8.2,
|
||||||
|
clock -llvm,
|
||||||
|
any.cmdargs ==0.10.21,
|
||||||
|
cmdargs +quotation -testprog,
|
||||||
|
any.colour ==2.3.5,
|
||||||
|
any.comonad ==5.0.8,
|
||||||
|
comonad +containers +distributive +indexed-traversable,
|
||||||
|
any.composition-prelude ==3.0.0.2,
|
||||||
|
composition-prelude -development,
|
||||||
|
any.concurrent-output ==1.10.12,
|
||||||
|
any.conduit ==1.3.4.1,
|
||||||
|
any.conduit-extra ==1.3.5,
|
||||||
|
any.containers ==0.6.2.1,
|
||||||
|
any.contravariant ==1.5.3,
|
||||||
|
contravariant +semigroups +statevar +tagged,
|
||||||
|
any.cryptohash-sha256 ==0.11.102.0,
|
||||||
|
cryptohash-sha256 -exe +use-cbits,
|
||||||
|
any.data-default-class ==0.1.2.0,
|
||||||
|
any.data-fix ==0.3.1,
|
||||||
|
any.deepseq ==1.4.4.0,
|
||||||
|
any.deferred-folds ==0.9.17,
|
||||||
|
any.directory ==1.3.6.0,
|
||||||
|
any.distributive ==0.6.2.1,
|
||||||
|
distributive +semigroups +tagged,
|
||||||
|
any.dlist ==1.0,
|
||||||
|
dlist -werror,
|
||||||
|
any.easy-file ==0.2.2,
|
||||||
|
any.errors ==2.3.0,
|
||||||
|
any.exceptions ==0.10.4,
|
||||||
|
any.fast-logger ==3.0.3,
|
||||||
|
any.filepath ==1.4.2.1,
|
||||||
|
any.focus ==1.0.2,
|
||||||
|
any.foldl ==1.4.11,
|
||||||
|
any.free ==5.1.6,
|
||||||
|
any.fusion-plugin-types ==0.1.0,
|
||||||
|
any.generic-arbitrary ==0.1.0,
|
||||||
|
any.generics-sop ==0.5.1.1,
|
||||||
|
any.ghc-boot-th ==8.10.4,
|
||||||
|
any.ghc-prim ==0.6.1,
|
||||||
|
ghcup -internal-downloader -tar -tui,
|
||||||
|
any.happy ==1.20.0,
|
||||||
|
any.hashable ==1.3.1.0,
|
||||||
|
hashable +integer-gmp,
|
||||||
|
any.haskell-src-exts ==1.23.1,
|
||||||
|
any.haskell-src-meta ==0.8.7,
|
||||||
|
any.haskus-utils-data ==1.4,
|
||||||
|
any.haskus-utils-types ==1.5.1,
|
||||||
|
any.haskus-utils-variant ==3.1,
|
||||||
|
any.heaps ==0.3.6.1,
|
||||||
|
any.hpath ==0.11.0,
|
||||||
|
any.hpath-directory ==0.14.1,
|
||||||
|
any.hpath-filepath ==0.10.4,
|
||||||
|
any.hpath-io ==0.14.1,
|
||||||
|
any.hpath-posix ==0.13.2,
|
||||||
|
any.hsc2hs ==0.68.7,
|
||||||
|
hsc2hs -in-ghc-tree,
|
||||||
|
any.hspec ==2.7.8,
|
||||||
|
any.hspec-core ==2.7.8,
|
||||||
|
any.hspec-discover ==2.7.8,
|
||||||
|
any.hspec-expectations ==0.8.2,
|
||||||
|
any.hspec-golden-aeson ==0.7.0.0,
|
||||||
|
any.indexed-profunctors ==0.1,
|
||||||
|
any.indexed-traversable ==0.1.1,
|
||||||
|
any.indexed-traversable-instances ==0.1,
|
||||||
|
any.integer-gmp ==1.0.3.0,
|
||||||
|
any.integer-logarithms ==1.0.3.1,
|
||||||
|
integer-logarithms -check-bounds +integer-gmp,
|
||||||
|
any.language-c ==0.8.3,
|
||||||
|
language-c -allwarnings +iecfpextension +separatesyb +usebytestrings,
|
||||||
|
any.libarchive ==3.0.2.1,
|
||||||
|
libarchive -cross -low-memory -system-libarchive,
|
||||||
|
any.libyaml ==0.1.2,
|
||||||
|
libyaml -no-unicode -system-libyaml,
|
||||||
|
any.lifted-base ==0.2.3.12,
|
||||||
|
any.list-t ==1.0.4,
|
||||||
|
any.lockfree-queue ==0.2.3.1,
|
||||||
|
lzma -static,
|
||||||
|
any.math-functions ==0.3.4.1,
|
||||||
|
math-functions +system-erf +system-expm1,
|
||||||
|
any.megaparsec ==9.0.1,
|
||||||
|
megaparsec -dev,
|
||||||
|
any.microlens ==0.4.12.0,
|
||||||
|
any.microlens-mtl ==0.2.0.1,
|
||||||
|
any.microlens-th ==0.4.3.9,
|
||||||
|
any.mmorph ==1.1.5,
|
||||||
|
any.monad-control ==1.0.2.3,
|
||||||
|
any.monad-logger ==0.3.36,
|
||||||
|
monad-logger +template_haskell,
|
||||||
|
any.monad-loops ==0.4.3,
|
||||||
|
monad-loops +base4,
|
||||||
|
any.mono-traversable ==1.0.15.1,
|
||||||
|
any.mtl ==2.2.2,
|
||||||
|
any.mwc-random ==0.15.0.1,
|
||||||
|
any.network ==3.1.2.1,
|
||||||
|
network -devel,
|
||||||
|
any.old-locale ==1.0.0.7,
|
||||||
|
any.old-time ==1.1.0.3,
|
||||||
|
any.optics ==0.4,
|
||||||
|
any.optics-core ==0.4,
|
||||||
|
optics-core -explicit-generic-labels,
|
||||||
|
any.optics-extra ==0.4,
|
||||||
|
any.optics-th ==0.4,
|
||||||
|
any.optics-vl ==0.2.1,
|
||||||
|
any.optparse-applicative ==0.16.1.0,
|
||||||
|
optparse-applicative +process,
|
||||||
|
any.os-release ==1.0.2,
|
||||||
|
os-release -devel,
|
||||||
|
any.parallel ==3.2.2.0,
|
||||||
|
any.parsec ==3.1.14.0,
|
||||||
|
any.parser-combinators ==1.3.0,
|
||||||
|
parser-combinators -dev,
|
||||||
|
any.pretty ==1.1.3.6,
|
||||||
|
any.pretty-terminal ==0.1.0.0,
|
||||||
|
any.primitive ==0.7.1.0,
|
||||||
|
any.primitive-extras ==0.8.2,
|
||||||
|
any.primitive-unlifted ==0.1.3.0,
|
||||||
|
any.process ==1.6.9.0,
|
||||||
|
any.profunctors ==5.6.2,
|
||||||
|
any.quickcheck-arbitrary-adt ==0.3.1.0,
|
||||||
|
any.quickcheck-io ==0.2.0,
|
||||||
|
any.random ==1.2.0,
|
||||||
|
any.recursion-schemes ==5.2.2,
|
||||||
|
recursion-schemes +template-haskell,
|
||||||
|
any.regex-posix ==0.96.0.0,
|
||||||
|
regex-posix -_regex-posix-clib,
|
||||||
|
any.resourcet ==1.2.4.2,
|
||||||
|
any.rts ==1.0,
|
||||||
|
any.safe ==0.3.19,
|
||||||
|
any.safe-exceptions ==0.1.7.1,
|
||||||
|
any.scientific ==0.3.6.2,
|
||||||
|
scientific -bytestring-builder -integer-simple,
|
||||||
|
any.semigroupoids ==5.3.5,
|
||||||
|
semigroupoids +comonad +containers +contravariant +distributive +tagged +unordered-containers,
|
||||||
|
any.setenv ==0.1.1.3,
|
||||||
|
any.sop-core ==0.5.0.1,
|
||||||
|
any.split ==0.2.3.4,
|
||||||
|
any.splitmix ==0.1.0.3,
|
||||||
|
splitmix -optimised-mixer,
|
||||||
|
any.stm ==2.5.0.0,
|
||||||
|
any.stm-chans ==3.0.0.4,
|
||||||
|
any.streaming-commons ==0.2.2.1,
|
||||||
|
streaming-commons -use-bytestring-builder,
|
||||||
|
any.streamly ==0.7.3,
|
||||||
|
streamly -debug -dev -examples -examples-sdl -fusion-plugin -has-llvm -inspection -no-fusion -streamk,
|
||||||
|
any.streamly-bytestring ==0.1.2,
|
||||||
|
any.streamly-posix ==0.1.0.1,
|
||||||
|
any.strict ==0.4.0.1,
|
||||||
|
strict +assoc,
|
||||||
|
any.strict-base ==0.4.0.0,
|
||||||
|
any.string-interpolate ==0.3.1.0,
|
||||||
|
string-interpolate -bytestring-builder -extended-benchmarks -text-builder,
|
||||||
|
any.syb ==0.7.2.1,
|
||||||
|
any.tagged ==0.8.6.1,
|
||||||
|
tagged +deepseq +transformers,
|
||||||
|
any.tasty ==1.3.1,
|
||||||
|
tasty +clock,
|
||||||
|
any.tasty-hunit ==0.10.0.3,
|
||||||
|
any.tasty-quickcheck ==0.10.1.2,
|
||||||
|
any.template-haskell ==2.16.0.0,
|
||||||
|
any.terminal-size ==0.3.2.1,
|
||||||
|
any.terminfo ==0.4.1.4,
|
||||||
|
any.text ==1.2.4.1,
|
||||||
|
any.text-conversions ==0.3.1,
|
||||||
|
any.text-short ==0.1.3,
|
||||||
|
text-short -asserts,
|
||||||
|
any.tf-random ==0.5,
|
||||||
|
any.th-abstraction ==0.4.2.0,
|
||||||
|
any.th-compat ==0.1.1,
|
||||||
|
any.th-lift ==0.8.2,
|
||||||
|
any.th-lift-instances ==0.1.18,
|
||||||
|
any.th-orphans ==0.13.11,
|
||||||
|
any.th-reify-many ==0.1.9,
|
||||||
|
any.these ==1.1.1.1,
|
||||||
|
these +assoc,
|
||||||
|
any.time ==1.9.3,
|
||||||
|
any.time-compat ==1.9.5,
|
||||||
|
time-compat -old-locale,
|
||||||
|
any.transformers ==0.5.6.2,
|
||||||
|
any.transformers-base ==0.4.5.2,
|
||||||
|
transformers-base +orphaninstances,
|
||||||
|
any.transformers-compat ==0.6.6,
|
||||||
|
transformers-compat -five +five-three -four +generic-deriving +mtl -three -two,
|
||||||
|
any.typed-process ==0.2.6.0,
|
||||||
|
any.unbounded-delays ==0.1.1.1,
|
||||||
|
any.unix ==2.7.2.2,
|
||||||
|
any.unix-bytestring ==0.3.7.3,
|
||||||
|
any.unix-compat ==0.5.3,
|
||||||
|
unix-compat -old-time,
|
||||||
|
any.unix-time ==0.4.7,
|
||||||
|
any.unliftio-core ==0.2.0.1,
|
||||||
|
any.unordered-containers ==0.2.13.0,
|
||||||
|
unordered-containers -debug,
|
||||||
|
any.uri-bytestring ==0.3.3.0,
|
||||||
|
uri-bytestring -lib-werror,
|
||||||
|
any.utf8-string ==1.0.2,
|
||||||
|
any.uuid-types ==1.0.4,
|
||||||
|
any.vector ==0.12.2.0,
|
||||||
|
vector +boundschecks -internalchecks -unsafechecks -wall,
|
||||||
|
any.vector-algorithms ==0.8.0.4,
|
||||||
|
vector-algorithms +bench +boundschecks -internalchecks -llvm +properties -unsafechecks,
|
||||||
|
any.versions ==4.0.3,
|
||||||
|
any.vty ==5.33,
|
||||||
|
any.wcwidth ==0.0.2,
|
||||||
|
wcwidth -cli +split-base,
|
||||||
|
any.word8 ==0.1.3,
|
||||||
|
any.yaml ==0.11.5.0,
|
||||||
|
yaml +no-examples +no-exe,
|
||||||
|
zlib -non-blocking-ffi -pkg-config -static
|
||||||
|
index-state: hackage.haskell.org 2021-03-07T18:36:25Z
|
||||||
20
cabal.ghc884.project
Normal file
20
cabal.ghc884.project
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
packages: ./ghcup.cabal
|
||||||
|
|
||||||
|
with-compiler: ghc-8.8.4
|
||||||
|
|
||||||
|
optional-packages: ./3rdparty/*/*.cabal
|
||||||
|
|
||||||
|
optimization: 2
|
||||||
|
|
||||||
|
package streamly
|
||||||
|
ghc-options: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
|
|
||||||
|
package ghcup
|
||||||
|
ghc-options: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
|
|
||||||
|
constraints: http-io-streams -brotli
|
||||||
|
|
||||||
|
package libarchive
|
||||||
|
flags: -system-libarchive
|
||||||
|
|
||||||
|
allow-newer: base, ghc-prim, template-haskell
|
||||||
262
cabal.ghc884.project.freeze
Normal file
262
cabal.ghc884.project.freeze
Normal file
@@ -0,0 +1,262 @@
|
|||||||
|
active-repositories: hackage.haskell.org:merge
|
||||||
|
constraints: any.Cabal ==3.0.1.0,
|
||||||
|
any.HUnit ==1.6.2.0,
|
||||||
|
any.IfElse ==0.85,
|
||||||
|
any.QuickCheck ==2.14.2,
|
||||||
|
QuickCheck -old-random +templatehaskell,
|
||||||
|
any.StateVar ==1.2.1,
|
||||||
|
any.abstract-deque ==0.3,
|
||||||
|
abstract-deque -usecas,
|
||||||
|
any.aeson ==1.5.6.0,
|
||||||
|
aeson -bytestring-builder -cffi -developer -fast,
|
||||||
|
any.aeson-pretty ==0.8.8,
|
||||||
|
aeson-pretty -lib-only,
|
||||||
|
any.alex ==3.2.6,
|
||||||
|
alex +small_base,
|
||||||
|
any.ansi-terminal ==0.11,
|
||||||
|
ansi-terminal -example,
|
||||||
|
any.ansi-wl-pprint ==0.6.9,
|
||||||
|
ansi-wl-pprint -example,
|
||||||
|
any.array ==0.5.4.0,
|
||||||
|
any.ascii-string ==1.0.1.4,
|
||||||
|
any.assoc ==1.0.2,
|
||||||
|
any.async ==2.2.3,
|
||||||
|
async -bench,
|
||||||
|
any.atomic-primops ==0.8.4,
|
||||||
|
atomic-primops -debug,
|
||||||
|
any.attoparsec ==0.13.2.5,
|
||||||
|
attoparsec -developer,
|
||||||
|
any.auto-update ==0.1.6,
|
||||||
|
any.base ==4.13.0.0,
|
||||||
|
any.base-compat ==0.11.2,
|
||||||
|
any.base-compat-batteries ==0.11.2,
|
||||||
|
any.base-orphans ==0.8.4,
|
||||||
|
any.base16-bytestring ==1.0.1.0,
|
||||||
|
any.base64-bytestring ==1.2.0.1,
|
||||||
|
any.bifunctors ==5.5.10,
|
||||||
|
bifunctors +semigroups +tagged,
|
||||||
|
any.binary ==0.8.7.0,
|
||||||
|
any.blaze-builder ==0.4.2.1,
|
||||||
|
any.bytestring ==0.10.10.1,
|
||||||
|
any.bz2 ==1.0.1.0,
|
||||||
|
bz2 -cross +with-bzlib,
|
||||||
|
any.c2hs ==0.28.7,
|
||||||
|
c2hs +base3 -regression,
|
||||||
|
any.call-stack ==0.3.0,
|
||||||
|
any.case-insensitive ==1.2.1.0,
|
||||||
|
any.casing ==0.1.4.1,
|
||||||
|
any.cereal ==0.5.8.1,
|
||||||
|
cereal -bytestring-builder,
|
||||||
|
any.chs-cabal ==0.1.1.0,
|
||||||
|
any.chs-deps ==0.1.0.0,
|
||||||
|
chs-deps -cross,
|
||||||
|
any.clock ==0.8.2,
|
||||||
|
clock -llvm,
|
||||||
|
any.cmdargs ==0.10.21,
|
||||||
|
cmdargs +quotation -testprog,
|
||||||
|
any.colour ==2.3.5,
|
||||||
|
any.comonad ==5.0.8,
|
||||||
|
comonad +containers +distributive +indexed-traversable,
|
||||||
|
any.composition-prelude ==3.0.0.2,
|
||||||
|
composition-prelude -development,
|
||||||
|
any.concurrent-output ==1.10.12,
|
||||||
|
any.conduit ==1.3.4.1,
|
||||||
|
any.conduit-extra ==1.3.5,
|
||||||
|
any.containers ==0.6.2.1,
|
||||||
|
any.contravariant ==1.5.3,
|
||||||
|
contravariant +semigroups +statevar +tagged,
|
||||||
|
any.cryptohash-sha256 ==0.11.102.0,
|
||||||
|
cryptohash-sha256 -exe +use-cbits,
|
||||||
|
any.data-default-class ==0.1.2.0,
|
||||||
|
any.data-fix ==0.3.1,
|
||||||
|
any.deepseq ==1.4.4.0,
|
||||||
|
any.deferred-folds ==0.9.17,
|
||||||
|
any.directory ==1.3.6.0,
|
||||||
|
any.distributive ==0.6.2.1,
|
||||||
|
distributive +semigroups +tagged,
|
||||||
|
any.dlist ==1.0,
|
||||||
|
dlist -werror,
|
||||||
|
any.easy-file ==0.2.2,
|
||||||
|
any.errors ==2.3.0,
|
||||||
|
any.exceptions ==0.10.4,
|
||||||
|
exceptions +transformers-0-4,
|
||||||
|
any.fast-logger ==3.0.3,
|
||||||
|
any.filepath ==1.4.2.1,
|
||||||
|
any.focus ==1.0.2,
|
||||||
|
any.foldl ==1.4.11,
|
||||||
|
any.free ==5.1.6,
|
||||||
|
any.fusion-plugin-types ==0.1.0,
|
||||||
|
any.generic-arbitrary ==0.1.0,
|
||||||
|
any.generics-sop ==0.5.1.1,
|
||||||
|
any.ghc-boot-th ==8.8.4,
|
||||||
|
any.ghc-prim ==0.5.3,
|
||||||
|
ghcup -internal-downloader -tar -tui,
|
||||||
|
any.happy ==1.20.0,
|
||||||
|
any.hashable ==1.3.1.0,
|
||||||
|
hashable +integer-gmp,
|
||||||
|
any.haskell-src-exts ==1.23.1,
|
||||||
|
any.haskell-src-meta ==0.8.7,
|
||||||
|
any.haskus-utils-data ==1.4,
|
||||||
|
any.haskus-utils-types ==1.5.1,
|
||||||
|
any.haskus-utils-variant ==3.1,
|
||||||
|
any.heaps ==0.3.6.1,
|
||||||
|
any.hpath ==0.11.0,
|
||||||
|
any.hpath-directory ==0.14.1,
|
||||||
|
any.hpath-filepath ==0.10.4,
|
||||||
|
any.hpath-io ==0.14.1,
|
||||||
|
any.hpath-posix ==0.13.2,
|
||||||
|
any.hsc2hs ==0.68.7,
|
||||||
|
hsc2hs -in-ghc-tree,
|
||||||
|
any.hspec ==2.7.8,
|
||||||
|
any.hspec-core ==2.7.8,
|
||||||
|
any.hspec-discover ==2.7.8,
|
||||||
|
any.hspec-expectations ==0.8.2,
|
||||||
|
any.hspec-golden-aeson ==0.7.0.0,
|
||||||
|
any.indexed-profunctors ==0.1,
|
||||||
|
any.indexed-traversable ==0.1.1,
|
||||||
|
any.indexed-traversable-instances ==0.1,
|
||||||
|
any.integer-gmp ==1.0.2.0,
|
||||||
|
any.integer-logarithms ==1.0.3.1,
|
||||||
|
integer-logarithms -check-bounds +integer-gmp,
|
||||||
|
any.language-c ==0.8.3,
|
||||||
|
language-c -allwarnings +iecfpextension +separatesyb +usebytestrings,
|
||||||
|
any.libarchive ==3.0.2.1,
|
||||||
|
libarchive -cross -low-memory -system-libarchive,
|
||||||
|
any.libyaml ==0.1.2,
|
||||||
|
libyaml -no-unicode -system-libyaml,
|
||||||
|
any.lifted-base ==0.2.3.12,
|
||||||
|
any.list-t ==1.0.4,
|
||||||
|
any.lockfree-queue ==0.2.3.1,
|
||||||
|
lzma -static,
|
||||||
|
any.math-functions ==0.3.4.1,
|
||||||
|
math-functions +system-erf +system-expm1,
|
||||||
|
any.megaparsec ==9.0.1,
|
||||||
|
megaparsec -dev,
|
||||||
|
any.microlens ==0.4.12.0,
|
||||||
|
any.microlens-mtl ==0.2.0.1,
|
||||||
|
any.microlens-th ==0.4.3.9,
|
||||||
|
any.mmorph ==1.1.5,
|
||||||
|
any.monad-control ==1.0.2.3,
|
||||||
|
any.monad-logger ==0.3.36,
|
||||||
|
monad-logger +template_haskell,
|
||||||
|
any.monad-loops ==0.4.3,
|
||||||
|
monad-loops +base4,
|
||||||
|
any.mono-traversable ==1.0.15.1,
|
||||||
|
any.mtl ==2.2.2,
|
||||||
|
any.mwc-random ==0.15.0.1,
|
||||||
|
any.network ==3.1.2.1,
|
||||||
|
network -devel,
|
||||||
|
any.old-locale ==1.0.0.7,
|
||||||
|
any.old-time ==1.1.0.3,
|
||||||
|
any.optics ==0.4,
|
||||||
|
any.optics-core ==0.4,
|
||||||
|
optics-core -explicit-generic-labels,
|
||||||
|
any.optics-extra ==0.4,
|
||||||
|
any.optics-th ==0.4,
|
||||||
|
any.optics-vl ==0.2.1,
|
||||||
|
any.optparse-applicative ==0.16.1.0,
|
||||||
|
optparse-applicative +process,
|
||||||
|
any.os-release ==1.0.2,
|
||||||
|
os-release -devel,
|
||||||
|
any.parallel ==3.2.2.0,
|
||||||
|
any.parsec ==3.1.14.0,
|
||||||
|
any.parser-combinators ==1.3.0,
|
||||||
|
parser-combinators -dev,
|
||||||
|
any.pretty ==1.1.3.6,
|
||||||
|
any.pretty-terminal ==0.1.0.0,
|
||||||
|
any.primitive ==0.7.1.0,
|
||||||
|
any.primitive-extras ==0.8.2,
|
||||||
|
any.primitive-unlifted ==0.1.3.0,
|
||||||
|
any.process ==1.6.9.0,
|
||||||
|
any.profunctors ==5.6.2,
|
||||||
|
any.quickcheck-arbitrary-adt ==0.3.1.0,
|
||||||
|
any.quickcheck-io ==0.2.0,
|
||||||
|
any.random ==1.2.0,
|
||||||
|
any.recursion-schemes ==5.2.2,
|
||||||
|
recursion-schemes +template-haskell,
|
||||||
|
any.regex-posix ==0.96.0.0,
|
||||||
|
regex-posix -_regex-posix-clib,
|
||||||
|
any.resourcet ==1.2.4.2,
|
||||||
|
any.rts ==1.0,
|
||||||
|
any.safe ==0.3.19,
|
||||||
|
any.safe-exceptions ==0.1.7.1,
|
||||||
|
any.scientific ==0.3.6.2,
|
||||||
|
scientific -bytestring-builder -integer-simple,
|
||||||
|
any.semigroupoids ==5.3.5,
|
||||||
|
semigroupoids +comonad +containers +contravariant +distributive +tagged +unordered-containers,
|
||||||
|
any.setenv ==0.1.1.3,
|
||||||
|
any.sop-core ==0.5.0.1,
|
||||||
|
any.split ==0.2.3.4,
|
||||||
|
any.splitmix ==0.1.0.3,
|
||||||
|
splitmix -optimised-mixer,
|
||||||
|
any.stm ==2.5.0.0,
|
||||||
|
any.stm-chans ==3.0.0.4,
|
||||||
|
any.streaming-commons ==0.2.2.1,
|
||||||
|
streaming-commons -use-bytestring-builder,
|
||||||
|
any.streamly ==0.7.3,
|
||||||
|
streamly -debug -dev -examples -examples-sdl -fusion-plugin -has-llvm -inspection -no-fusion -streamk,
|
||||||
|
any.streamly-bytestring ==0.1.2,
|
||||||
|
any.streamly-posix ==0.1.0.1,
|
||||||
|
any.strict ==0.4.0.1,
|
||||||
|
strict +assoc,
|
||||||
|
any.strict-base ==0.4.0.0,
|
||||||
|
any.string-interpolate ==0.3.1.0,
|
||||||
|
string-interpolate -bytestring-builder -extended-benchmarks -text-builder,
|
||||||
|
any.syb ==0.7.2.1,
|
||||||
|
any.tagged ==0.8.6.1,
|
||||||
|
tagged +deepseq +transformers,
|
||||||
|
any.tasty ==1.3.1,
|
||||||
|
tasty +clock,
|
||||||
|
any.tasty-hunit ==0.10.0.3,
|
||||||
|
any.tasty-quickcheck ==0.10.1.2,
|
||||||
|
any.template-haskell ==2.15.0.0,
|
||||||
|
any.terminal-size ==0.3.2.1,
|
||||||
|
any.terminfo ==0.4.1.4,
|
||||||
|
any.text ==1.2.4.0,
|
||||||
|
any.text-conversions ==0.3.1,
|
||||||
|
any.text-short ==0.1.3,
|
||||||
|
text-short -asserts,
|
||||||
|
any.tf-random ==0.5,
|
||||||
|
any.th-abstraction ==0.4.2.0,
|
||||||
|
any.th-compat ==0.1.1,
|
||||||
|
any.th-lift ==0.8.2,
|
||||||
|
any.th-lift-instances ==0.1.18,
|
||||||
|
any.th-orphans ==0.13.11,
|
||||||
|
any.th-reify-many ==0.1.9,
|
||||||
|
any.these ==1.1.1.1,
|
||||||
|
these +assoc,
|
||||||
|
any.time ==1.9.3,
|
||||||
|
any.time-compat ==1.9.5,
|
||||||
|
time-compat -old-locale,
|
||||||
|
any.transformers ==0.5.6.2,
|
||||||
|
any.transformers-base ==0.4.5.2,
|
||||||
|
transformers-base +orphaninstances,
|
||||||
|
any.transformers-compat ==0.6.6,
|
||||||
|
transformers-compat -five +five-three -four +generic-deriving +mtl -three -two,
|
||||||
|
any.typed-process ==0.2.6.0,
|
||||||
|
any.unbounded-delays ==0.1.1.1,
|
||||||
|
any.unix ==2.7.2.2,
|
||||||
|
any.unix-bytestring ==0.3.7.3,
|
||||||
|
any.unix-compat ==0.5.3,
|
||||||
|
unix-compat -old-time,
|
||||||
|
any.unix-time ==0.4.7,
|
||||||
|
any.unliftio-core ==0.2.0.1,
|
||||||
|
any.unordered-containers ==0.2.13.0,
|
||||||
|
unordered-containers -debug,
|
||||||
|
any.uri-bytestring ==0.3.3.0,
|
||||||
|
uri-bytestring -lib-werror,
|
||||||
|
any.utf8-string ==1.0.2,
|
||||||
|
any.uuid-types ==1.0.4,
|
||||||
|
any.vector ==0.12.2.0,
|
||||||
|
vector +boundschecks -internalchecks -unsafechecks -wall,
|
||||||
|
any.vector-algorithms ==0.8.0.4,
|
||||||
|
vector-algorithms +bench +boundschecks -internalchecks -llvm +properties -unsafechecks,
|
||||||
|
any.versions ==4.0.3,
|
||||||
|
any.vty ==5.33,
|
||||||
|
any.wcwidth ==0.0.2,
|
||||||
|
wcwidth -cli +split-base,
|
||||||
|
any.word8 ==0.1.3,
|
||||||
|
any.yaml ==0.11.5.0,
|
||||||
|
yaml +no-examples +no-exe,
|
||||||
|
zlib -non-blocking-ffi -pkg-config -static
|
||||||
|
index-state: hackage.haskell.org 2021-03-07T18:36:25Z
|
||||||
@@ -1,19 +1,26 @@
|
|||||||
packages: ./ghcup.cabal
|
packages: ./ghcup.cabal
|
||||||
|
|
||||||
with-compiler: ghc-8.8.3
|
optional-packages: ./vendored/*/*.cabal
|
||||||
|
|
||||||
optimization: 2
|
optimization: 2
|
||||||
|
|
||||||
package streamly
|
|
||||||
ghc-options: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
|
||||||
|
|
||||||
package ghcup
|
package ghcup
|
||||||
ghc-options: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
tests: True
|
||||||
|
flags: +tui
|
||||||
|
|
||||||
package tar-bytestring
|
source-repository-package
|
||||||
ghc-options: -O2
|
type: git
|
||||||
|
location: https://github.com/Bodigrim/tar
|
||||||
|
tag: ac197ec7ea4838dc2b4e22b9b888b080cedf29cf
|
||||||
|
|
||||||
|
source-repository-package
|
||||||
|
type: git
|
||||||
|
location: https://github.com/bgamari/terminal-size
|
||||||
|
tag: 34ea816bd63f75f800eedac12c6908c6f3736036
|
||||||
|
|
||||||
constraints: http-io-streams -brotli
|
constraints: http-io-streams -brotli
|
||||||
|
|
||||||
allow-newer: base
|
package libarchive
|
||||||
|
flags: -system-libarchive
|
||||||
|
|
||||||
|
allow-newer: base, ghc-prim, template-haskell, language-c
|
||||||
|
|||||||
@@ -1,238 +1,2 @@
|
|||||||
constraints: any.Cabal ==3.0.1.0,
|
-- windows picks weird version
|
||||||
any.HsOpenSSL ==0.11.4.18,
|
constraints: any.hsc2hs ==0.68.7
|
||||||
HsOpenSSL -fast-bignum -homebrew-openssl -macports-openssl -old-locale,
|
|
||||||
any.IfElse ==0.85,
|
|
||||||
any.QuickCheck ==2.14,
|
|
||||||
QuickCheck +templatehaskell,
|
|
||||||
any.StateVar ==1.2,
|
|
||||||
any.abstract-deque ==0.3,
|
|
||||||
abstract-deque -usecas,
|
|
||||||
any.aeson ==1.4.7.1,
|
|
||||||
aeson -bytestring-builder -cffi -developer -fast,
|
|
||||||
any.aeson-pretty ==0.8.8,
|
|
||||||
aeson-pretty -lib-only,
|
|
||||||
any.alex ==3.2.5,
|
|
||||||
alex +small_base,
|
|
||||||
any.ansi-terminal ==0.10.3,
|
|
||||||
ansi-terminal -example,
|
|
||||||
any.ansi-wl-pprint ==0.6.9,
|
|
||||||
ansi-wl-pprint -example,
|
|
||||||
any.array ==0.5.4.0,
|
|
||||||
any.ascii-string ==1.0.1.4,
|
|
||||||
any.assoc ==1.0.1,
|
|
||||||
any.async ==2.2.2,
|
|
||||||
async -bench,
|
|
||||||
any.atomic-primops ==0.8.3,
|
|
||||||
atomic-primops -debug,
|
|
||||||
any.attoparsec ==0.13.2.4,
|
|
||||||
attoparsec -developer,
|
|
||||||
any.auto-update ==0.1.6,
|
|
||||||
any.base ==4.13.0.0,
|
|
||||||
any.base-compat ==0.11.1,
|
|
||||||
any.base-compat-batteries ==0.11.1,
|
|
||||||
any.base-orphans ==0.8.2,
|
|
||||||
any.base-prelude ==1.3,
|
|
||||||
any.base16-bytestring ==0.1.1.6,
|
|
||||||
any.base64-bytestring ==1.0.0.3,
|
|
||||||
any.bifunctors ==5.5.7,
|
|
||||||
bifunctors +semigroups +tagged,
|
|
||||||
any.binary ==0.8.7.0,
|
|
||||||
any.blaze-builder ==0.4.1.0,
|
|
||||||
any.bytestring ==0.10.10.0,
|
|
||||||
any.bytestring-builder ==0.10.8.2.0,
|
|
||||||
bytestring-builder +bytestring_has_builder,
|
|
||||||
any.bz2 ==1.0.0.2,
|
|
||||||
bz2 -cross +with-bzlib,
|
|
||||||
any.c2hs ==0.28.6,
|
|
||||||
c2hs +base3 -regression,
|
|
||||||
any.cabal-doctest ==1.0.8,
|
|
||||||
any.case-insensitive ==1.2.1.0,
|
|
||||||
any.cereal ==0.5.8.1,
|
|
||||||
cereal -bytestring-builder,
|
|
||||||
any.clock ==0.8,
|
|
||||||
clock -llvm,
|
|
||||||
any.cmdargs ==0.10.20,
|
|
||||||
cmdargs +quotation -testprog,
|
|
||||||
any.colour ==2.3.5,
|
|
||||||
any.comonad ==5.0.6,
|
|
||||||
comonad +containers +distributive +test-doctests,
|
|
||||||
any.concurrent-output ==1.10.11,
|
|
||||||
any.conduit ==1.3.2,
|
|
||||||
any.conduit-extra ==1.3.5,
|
|
||||||
any.containers ==0.6.2.1,
|
|
||||||
any.contravariant ==1.5.2,
|
|
||||||
contravariant +semigroups +statevar +tagged,
|
|
||||||
any.data-default-class ==0.1.2.0,
|
|
||||||
any.data-default-instances-base ==0.1.0.1,
|
|
||||||
any.deepseq ==1.4.4.0,
|
|
||||||
any.deferred-folds ==0.9.10.1,
|
|
||||||
any.directory ==1.3.6.0,
|
|
||||||
any.distributive ==0.6.1,
|
|
||||||
distributive +semigroups +tagged,
|
|
||||||
any.dlist ==0.8.0.8,
|
|
||||||
any.easy-file ==0.2.2,
|
|
||||||
any.errors ==2.3.0,
|
|
||||||
any.exceptions ==0.10.4,
|
|
||||||
exceptions +transformers-0-4,
|
|
||||||
any.extra ==1.7.1,
|
|
||||||
any.fast-logger ==3.0.1,
|
|
||||||
any.filepath ==1.4.2.1,
|
|
||||||
any.focus ==1.0.1.3,
|
|
||||||
any.foldl ==1.4.6,
|
|
||||||
any.free ==5.1.3,
|
|
||||||
any.fusion-plugin-types ==0.1.0,
|
|
||||||
any.generics-sop ==0.5.1.0,
|
|
||||||
any.ghc-boot-th ==8.8.3,
|
|
||||||
any.ghc-prim ==0.5.3,
|
|
||||||
any.happy ==1.19.12,
|
|
||||||
happy +small_base,
|
|
||||||
any.hashable ==1.3.0.0,
|
|
||||||
hashable -examples +integer-gmp +sse2 -sse41,
|
|
||||||
any.haskell-src-exts ==1.23.0,
|
|
||||||
any.haskell-src-meta ==0.8.5,
|
|
||||||
any.haskus-utils-data ==1.2,
|
|
||||||
any.haskus-utils-types ==1.5,
|
|
||||||
any.haskus-utils-variant ==3.0,
|
|
||||||
any.heaps ==0.3.6.1,
|
|
||||||
any.hopenssl ==2.2.4,
|
|
||||||
hopenssl -link-libz,
|
|
||||||
any.hpath ==0.11.0,
|
|
||||||
any.hpath-directory ==0.13.2,
|
|
||||||
any.hpath-filepath ==0.10.4,
|
|
||||||
any.hpath-io ==0.13.1,
|
|
||||||
any.hpath-posix ==0.13.1,
|
|
||||||
any.hsc2hs ==0.68.7,
|
|
||||||
hsc2hs -in-ghc-tree,
|
|
||||||
any.http-io-streams ==0.1.2.0,
|
|
||||||
http-io-streams -brotli,
|
|
||||||
any.indexed-profunctors ==0.1,
|
|
||||||
any.integer-gmp ==1.0.2.0,
|
|
||||||
any.integer-logarithms ==1.0.3,
|
|
||||||
integer-logarithms -check-bounds +integer-gmp,
|
|
||||||
any.io-streams ==1.5.1.0,
|
|
||||||
io-streams -nointeractivetests,
|
|
||||||
any.language-bash ==0.9.0,
|
|
||||||
any.language-c ==0.8.3,
|
|
||||||
language-c -allwarnings +iecfpextension +separatesyb +usebytestrings,
|
|
||||||
any.lifted-base ==0.2.3.12,
|
|
||||||
any.list-t ==1.0.4,
|
|
||||||
any.lockfree-queue ==0.2.3.1,
|
|
||||||
any.lzma ==0.0.0.3,
|
|
||||||
any.math-functions ==0.3.3.0,
|
|
||||||
math-functions +system-erf +system-expm1,
|
|
||||||
any.megaparsec ==8.0.0,
|
|
||||||
megaparsec -dev,
|
|
||||||
any.mmorph ==1.1.3,
|
|
||||||
any.monad-control ==1.0.2.3,
|
|
||||||
any.monad-logger ==0.3.32,
|
|
||||||
monad-logger +template_haskell,
|
|
||||||
any.monad-loops ==0.4.3,
|
|
||||||
monad-loops +base4,
|
|
||||||
any.mono-traversable ==1.0.15.1,
|
|
||||||
any.mtl ==2.2.2,
|
|
||||||
any.mwc-random ==0.14.0.0,
|
|
||||||
any.network ==3.1.1.1,
|
|
||||||
any.network-uri ==2.6.3.0,
|
|
||||||
any.old-locale ==1.0.0.7,
|
|
||||||
any.old-time ==1.1.0.3,
|
|
||||||
any.openssl-streams ==1.2.2.0,
|
|
||||||
any.optics ==0.2,
|
|
||||||
any.optics-core ==0.2,
|
|
||||||
any.optics-extra ==0.2,
|
|
||||||
any.optics-th ==0.2,
|
|
||||||
any.optics-vl ==0.2,
|
|
||||||
any.optparse-applicative ==0.15.1.0,
|
|
||||||
any.parsec ==3.1.14.0,
|
|
||||||
any.parser-combinators ==1.2.1,
|
|
||||||
parser-combinators -dev,
|
|
||||||
any.pretty ==1.1.3.6,
|
|
||||||
any.pretty-terminal ==0.1.0.0,
|
|
||||||
any.prettyprinter ==1.6.1,
|
|
||||||
prettyprinter -buildreadme,
|
|
||||||
any.primitive ==0.7.0.1,
|
|
||||||
any.primitive-extras ==0.8,
|
|
||||||
any.primitive-unlifted ==0.1.3.0,
|
|
||||||
any.process ==1.6.8.0,
|
|
||||||
any.profunctors ==5.5.2,
|
|
||||||
any.random ==1.1,
|
|
||||||
any.recursion-schemes ==5.1.3,
|
|
||||||
recursion-schemes +template-haskell,
|
|
||||||
any.regex-base ==0.94.0.0,
|
|
||||||
any.regex-posix ==0.96.0.0,
|
|
||||||
regex-posix -_regex-posix-clib,
|
|
||||||
any.resourcet ==1.2.3,
|
|
||||||
any.rts ==1.0,
|
|
||||||
any.safe ==0.3.18,
|
|
||||||
any.safe-exceptions ==0.1.7.0,
|
|
||||||
any.scientific ==0.3.6.2,
|
|
||||||
scientific -bytestring-builder -integer-simple,
|
|
||||||
any.semigroupoids ==5.3.4,
|
|
||||||
semigroupoids +comonad +containers +contravariant +distributive +doctests +tagged +unordered-containers,
|
|
||||||
any.semigroups ==0.19.1,
|
|
||||||
semigroups +binary +bytestring -bytestring-builder +containers +deepseq +hashable +tagged +template-haskell +text +transformers +unordered-containers,
|
|
||||||
any.sop-core ==0.5.0.1,
|
|
||||||
any.split ==0.2.3.4,
|
|
||||||
any.splitmix ==0.0.4,
|
|
||||||
splitmix -optimised-mixer +random,
|
|
||||||
any.stm ==2.5.0.0,
|
|
||||||
any.stm-chans ==3.0.0.4,
|
|
||||||
any.streaming-commons ==0.2.1.2,
|
|
||||||
streaming-commons -use-bytestring-builder,
|
|
||||||
any.streamly ==0.7.1,
|
|
||||||
streamly -debug -dev -examples -examples-sdl -fusion-plugin -has-llvm -inspection -no-charts -no-fusion -streamk,
|
|
||||||
any.streamly-bytestring ==0.1.2,
|
|
||||||
any.streamly-posix ==0.1.0.0,
|
|
||||||
any.strict-base ==0.4.0.0,
|
|
||||||
any.string-interpolate ==0.2.0.0,
|
|
||||||
any.syb ==0.7.1,
|
|
||||||
any.table-layout ==0.8.0.5,
|
|
||||||
any.tagged ==0.8.6,
|
|
||||||
tagged +deepseq +transformers,
|
|
||||||
any.tar-bytestring ==0.6.3.1,
|
|
||||||
any.template-haskell ==2.15.0.0,
|
|
||||||
any.terminal-progress-bar ==0.4.1,
|
|
||||||
any.terminal-size ==0.3.2.1,
|
|
||||||
any.text ==1.2.4.0,
|
|
||||||
any.text-conversions ==0.3.0,
|
|
||||||
any.text-short ==0.1.3,
|
|
||||||
text-short -asserts,
|
|
||||||
any.th-abstraction ==0.3.2.0,
|
|
||||||
any.th-expand-syns ==0.4.6.0,
|
|
||||||
any.th-lift ==0.8.1,
|
|
||||||
any.th-lift-instances ==0.1.14,
|
|
||||||
any.th-orphans ==0.13.9,
|
|
||||||
any.th-reify-many ==0.1.9,
|
|
||||||
any.these ==1.0.1,
|
|
||||||
these +aeson +assoc +quickcheck +semigroupoids,
|
|
||||||
any.time ==1.9.3,
|
|
||||||
any.time-compat ==1.9.3,
|
|
||||||
time-compat -old-locale,
|
|
||||||
any.transformers ==0.5.6.2,
|
|
||||||
any.transformers-base ==0.4.5.2,
|
|
||||||
transformers-base +orphaninstances,
|
|
||||||
any.transformers-compat ==0.6.5,
|
|
||||||
transformers-compat -five +five-three -four +generic-deriving +mtl -three -two,
|
|
||||||
any.typed-process ==0.2.6.0,
|
|
||||||
any.unix ==2.7.2.2,
|
|
||||||
any.unix-bytestring ==0.3.7.3,
|
|
||||||
any.unix-compat ==0.5.2,
|
|
||||||
unix-compat -old-time,
|
|
||||||
any.unix-time ==0.4.7,
|
|
||||||
any.unliftio-core ==0.2.0.1,
|
|
||||||
any.unordered-containers ==0.2.10.0,
|
|
||||||
unordered-containers -debug,
|
|
||||||
any.uri-bytestring ==0.3.2.2,
|
|
||||||
uri-bytestring -lib-werror,
|
|
||||||
any.utf8-string ==1.0.1.1,
|
|
||||||
any.uuid-types ==1.0.3,
|
|
||||||
any.vector ==0.12.1.2,
|
|
||||||
vector +boundschecks -internalchecks -unsafechecks -wall,
|
|
||||||
any.vector-algorithms ==0.8.0.3,
|
|
||||||
vector-algorithms +bench +boundschecks -internalchecks -llvm +properties -unsafechecks,
|
|
||||||
any.vector-builder ==0.3.8,
|
|
||||||
any.vector-th-unbox ==0.2.1.7,
|
|
||||||
any.versions ==3.5.3,
|
|
||||||
any.word8 ==0.1.3,
|
|
||||||
any.zlib ==0.6.2.1,
|
|
||||||
zlib -non-blocking-ffi -pkg-config,
|
|
||||||
any.zlib-bindings ==0.1.1.5
|
|
||||||
|
|||||||
63
config.yaml
Normal file
63
config.yaml
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
# 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'
|
||||||
|
show-all-tools:
|
||||||
|
KChar: 't'
|
||||||
|
|
||||||
|
# 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,14 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -ex
|
|
||||||
|
|
||||||
cd /app
|
|
||||||
|
|
||||||
cabal v2-update
|
|
||||||
|
|
||||||
cabal v2-install \
|
|
||||||
--install-method=copy \
|
|
||||||
--overwrite-policy=always \
|
|
||||||
--installdir="/bin" \
|
|
||||||
--ghc-options='-optl-static'
|
|
||||||
|
|
||||||
1415
ghcup-0.0.2.yaml
Normal file
1415
ghcup-0.0.2.yaml
Normal file
File diff suppressed because it is too large
Load Diff
1500
ghcup-0.0.3.yaml
Normal file
1500
ghcup-0.0.3.yaml
Normal file
File diff suppressed because it is too large
Load Diff
1932
ghcup-0.0.4.yaml
Normal file
1932
ghcup-0.0.4.yaml
Normal file
File diff suppressed because it is too large
Load Diff
2175
ghcup-0.0.5.yaml
Normal file
2175
ghcup-0.0.5.yaml
Normal file
File diff suppressed because it is too large
Load Diff
643
ghcup.cabal
643
ghcup.cabal
@@ -1,217 +1,197 @@
|
|||||||
cabal-version: 3.0
|
cabal-version: 3.0
|
||||||
name: ghcup
|
name: ghcup
|
||||||
version: 0.1.0.0
|
version: 0.1.15.2
|
||||||
synopsis: ghc toolchain installer as an exe/library
|
license: LGPL-3.0-only
|
||||||
|
license-file: LICENSE
|
||||||
|
copyright: Julian Ospald 2020
|
||||||
|
maintainer: hasufell@posteo.de
|
||||||
|
author: Julian Ospald
|
||||||
|
homepage: https://gitlab.haskell.org/haskell/ghcup-hs
|
||||||
|
bug-reports: https://gitlab.haskell.org/haskell/ghcup-hs/issues
|
||||||
|
synopsis: ghc toolchain installer
|
||||||
description:
|
description:
|
||||||
A rewrite of the shell script ghcup, for providing
|
A rewrite of the shell script ghcup, for providing
|
||||||
a more stable user experience and exposing an API.
|
a more stable user experience and exposing an API.
|
||||||
|
|
||||||
homepage: https://github.com/hasufell/ghcup-hs
|
category: System
|
||||||
bug-reports: https://github.com/hasufell/ghcup-hs/issues
|
build-type: Simple
|
||||||
license: LGPL-3.0-only
|
extra-doc-files:
|
||||||
license-file: LICENSE
|
CHANGELOG.md
|
||||||
author: Julian Ospald
|
config.yaml
|
||||||
maintainer: hasufell@posteo.de
|
ghcup-0.0.4.yaml
|
||||||
copyright: Julian Ospald 2020
|
ghcup-0.0.5.yaml
|
||||||
category: System
|
HACKING.md
|
||||||
build-type: Simple
|
README.md
|
||||||
extra-source-files: CHANGELOG.md
|
RELEASING.md
|
||||||
|
|
||||||
source-repository head
|
source-repository head
|
||||||
type: git
|
type: git
|
||||||
location: https://github.com/hasufell/ghcup-hs
|
location: https://gitlab.haskell.org/haskell/ghcup-hs.git
|
||||||
|
|
||||||
|
flag tui
|
||||||
|
description:
|
||||||
|
Build the brick powered tui (ghcup tui). This is disabled on windows.
|
||||||
|
|
||||||
flag Curl
|
|
||||||
description: Use curl instead of http-io-streams for download
|
|
||||||
default: False
|
default: False
|
||||||
manual: True
|
manual: True
|
||||||
|
|
||||||
common HsOpenSSL
|
flag internal-downloader
|
||||||
build-depends: HsOpenSSL >=0.11.4.18
|
description:
|
||||||
|
Compile the internal downloader, which links against OpenSSL. This is disabled on windows.
|
||||||
|
|
||||||
common aeson
|
default: False
|
||||||
build-depends: aeson >=1.4
|
manual: True
|
||||||
|
|
||||||
common aeson-pretty
|
flag tar
|
||||||
build-depends: aeson-pretty >=0.8.8
|
description: Use tar-bytestring instead of libarchive.
|
||||||
|
default: False
|
||||||
|
manual: True
|
||||||
|
|
||||||
common ascii-string
|
library
|
||||||
build-depends: ascii-string >=1.0
|
exposed-modules:
|
||||||
|
GHCup
|
||||||
|
GHCup.Download
|
||||||
|
GHCup.Download.Utils
|
||||||
|
GHCup.Errors
|
||||||
|
GHCup.Platform
|
||||||
|
GHCup.Requirements
|
||||||
|
GHCup.Types
|
||||||
|
GHCup.Types.JSON
|
||||||
|
GHCup.Types.Optics
|
||||||
|
GHCup.Utils
|
||||||
|
GHCup.Utils.Dirs
|
||||||
|
GHCup.Utils.File
|
||||||
|
GHCup.Utils.File.Common
|
||||||
|
GHCup.Utils.Logger
|
||||||
|
GHCup.Utils.MegaParsec
|
||||||
|
GHCup.Utils.Prelude
|
||||||
|
GHCup.Utils.String.QQ
|
||||||
|
GHCup.Utils.Version.QQ
|
||||||
|
GHCup.Version
|
||||||
|
|
||||||
common async
|
hs-source-dirs: lib
|
||||||
build-depends: async >=0.8
|
other-modules: Paths_ghcup
|
||||||
|
autogen-modules: Paths_ghcup
|
||||||
common attoparsec
|
|
||||||
build-depends: attoparsec >=0.13
|
|
||||||
|
|
||||||
common base
|
|
||||||
build-depends: base >=4.12 && <5
|
|
||||||
|
|
||||||
common binary
|
|
||||||
build-depends: binary >=0.8.6.0
|
|
||||||
|
|
||||||
common bytestring
|
|
||||||
build-depends: bytestring >=0.10
|
|
||||||
|
|
||||||
common bz2
|
|
||||||
build-depends: bz2 >=0.5.0.5
|
|
||||||
|
|
||||||
common case-insensitive
|
|
||||||
build-depends: case-insensitive >=1.2.1.0
|
|
||||||
|
|
||||||
common concurrent-output
|
|
||||||
build-depends: concurrent-output >=1.10.11
|
|
||||||
|
|
||||||
common containers
|
|
||||||
build-depends: containers >=0.6
|
|
||||||
|
|
||||||
common generics-sop
|
|
||||||
build-depends: generics-sop >=0.5
|
|
||||||
|
|
||||||
common haskus-utils-types
|
|
||||||
build-depends: haskus-utils-types >=1.5
|
|
||||||
|
|
||||||
common haskus-utils-variant
|
|
||||||
build-depends: haskus-utils-variant >=3.0
|
|
||||||
|
|
||||||
common hopenssl
|
|
||||||
build-depends: hopenssl >=2.2.4
|
|
||||||
|
|
||||||
common hpath
|
|
||||||
build-depends: hpath >=0.11
|
|
||||||
|
|
||||||
common hpath-directory
|
|
||||||
build-depends: hpath-directory >=0.13.2
|
|
||||||
|
|
||||||
common hpath-filepath
|
|
||||||
build-depends: hpath-filepath >=0.10.3
|
|
||||||
|
|
||||||
common hpath-io
|
|
||||||
build-depends: hpath-io >=0.13.1
|
|
||||||
|
|
||||||
common hpath-posix
|
|
||||||
build-depends: hpath-posix >=0.11.1
|
|
||||||
|
|
||||||
common http-io-streams
|
|
||||||
build-depends: http-io-streams >=0.1.2.0
|
|
||||||
|
|
||||||
common io-streams
|
|
||||||
build-depends: io-streams >=1.5
|
|
||||||
|
|
||||||
common language-bash
|
|
||||||
build-depends: language-bash >=0.9
|
|
||||||
|
|
||||||
common lzma
|
|
||||||
build-depends: lzma >=0.0.0.3
|
|
||||||
|
|
||||||
common megaparsec
|
|
||||||
build-depends: megaparsec >=8.0.0
|
|
||||||
|
|
||||||
common monad-logger
|
|
||||||
build-depends: monad-logger >=0.3.31
|
|
||||||
|
|
||||||
common mtl
|
|
||||||
build-depends: mtl >=2.2
|
|
||||||
|
|
||||||
common optics
|
|
||||||
build-depends: optics >=0.2
|
|
||||||
|
|
||||||
common optics-vl
|
|
||||||
build-depends: optics-vl >=0.2
|
|
||||||
|
|
||||||
common optparse-applicative
|
|
||||||
build-depends: optparse-applicative >=0.15.1.0
|
|
||||||
|
|
||||||
common parsec
|
|
||||||
build-depends: parsec >=3.1
|
|
||||||
|
|
||||||
common pretty-terminal
|
|
||||||
build-depends: pretty-terminal >=0.1.0.0
|
|
||||||
|
|
||||||
common regex-posix
|
|
||||||
build-depends: regex-posix >=0.96
|
|
||||||
|
|
||||||
common resourcet
|
|
||||||
build-depends: resourcet >=1.2.2
|
|
||||||
|
|
||||||
common safe
|
|
||||||
build-depends: safe >=0.3.18
|
|
||||||
|
|
||||||
common safe-exceptions
|
|
||||||
build-depends: safe-exceptions >=0.1
|
|
||||||
|
|
||||||
common streamly
|
|
||||||
build-depends: streamly >=0.7.1
|
|
||||||
|
|
||||||
common streamly-posix
|
|
||||||
build-depends: streamly-posix >=0.1.0.0
|
|
||||||
|
|
||||||
common streamly-bytestring
|
|
||||||
build-depends: streamly-bytestring >=0.1.2
|
|
||||||
|
|
||||||
common strict-base
|
|
||||||
build-depends: strict-base >=0.4
|
|
||||||
|
|
||||||
common string-interpolate
|
|
||||||
build-depends: string-interpolate >=0.2.0.0
|
|
||||||
|
|
||||||
common table-layout
|
|
||||||
build-depends: table-layout >=0.8
|
|
||||||
|
|
||||||
common tar-bytestring
|
|
||||||
build-depends: tar-bytestring >=0.6.3.1
|
|
||||||
|
|
||||||
common template-haskell
|
|
||||||
build-depends: template-haskell >=2.7
|
|
||||||
|
|
||||||
common terminal-progress-bar
|
|
||||||
build-depends: terminal-progress-bar >=0.4.1
|
|
||||||
|
|
||||||
common text
|
|
||||||
build-depends: text >=1.2
|
|
||||||
|
|
||||||
common time
|
|
||||||
build-depends: time >=1.9.3
|
|
||||||
|
|
||||||
common transformers
|
|
||||||
build-depends: transformers >=0.5
|
|
||||||
|
|
||||||
common unix
|
|
||||||
build-depends: unix >=2.7
|
|
||||||
|
|
||||||
common unix-bytestring
|
|
||||||
build-depends: unix-bytestring >=0.3
|
|
||||||
|
|
||||||
common uri-bytestring
|
|
||||||
build-depends: uri-bytestring >=0.3.2.2
|
|
||||||
|
|
||||||
common utf8-string
|
|
||||||
build-depends: utf8-string >=1.0
|
|
||||||
|
|
||||||
common vector
|
|
||||||
build-depends: vector >=0.12
|
|
||||||
|
|
||||||
common versions
|
|
||||||
build-depends: versions >=3.5
|
|
||||||
|
|
||||||
common waargonaut
|
|
||||||
build-depends: waargonaut >=0.8
|
|
||||||
|
|
||||||
common word8
|
|
||||||
build-depends: word8 >=0.1.3
|
|
||||||
|
|
||||||
common zlib
|
|
||||||
build-depends: zlib >=0.6.2.1
|
|
||||||
|
|
||||||
common config
|
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
|
default-extensions:
|
||||||
|
DeriveGeneric
|
||||||
|
LambdaCase
|
||||||
|
MultiWayIf
|
||||||
|
NamedFieldPuns
|
||||||
|
PackageImports
|
||||||
|
QuasiQuotes
|
||||||
|
RecordWildCards
|
||||||
|
ScopedTypeVariables
|
||||||
|
Strict
|
||||||
|
StrictData
|
||||||
|
TupleSections
|
||||||
|
TypeApplications
|
||||||
|
TypeFamilies
|
||||||
|
ViewPatterns
|
||||||
|
|
||||||
ghc-options:
|
ghc-options:
|
||||||
-Wall -fwarn-tabs -fwarn-incomplete-uni-patterns
|
-Wall -fwarn-tabs -fwarn-incomplete-uni-patterns
|
||||||
-fwarn-incomplete-record-updates -threaded
|
-fwarn-incomplete-record-updates
|
||||||
|
|
||||||
|
build-depends:
|
||||||
|
, aeson >=1.4 && <1.6
|
||||||
|
, async >=0.8 && <2.3
|
||||||
|
, base >=4.13 && <5
|
||||||
|
, base16-bytestring >=0.1.1.6 && <1.1
|
||||||
|
, binary ^>=0.8.6.0
|
||||||
|
, bytestring ^>=0.10
|
||||||
|
, case-insensitive ^>=1.2.1.0
|
||||||
|
, casing ^>=0.1.4.1
|
||||||
|
, concurrent-output ^>=1.10.11
|
||||||
|
, containers ^>=0.6
|
||||||
|
, cryptohash-sha256 ^>=0.11.101.0
|
||||||
|
, deepseq ^>=1.4.4.0
|
||||||
|
, directory ^>=1.3.6.0
|
||||||
|
, disk-free-space ^>=0.1.0.1
|
||||||
|
, extra ^>=1.7.9
|
||||||
|
, filepath ^>=1.4.2.1
|
||||||
|
, generics-sop ^>=0.5
|
||||||
|
, haskus-utils-types ^>=1.5
|
||||||
|
, haskus-utils-variant >=3.0 && <3.2
|
||||||
|
, lzma-static ^>=5.2.5.3
|
||||||
|
, megaparsec >=8.0.0 && <9.1
|
||||||
|
, monad-logger ^>=0.3.31
|
||||||
|
, mtl ^>=2.2
|
||||||
|
, optics >=0.2 && <0.5
|
||||||
|
, optics-vl ^>=0.2
|
||||||
|
, os-release ^>=1.0.0
|
||||||
|
, parsec ^>=3.1
|
||||||
|
, pretty ^>=1.1.3.1
|
||||||
|
, pretty-terminal ^>=0.1.0.0
|
||||||
|
, regex-posix ^>=0.96
|
||||||
|
, resourcet ^>=1.2.2
|
||||||
|
, safe ^>=0.3.18
|
||||||
|
, safe-exceptions ^>=0.1
|
||||||
|
, split ^>=0.2.3.4
|
||||||
|
, strict-base ^>=0.4
|
||||||
|
, string-interpolate >=0.2.0.0 && <0.4
|
||||||
|
, template-haskell >=2.7 && <2.18
|
||||||
|
, temporary ^>=1.3
|
||||||
|
, text ^>=1.2.4.0
|
||||||
|
, time ^>=1.9.3
|
||||||
|
, transformers ^>=0.5
|
||||||
|
, unliftio-core ^>=0.2.0.1
|
||||||
|
, unordered-containers ^>=0.2.10.0
|
||||||
|
, uri-bytestring ^>=0.3.2.2
|
||||||
|
, utf8-string ^>=1.0
|
||||||
|
, vector ^>=0.12
|
||||||
|
, versions >=4.0.1 && <5.1
|
||||||
|
, word8 ^>=0.1.3
|
||||||
|
, yaml ^>=0.11.4.0
|
||||||
|
, zip ^>=1.7.1
|
||||||
|
, zlib ^>=0.6.2.2
|
||||||
|
|
||||||
|
if (flag(internal-downloader) && !os(windows))
|
||||||
|
exposed-modules: GHCup.Download.IOStreams
|
||||||
|
cpp-options: -DINTERNAL_DOWNLOADER
|
||||||
|
build-depends:
|
||||||
|
, HsOpenSSL >=0.11.4.18
|
||||||
|
, http-io-streams >=0.1.2.0
|
||||||
|
, io-streams >=1.5.2.1
|
||||||
|
, terminal-progress-bar >=0.4.1
|
||||||
|
|
||||||
|
if flag(tar)
|
||||||
|
cpp-options: -DTAR
|
||||||
|
build-depends: tar
|
||||||
|
|
||||||
|
else
|
||||||
|
build-depends: libarchive ^>=3.0.0.0
|
||||||
|
|
||||||
|
if os(windows)
|
||||||
|
cpp-options: -DIS_WINDOWS
|
||||||
|
other-modules: GHCup.Utils.File.Windows
|
||||||
|
build-depends:
|
||||||
|
, bzlib
|
||||||
|
, process ^>=1.6.11.0
|
||||||
|
, retry ^>=0.8.1.2
|
||||||
|
, Win32 ^>=2.10
|
||||||
|
|
||||||
|
else
|
||||||
|
other-modules: GHCup.Utils.File.Posix
|
||||||
|
build-depends:
|
||||||
|
, bz2 >=0.5.0.5 && <1.1
|
||||||
|
, hpath-posix ^>=0.13.3
|
||||||
|
, process ^>=1.6.9
|
||||||
|
, unix ^>=2.7
|
||||||
|
, unix-bytestring ^>=0.3.7.3
|
||||||
|
|
||||||
|
if (flag(tui) && !os(windows))
|
||||||
|
cpp-options: -DBRICK
|
||||||
|
build-depends: vty >=5.28.2 && <5.34
|
||||||
|
|
||||||
|
executable ghcup
|
||||||
|
main-is: Main.hs
|
||||||
|
hs-source-dirs: app/ghcup
|
||||||
|
default-language: Haskell2010
|
||||||
default-extensions:
|
default-extensions:
|
||||||
LambdaCase
|
LambdaCase
|
||||||
MultiWayIf
|
MultiWayIf
|
||||||
|
NamedFieldPuns
|
||||||
PackageImports
|
PackageImports
|
||||||
RecordWildCards
|
RecordWildCards
|
||||||
ScopedTypeVariables
|
ScopedTypeVariables
|
||||||
@@ -219,164 +199,141 @@ common config
|
|||||||
StrictData
|
StrictData
|
||||||
TupleSections
|
TupleSections
|
||||||
|
|
||||||
library
|
ghc-options:
|
||||||
import:
|
-Wall -fwarn-tabs -fwarn-incomplete-uni-patterns
|
||||||
config
|
-fwarn-incomplete-record-updates -threaded
|
||||||
, base
|
|
||||||
, HsOpenSSL
|
|
||||||
, aeson
|
|
||||||
, ascii-string
|
|
||||||
, async
|
|
||||||
, attoparsec
|
|
||||||
, binary
|
|
||||||
, bytestring
|
|
||||||
, bz2
|
|
||||||
, case-insensitive
|
|
||||||
, concurrent-output
|
|
||||||
, containers
|
|
||||||
, generics-sop
|
|
||||||
, haskus-utils-types
|
|
||||||
, haskus-utils-variant
|
|
||||||
, hopenssl
|
|
||||||
, hpath
|
|
||||||
, hpath-directory
|
|
||||||
, hpath-filepath
|
|
||||||
, hpath-io
|
|
||||||
, hpath-posix
|
|
||||||
, language-bash
|
|
||||||
, lzma
|
|
||||||
, monad-logger
|
|
||||||
, mtl
|
|
||||||
, optics
|
|
||||||
, optics-vl
|
|
||||||
, parsec
|
|
||||||
, pretty-terminal
|
|
||||||
, regex-posix
|
|
||||||
, resourcet
|
|
||||||
, safe
|
|
||||||
, safe-exceptions
|
|
||||||
, streamly
|
|
||||||
, streamly-posix
|
|
||||||
, streamly-bytestring
|
|
||||||
, strict-base
|
|
||||||
, string-interpolate
|
|
||||||
, tar-bytestring
|
|
||||||
, template-haskell
|
|
||||||
, text
|
|
||||||
, time
|
|
||||||
, transformers
|
|
||||||
, unix
|
|
||||||
, unix-bytestring
|
|
||||||
, uri-bytestring
|
|
||||||
, utf8-string
|
|
||||||
, vector
|
|
||||||
, versions
|
|
||||||
, word8
|
|
||||||
, zlib
|
|
||||||
|
|
||||||
-- deps
|
build-depends:
|
||||||
-- cabal-fmt: expand lib
|
, base >=4.13 && <5
|
||||||
exposed-modules:
|
, bytestring ^>=0.10
|
||||||
GHCup
|
, containers ^>=0.6
|
||||||
GHCup.Download
|
, filepath ^>=1.4.2.1
|
||||||
GHCup.Download.Utils
|
, ghcup
|
||||||
GHCup.Errors
|
, haskus-utils-variant >=3.0 && <3.2
|
||||||
GHCup.Platform
|
, megaparsec >=8.0.0 && <9.1
|
||||||
GHCup.Types
|
, monad-logger ^>=0.3.31
|
||||||
GHCup.Types.JSON
|
, mtl ^>=2.2
|
||||||
GHCup.Types.Optics
|
, optparse-applicative >=0.15.1.0 && <0.17
|
||||||
GHCup.Utils
|
, pretty ^>=1.1.3.1
|
||||||
GHCup.Utils.Bash
|
, pretty-terminal ^>=0.1.0.0
|
||||||
GHCup.Utils.Dirs
|
, resourcet ^>=1.2.2
|
||||||
GHCup.Utils.File
|
, safe ^>=0.3.18
|
||||||
GHCup.Utils.Logger
|
, safe-exceptions ^>=0.1
|
||||||
GHCup.Utils.Prelude
|
, string-interpolate >=0.2.0.0 && <0.4
|
||||||
GHCup.Utils.String.QQ
|
, template-haskell >=2.7 && <2.18
|
||||||
GHCup.Utils.Version.QQ
|
, text ^>=1.2.4.0
|
||||||
GHCup.Version
|
, uri-bytestring ^>=0.3.2.2
|
||||||
|
, utf8-string ^>=1.0
|
||||||
|
, versions >=4.0.1 && <5.1
|
||||||
|
|
||||||
-- other-modules:
|
if flag(internal-downloader)
|
||||||
-- other-extensions:
|
cpp-options: -DINTERNAL_DOWNLOADER
|
||||||
hs-source-dirs: lib
|
|
||||||
|
if (flag(tui) && !os(windows))
|
||||||
|
cpp-options: -DBRICK
|
||||||
|
other-modules: BrickMain
|
||||||
|
build-depends:
|
||||||
|
, brick >=0.5 && <0.62
|
||||||
|
, transformers ^>=0.5
|
||||||
|
, vector ^>=0.12
|
||||||
|
, vty >=5.28.2 && <5.34
|
||||||
|
|
||||||
|
if os(windows)
|
||||||
|
cpp-options: -DIS_WINDOWS
|
||||||
|
|
||||||
|
if flag(tar)
|
||||||
|
cpp-options: -DTAR
|
||||||
|
|
||||||
if !flag(curl)
|
|
||||||
import:
|
|
||||||
, http-io-streams
|
|
||||||
, io-streams
|
|
||||||
, terminal-progress-bar
|
|
||||||
exposed-modules: GHCup.Download.IOStreams
|
|
||||||
else
|
else
|
||||||
cpp-options: -DCURL
|
build-depends: libarchive ^>=3.0.0.0
|
||||||
|
|
||||||
executable ghcup
|
|
||||||
import:
|
|
||||||
config
|
|
||||||
, base
|
|
||||||
, bytestring
|
|
||||||
, containers
|
|
||||||
, haskus-utils-variant
|
|
||||||
, hpath
|
|
||||||
, hpath-io
|
|
||||||
, megaparsec
|
|
||||||
, monad-logger
|
|
||||||
, mtl
|
|
||||||
, optparse-applicative
|
|
||||||
, pretty-terminal
|
|
||||||
, resourcet
|
|
||||||
, string-interpolate
|
|
||||||
, table-layout
|
|
||||||
, text
|
|
||||||
, uri-bytestring
|
|
||||||
, utf8-string
|
|
||||||
, versions
|
|
||||||
|
|
||||||
--
|
|
||||||
main-is: Main.hs
|
|
||||||
|
|
||||||
-- other-modules:
|
|
||||||
-- other-extensions:
|
|
||||||
build-depends: ghcup
|
|
||||||
hs-source-dirs: app/ghcup
|
|
||||||
default-language: Haskell2010
|
|
||||||
|
|
||||||
executable ghcup-gen
|
executable ghcup-gen
|
||||||
import:
|
main-is: Main.hs
|
||||||
config
|
hs-source-dirs: app/ghcup-gen
|
||||||
, base
|
other-modules: Validate
|
||||||
, aeson
|
default-language: Haskell2010
|
||||||
, aeson-pretty
|
default-extensions:
|
||||||
, bytestring
|
DeriveGeneric
|
||||||
, containers
|
LambdaCase
|
||||||
, haskus-utils-variant
|
MultiWayIf
|
||||||
, hpath
|
NamedFieldPuns
|
||||||
, monad-logger
|
PackageImports
|
||||||
, mtl
|
QuasiQuotes
|
||||||
, optics
|
RecordWildCards
|
||||||
, optparse-applicative
|
ScopedTypeVariables
|
||||||
, pretty-terminal
|
Strict
|
||||||
, resourcet
|
StrictData
|
||||||
, safe-exceptions
|
TupleSections
|
||||||
, string-interpolate
|
TypeApplications
|
||||||
, table-layout
|
TypeFamilies
|
||||||
, text
|
ViewPatterns
|
||||||
, transformers
|
|
||||||
, uri-bytestring
|
|
||||||
, utf8-string
|
|
||||||
, versions
|
|
||||||
|
|
||||||
--
|
ghc-options:
|
||||||
main-is: Main.hs
|
-Wall -fwarn-tabs -fwarn-incomplete-uni-patterns
|
||||||
other-modules: GHCupDownloads
|
-fwarn-incomplete-record-updates -threaded
|
||||||
Validate
|
|
||||||
|
|
||||||
-- other-extensions:
|
build-depends:
|
||||||
build-depends: ghcup
|
, base >=4.13 && <5
|
||||||
hs-source-dirs: app/ghcup-gen
|
, bytestring ^>=0.10
|
||||||
default-language: Haskell2010
|
, containers ^>=0.6
|
||||||
|
, filepath ^>=1.4.2.1
|
||||||
|
, ghcup
|
||||||
|
, haskus-utils-variant >=3.0 && <3.2
|
||||||
|
, monad-logger ^>=0.3.31
|
||||||
|
, mtl ^>=2.2
|
||||||
|
, optics >=0.2 && <0.5
|
||||||
|
, optparse-applicative >=0.15.1.0 && <0.17
|
||||||
|
, pretty ^>=1.1.3.1
|
||||||
|
, pretty-terminal ^>=0.1.0.0
|
||||||
|
, regex-posix ^>=0.96
|
||||||
|
, resourcet ^>=1.2.2
|
||||||
|
, safe-exceptions ^>=0.1
|
||||||
|
, string-interpolate >=0.2.0.0 && <0.4
|
||||||
|
, text ^>=1.2.4.0
|
||||||
|
, transformers ^>=0.5
|
||||||
|
, uri-bytestring ^>=0.3.2.2
|
||||||
|
, versions >=4.0.1 && <5.1
|
||||||
|
, yaml ^>=0.11.4.0
|
||||||
|
|
||||||
|
if flag(tar)
|
||||||
|
cpp-options: -DTAR
|
||||||
|
build-depends: tar
|
||||||
|
|
||||||
|
else
|
||||||
|
build-depends: libarchive ^>=3.0.0.0
|
||||||
|
|
||||||
test-suite ghcup-test
|
test-suite ghcup-test
|
||||||
default-language: Haskell2010
|
type: exitcode-stdio-1.0
|
||||||
type: exitcode-stdio-1.0
|
main-is: Main.hs
|
||||||
hs-source-dirs: test
|
hs-source-dirs: test
|
||||||
main-is: MyLibTest.hs
|
other-modules:
|
||||||
build-depends: base >=4.12.0.0
|
GHCup.ArbitraryTypes
|
||||||
|
GHCup.Types.JSONSpec
|
||||||
|
Spec
|
||||||
|
|
||||||
|
default-language: Haskell2010
|
||||||
|
default-extensions:
|
||||||
|
LambdaCase
|
||||||
|
MultiWayIf
|
||||||
|
PackageImports
|
||||||
|
RecordWildCards
|
||||||
|
ScopedTypeVariables
|
||||||
|
TupleSections
|
||||||
|
|
||||||
|
ghc-options:
|
||||||
|
-Wall -fwarn-tabs -fwarn-incomplete-uni-patterns
|
||||||
|
-fwarn-incomplete-record-updates
|
||||||
|
|
||||||
|
build-depends:
|
||||||
|
, base >=4.13 && <5
|
||||||
|
, bytestring ^>=0.10
|
||||||
|
, containers ^>=0.6
|
||||||
|
, generic-arbitrary ^>=0.1.0
|
||||||
|
, ghcup
|
||||||
|
, hspec ^>=2.7.10
|
||||||
|
, hspec-golden-aeson ^>=0.9
|
||||||
|
, QuickCheck ^>=2.14.1
|
||||||
|
, quickcheck-arbitrary-adt ^>=0.3.1.0
|
||||||
|
, text ^>=1.2.4.0
|
||||||
|
, uri-bytestring ^>=0.3.2.2
|
||||||
|
, versions >=4.0.1 && <5.1
|
||||||
|
|||||||
16026
golden/GHCupInfo.json
Normal file
16026
golden/GHCupInfo.json
Normal file
File diff suppressed because it is too large
Load Diff
10
hie.yaml
Normal file
10
hie.yaml
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
cradle:
|
||||||
|
cabal:
|
||||||
|
- component: "ghcup:lib:ghcup"
|
||||||
|
path: ./lib
|
||||||
|
- component: "ghcup:exe:ghcup"
|
||||||
|
path: ./app/ghcup
|
||||||
|
- component: "ghcup:exe:ghcup-gen"
|
||||||
|
path: "./app/ghcup-gen"
|
||||||
|
- component: "ghcup:test:ghcup-test"
|
||||||
|
path: ./test
|
||||||
2004
lib/GHCup.hs
2004
lib/GHCup.hs
File diff suppressed because it is too large
Load Diff
@@ -9,61 +9,88 @@
|
|||||||
{-# LANGUAGE TypeFamilies #-}
|
{-# LANGUAGE TypeFamilies #-}
|
||||||
|
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Download
|
||||||
|
Description : Downloading
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
|
||||||
|
Module for handling all download related functions.
|
||||||
|
|
||||||
|
Generally we support downloading via:
|
||||||
|
|
||||||
|
- curl (default)
|
||||||
|
- wget
|
||||||
|
- internal downloader (only when compiled)
|
||||||
|
-}
|
||||||
module GHCup.Download where
|
module GHCup.Download where
|
||||||
|
|
||||||
#if !defined(CURL)
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
import GHCup.Download.IOStreams
|
import GHCup.Download.IOStreams
|
||||||
import GHCup.Download.Utils
|
import GHCup.Download.Utils
|
||||||
#endif
|
#endif
|
||||||
import GHCup.Errors
|
import GHCup.Errors
|
||||||
import GHCup.Platform
|
|
||||||
import GHCup.Types
|
import GHCup.Types
|
||||||
import GHCup.Types.JSON ( )
|
import GHCup.Types.JSON ( )
|
||||||
import GHCup.Types.Optics
|
import GHCup.Types.Optics
|
||||||
import GHCup.Utils
|
import GHCup.Utils.Dirs
|
||||||
#if defined(CURL)
|
|
||||||
import GHCup.Utils.File
|
import GHCup.Utils.File
|
||||||
#endif
|
|
||||||
import GHCup.Utils.Prelude
|
import GHCup.Utils.Prelude
|
||||||
import GHCup.Version
|
import GHCup.Version
|
||||||
|
|
||||||
import Control.Applicative
|
import Control.Applicative
|
||||||
import Control.Exception.Safe
|
import Control.Exception.Safe
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
|
#if !MIN_VERSION_base(4,13,0)
|
||||||
|
import Control.Monad.Fail ( MonadFail )
|
||||||
|
#endif
|
||||||
import Control.Monad.Logger
|
import Control.Monad.Logger
|
||||||
import Control.Monad.Reader
|
import Control.Monad.Reader
|
||||||
import Control.Monad.Trans.Resource
|
import Control.Monad.Trans.Resource
|
||||||
hiding ( throwM )
|
hiding ( throwM )
|
||||||
import Data.Aeson
|
import Data.Aeson
|
||||||
|
import Data.Bifunctor
|
||||||
import Data.ByteString ( ByteString )
|
import Data.ByteString ( ByteString )
|
||||||
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
import Data.CaseInsensitive ( CI )
|
import Data.CaseInsensitive ( CI )
|
||||||
|
#endif
|
||||||
|
import Data.List.Extra
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.String.Interpolate
|
import Data.String.Interpolate
|
||||||
import Data.Time.Clock
|
import Data.Time.Clock
|
||||||
import Data.Time.Clock.POSIX
|
import Data.Time.Clock.POSIX
|
||||||
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
import Data.Time.Format
|
import Data.Time.Format
|
||||||
|
#endif
|
||||||
import Data.Versions
|
import Data.Versions
|
||||||
|
import Data.Word8
|
||||||
import GHC.IO.Exception
|
import GHC.IO.Exception
|
||||||
import HPath
|
|
||||||
import HPath.IO as HIO
|
|
||||||
import Haskus.Utils.Variant.Excepts
|
import Haskus.Utils.Variant.Excepts
|
||||||
import OpenSSL.Digest
|
|
||||||
import Optics
|
import Optics
|
||||||
import Prelude hiding ( abs
|
import Prelude hiding ( abs
|
||||||
, readFile
|
, readFile
|
||||||
, writeFile
|
, writeFile
|
||||||
)
|
)
|
||||||
|
import System.Directory
|
||||||
|
import System.Environment
|
||||||
|
import System.FilePath
|
||||||
import System.IO.Error
|
import System.IO.Error
|
||||||
import URI.ByteString
|
import URI.ByteString
|
||||||
|
|
||||||
|
import qualified Crypto.Hash.SHA256 as SHA256
|
||||||
|
import qualified Data.ByteString as B
|
||||||
|
import qualified Data.ByteString.Base16 as B16
|
||||||
import qualified Data.ByteString.Lazy as L
|
import qualified Data.ByteString.Lazy as L
|
||||||
import qualified Data.CaseInsensitive as CI
|
|
||||||
import qualified Data.Map.Strict as M
|
import qualified Data.Map.Strict as M
|
||||||
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
|
import qualified Data.CaseInsensitive as CI
|
||||||
|
#endif
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
import qualified Data.Text.Encoding as E
|
import qualified Data.Text.Encoding as E
|
||||||
import qualified System.Posix.Files.ByteString as PF
|
import qualified Data.Yaml as Y
|
||||||
import qualified System.Posix.RawFilePath.Directory
|
|
||||||
as RD
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -75,30 +102,80 @@ import qualified System.Posix.RawFilePath.Directory
|
|||||||
------------------
|
------------------
|
||||||
|
|
||||||
|
|
||||||
-- | Downloads the download information! But only if we need to ;P
|
|
||||||
getDownloads :: ( FromJSONKey Tool
|
|
||||||
, FromJSONKey Version
|
|
||||||
, FromJSON VersionInfo
|
|
||||||
, MonadIO m
|
|
||||||
, MonadCatch m
|
|
||||||
, MonadLogger m
|
|
||||||
, MonadThrow m
|
|
||||||
, MonadFail m
|
|
||||||
)
|
|
||||||
=> URLSource
|
|
||||||
-> Excepts '[JSONError , DownloadFailed] m GHCupDownloads
|
|
||||||
getDownloads urlSource = do
|
|
||||||
lift $ $(logDebug) [i|Receiving download info from: #{urlSource}|]
|
|
||||||
case urlSource of
|
|
||||||
GHCupURL -> do
|
|
||||||
bs <- reThrowAll DownloadFailed $ smartDl ghcupURL
|
|
||||||
lE' JSONDecodeError $ eitherDecode' bs
|
|
||||||
(OwnSource url) -> do
|
|
||||||
bs <- reThrowAll DownloadFailed $ downloadBS url
|
|
||||||
lE' JSONDecodeError $ eitherDecode' bs
|
|
||||||
(OwnSpec av) -> pure $ av
|
|
||||||
|
|
||||||
where
|
-- | Downloads the download information! But only if we need to ;P
|
||||||
|
getDownloadsF :: ( FromJSONKey Tool
|
||||||
|
, FromJSONKey Version
|
||||||
|
, FromJSON VersionInfo
|
||||||
|
, MonadIO m
|
||||||
|
, MonadCatch m
|
||||||
|
, MonadLogger m
|
||||||
|
, MonadThrow m
|
||||||
|
, MonadFail m
|
||||||
|
)
|
||||||
|
=> Settings
|
||||||
|
-> Dirs
|
||||||
|
-> Excepts
|
||||||
|
'[JSONError , DownloadFailed , FileDoesNotExistError]
|
||||||
|
m
|
||||||
|
GHCupInfo
|
||||||
|
getDownloadsF settings@Settings{ urlSource } dirs = do
|
||||||
|
case urlSource of
|
||||||
|
GHCupURL -> liftE $ getBase dirs settings
|
||||||
|
(OwnSource url) -> do
|
||||||
|
bs <- reThrowAll DownloadFailed $ downloadBS (downloader settings) url
|
||||||
|
lE' JSONDecodeError $ first show $ Y.decodeEither' (L.toStrict bs)
|
||||||
|
(OwnSpec av) -> pure av
|
||||||
|
(AddSource (Left ext)) -> do
|
||||||
|
base <- liftE $ getBase dirs settings
|
||||||
|
pure (mergeGhcupInfo base ext)
|
||||||
|
(AddSource (Right uri)) -> do
|
||||||
|
base <- liftE $ getBase dirs settings
|
||||||
|
bsExt <- reThrowAll DownloadFailed $ downloadBS (downloader settings) uri
|
||||||
|
ext <- lE' JSONDecodeError $ first show $ Y.decodeEither' (L.toStrict bsExt)
|
||||||
|
pure (mergeGhcupInfo base ext)
|
||||||
|
|
||||||
|
where
|
||||||
|
|
||||||
|
mergeGhcupInfo :: GHCupInfo -- ^ base to merge with
|
||||||
|
-> GHCupInfo -- ^ extension overwriting the base
|
||||||
|
-> GHCupInfo
|
||||||
|
mergeGhcupInfo (GHCupInfo tr base base2) (GHCupInfo _ ext ext2) =
|
||||||
|
let newDownloads = M.mapWithKey (\k a -> case M.lookup k ext of
|
||||||
|
Just a' -> M.union a' a
|
||||||
|
Nothing -> a
|
||||||
|
) base
|
||||||
|
newGlobalTools = M.union base2 ext2
|
||||||
|
in GHCupInfo tr newDownloads newGlobalTools
|
||||||
|
|
||||||
|
|
||||||
|
readFromCache :: (MonadIO m, MonadCatch m, MonadLogger m)
|
||||||
|
=> Dirs
|
||||||
|
-> Excepts '[JSONError, FileDoesNotExistError] m GHCupInfo
|
||||||
|
readFromCache Dirs {..} = do
|
||||||
|
lift $ $(logWarn)
|
||||||
|
[i|Could not get download info, trying cached version (this may not be recent!)|]
|
||||||
|
let path = view pathL' ghcupURL
|
||||||
|
let yaml_file = cacheDir </> (T.unpack . decUTF8Safe . urlBaseName $ path)
|
||||||
|
bs <-
|
||||||
|
handleIO' NoSuchThing
|
||||||
|
(\_ -> throwE $ FileDoesNotExistError yaml_file)
|
||||||
|
$ liftIO
|
||||||
|
$ L.readFile yaml_file
|
||||||
|
lE' JSONDecodeError $ first show $ Y.decodeEither' (L.toStrict bs)
|
||||||
|
|
||||||
|
|
||||||
|
getBase :: (MonadFail m, MonadIO m, MonadCatch m, MonadLogger m)
|
||||||
|
=> Dirs
|
||||||
|
-> Settings
|
||||||
|
-> Excepts '[JSONError , FileDoesNotExistError] m GHCupInfo
|
||||||
|
getBase dirs@Dirs{..} Settings{ downloader } =
|
||||||
|
handleIO (\_ -> readFromCache dirs)
|
||||||
|
$ catchE @_ @'[JSONError, FileDoesNotExistError]
|
||||||
|
(\(DownloadFailed _) -> readFromCache dirs)
|
||||||
|
(reThrowAll @_ @_ @'[JSONError, DownloadFailed] DownloadFailed (smartDl ghcupURL)
|
||||||
|
>>= (liftE . lE' @_ @_ @'[JSONError] JSONDecodeError . first show . Y.decodeEither' . L.toStrict))
|
||||||
|
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.
|
||||||
@@ -109,7 +186,11 @@ getDownloads urlSource = do
|
|||||||
--
|
--
|
||||||
-- Always save the local file with the mod time of the remote file.
|
-- Always save the local file with the mod time of the remote file.
|
||||||
smartDl :: forall m1
|
smartDl :: forall m1
|
||||||
. (MonadCatch m1, MonadIO m1, MonadFail m1, MonadLogger m1)
|
. ( MonadCatch m1
|
||||||
|
, MonadIO m1
|
||||||
|
, MonadFail m1
|
||||||
|
, MonadLogger m1
|
||||||
|
)
|
||||||
=> URI
|
=> URI
|
||||||
-> Excepts
|
-> Excepts
|
||||||
'[ FileDoesNotExistError
|
'[ FileDoesNotExistError
|
||||||
@@ -124,48 +205,51 @@ getDownloads urlSource = do
|
|||||||
L.ByteString
|
L.ByteString
|
||||||
smartDl uri' = do
|
smartDl uri' = do
|
||||||
let path = view pathL' uri'
|
let path = view pathL' uri'
|
||||||
cacheDir <- liftIO $ ghcupCacheDir
|
let json_file = cacheDir </> (T.unpack . decUTF8Safe . urlBaseName $ path)
|
||||||
json_file <- (cacheDir </>) <$> urlBaseName path
|
|
||||||
e <- liftIO $ doesFileExist json_file
|
e <- liftIO $ doesFileExist json_file
|
||||||
if e
|
if e
|
||||||
then do
|
then do
|
||||||
accessTime <-
|
accessTime <- liftIO $ getAccessTime json_file
|
||||||
PF.accessTimeHiRes
|
currentTime <- liftIO getCurrentTime
|
||||||
<$> (liftIO $ PF.getFileStatus (toFilePath json_file))
|
|
||||||
currentTime <- liftIO $ getPOSIXTime
|
|
||||||
|
|
||||||
-- access time won't work on most linuxes, but we can try regardless
|
-- access time won't work on most linuxes, but we can try regardless
|
||||||
if (currentTime - accessTime) > 300
|
if (utcTimeToPOSIXSeconds currentTime - utcTimeToPOSIXSeconds accessTime) > 300
|
||||||
then do -- no access in last 5 minutes, re-check upstream mod time
|
then do -- no access in last 5 minutes, re-check upstream mod time
|
||||||
getModTime >>= \case
|
getModTime >>= \case
|
||||||
Just modTime -> do
|
Just modTime -> do
|
||||||
fileMod <- liftIO $ getModificationTime json_file
|
fileMod <- liftIO $ getModificationTime json_file
|
||||||
if modTime > fileMod
|
if modTime > fileMod
|
||||||
then do
|
then dlWithMod modTime json_file
|
||||||
bs <- liftE $ downloadBS uri'
|
else liftIO $ L.readFile json_file
|
||||||
liftIO $ writeFileWithModTime modTime json_file bs
|
|
||||||
pure bs
|
|
||||||
else liftIO $ readFile json_file
|
|
||||||
Nothing -> do
|
Nothing -> do
|
||||||
lift $ $(logDebug) [i|Unable to get/parse Last-Modified header|]
|
lift $ $(logDebug) [i|Unable to get/parse Last-Modified header|]
|
||||||
liftIO $ deleteFile json_file
|
dlWithoutMod json_file
|
||||||
liftE $ downloadBS uri'
|
|
||||||
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 $ L.readFile json_file
|
||||||
else do
|
else do
|
||||||
liftIO $ createDirIfMissing newDirPerms cacheDir
|
|
||||||
getModTime >>= \case
|
getModTime >>= \case
|
||||||
Just modTime -> do
|
Just modTime -> dlWithMod modTime json_file
|
||||||
bs <- liftE $ downloadBS uri'
|
|
||||||
liftIO $ writeFileWithModTime modTime json_file bs
|
|
||||||
pure bs
|
|
||||||
Nothing -> do
|
Nothing -> do
|
||||||
|
-- although we don't know last-modified, we still save
|
||||||
|
-- it to a file, so we might use it in offline mode
|
||||||
lift $ $(logDebug) [i|Unable to get/parse Last-Modified header|]
|
lift $ $(logDebug) [i|Unable to get/parse Last-Modified header|]
|
||||||
liftE $ downloadBS uri'
|
dlWithoutMod json_file
|
||||||
|
|
||||||
where
|
where
|
||||||
|
dlWithMod modTime json_file = do
|
||||||
|
bs <- liftE $ downloadBS downloader uri'
|
||||||
|
liftIO $ writeFileWithModTime modTime json_file bs
|
||||||
|
pure bs
|
||||||
|
dlWithoutMod json_file = do
|
||||||
|
bs <- liftE $ downloadBS downloader uri'
|
||||||
|
liftIO $ hideError doesNotExistErrorType $ rmFile json_file
|
||||||
|
liftIO $ L.writeFile json_file bs
|
||||||
|
liftIO $ setModificationTime json_file (posixSecondsToUTCTime (fromIntegral @Int 0))
|
||||||
|
pure bs
|
||||||
|
|
||||||
|
|
||||||
getModTime = do
|
getModTime = do
|
||||||
#if defined(CURL)
|
#if !defined(INTERNAL_DOWNLOADER)
|
||||||
pure Nothing
|
pure Nothing
|
||||||
#else
|
#else
|
||||||
headers <-
|
headers <-
|
||||||
@@ -178,7 +262,6 @@ getDownloads urlSource = do
|
|||||||
$ getHead uri'
|
$ getHead uri'
|
||||||
)
|
)
|
||||||
pure $ parseModifiedHeader headers
|
pure $ parseModifiedHeader headers
|
||||||
#endif
|
|
||||||
|
|
||||||
parseModifiedHeader :: (M.Map (CI ByteString) ByteString) -> Maybe UTCTime
|
parseModifiedHeader :: (M.Map (CI ByteString) ByteString) -> Maybe UTCTime
|
||||||
parseModifiedHeader headers =
|
parseModifiedHeader headers =
|
||||||
@@ -186,55 +269,30 @@ getDownloads urlSource = do
|
|||||||
True
|
True
|
||||||
defaultTimeLocale
|
defaultTimeLocale
|
||||||
"%a, %d %b %Y %H:%M:%S %Z"
|
"%a, %d %b %Y %H:%M:%S %Z"
|
||||||
(T.unpack . E.decodeUtf8 $ h)
|
(T.unpack . decUTF8Safe $ h)
|
||||||
|
|
||||||
writeFileWithModTime :: UTCTime -> Path Abs -> L.ByteString -> IO ()
|
#endif
|
||||||
|
|
||||||
|
writeFileWithModTime :: UTCTime -> FilePath -> L.ByteString -> IO ()
|
||||||
writeFileWithModTime utctime path content = do
|
writeFileWithModTime utctime path content = do
|
||||||
let mod_time = utcTimeToPOSIXSeconds utctime
|
L.writeFile path content
|
||||||
writeFileL path (Just newFilePerms) content
|
setModificationTime path utctime
|
||||||
setModificationTimeHiRes path mod_time
|
|
||||||
|
|
||||||
|
|
||||||
|
getDownloadInfo :: Tool
|
||||||
getDownloadInfo :: (MonadLogger m, MonadCatch m, MonadIO m)
|
|
||||||
=> GHCupDownloads
|
|
||||||
-> Tool
|
|
||||||
-> Version
|
-> Version
|
||||||
-> Maybe PlatformRequest
|
|
||||||
-> Excepts
|
|
||||||
'[ DistroNotFound
|
|
||||||
, NoCompatiblePlatform
|
|
||||||
, NoCompatibleArch
|
|
||||||
, NoDownload
|
|
||||||
]
|
|
||||||
m
|
|
||||||
DownloadInfo
|
|
||||||
getDownloadInfo bDls t v mpfReq = do
|
|
||||||
(PlatformRequest arch' plat ver) <- case mpfReq of
|
|
||||||
Just x -> pure x
|
|
||||||
Nothing -> do
|
|
||||||
(PlatformResult rp rv) <- liftE getPlatform
|
|
||||||
ar <- lE getArchitecture
|
|
||||||
pure $ PlatformRequest ar rp rv
|
|
||||||
|
|
||||||
lE $ getDownloadInfo' t v arch' plat ver bDls
|
|
||||||
|
|
||||||
|
|
||||||
getDownloadInfo' :: Tool
|
|
||||||
-> Version
|
|
||||||
-- ^ tool version
|
-- ^ tool version
|
||||||
-> Architecture
|
-> PlatformRequest
|
||||||
-- ^ user arch
|
-> GHCupDownloads
|
||||||
-> Platform
|
-> Either NoDownload DownloadInfo
|
||||||
-- ^ user platform
|
getDownloadInfo t v (PlatformRequest a p mv) dls = maybe
|
||||||
-> Maybe Versioning
|
|
||||||
-- ^ optional version of the platform
|
|
||||||
-> GHCupDownloads
|
|
||||||
-> Either NoDownload DownloadInfo
|
|
||||||
getDownloadInfo' t v a p mv dls = maybe
|
|
||||||
(Left NoDownload)
|
(Left NoDownload)
|
||||||
Right
|
Right
|
||||||
(with_distro <|> without_distro_ver <|> without_distro)
|
(case p of
|
||||||
|
-- non-musl won't work on alpine
|
||||||
|
Linux Alpine -> with_distro <|> without_distro_ver
|
||||||
|
_ -> with_distro <|> without_distro_ver <|> without_distro
|
||||||
|
)
|
||||||
|
|
||||||
where
|
where
|
||||||
with_distro = distro_preview id id
|
with_distro = distro_preview id id
|
||||||
@@ -242,7 +300,18 @@ getDownloadInfo' t v a p mv dls = maybe
|
|||||||
without_distro = distro_preview (set _Linux UnknownLinux) (const Nothing)
|
without_distro = distro_preview (set _Linux UnknownLinux) (const Nothing)
|
||||||
|
|
||||||
distro_preview f g =
|
distro_preview f g =
|
||||||
preview (ix t % ix v % viArch % ix a % ix (f p) % ix (g mv)) dls
|
let platformVersionSpec =
|
||||||
|
preview (ix t % ix v % viArch % ix a % ix (f p)) dls
|
||||||
|
mv' = g mv
|
||||||
|
in fmap snd
|
||||||
|
. find
|
||||||
|
(\(mverRange, _) -> maybe
|
||||||
|
(isNothing mv')
|
||||||
|
(\range -> maybe False (`versionRange` range) mv')
|
||||||
|
mverRange
|
||||||
|
)
|
||||||
|
. M.toList
|
||||||
|
=<< platformVersionSpec
|
||||||
|
|
||||||
|
|
||||||
-- | Tries to download from the given http or https url
|
-- | Tries to download from the given http or https url
|
||||||
@@ -253,16 +322,16 @@ getDownloadInfo' t v a p mv dls = maybe
|
|||||||
--
|
--
|
||||||
-- The file must not exist.
|
-- The file must not exist.
|
||||||
download :: ( MonadMask m
|
download :: ( MonadMask m
|
||||||
, MonadReader Settings m
|
|
||||||
, MonadThrow m
|
, MonadThrow m
|
||||||
, MonadLogger m
|
, MonadLogger m
|
||||||
, MonadIO m
|
, MonadIO m
|
||||||
)
|
)
|
||||||
=> DownloadInfo
|
=> Settings
|
||||||
-> Path Abs -- ^ destination dir
|
-> DownloadInfo
|
||||||
-> Maybe (Path Rel) -- ^ optional filename
|
-> FilePath -- ^ destination dir
|
||||||
-> Excepts '[DigestError , DownloadFailed] m (Path Abs)
|
-> Maybe FilePath -- ^ optional filename
|
||||||
download dli dest mfn
|
-> Excepts '[DigestError , DownloadFailed] m FilePath
|
||||||
|
download settings@Settings{ downloader } dli dest mfn
|
||||||
| scheme == "https" = dl
|
| scheme == "https" = dl
|
||||||
| scheme == "http" = dl
|
| scheme == "http" = dl
|
||||||
| scheme == "file" = cp
|
| scheme == "file" = cp
|
||||||
@@ -272,41 +341,48 @@ 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
|
let destFile = getDestFile
|
||||||
fromFile <- parseAbs path
|
let fromFile = T.unpack . decUTF8Safe $ path
|
||||||
liftIO $ copyFile fromFile destFile Strict
|
liftIO $ copyFile fromFile destFile
|
||||||
pure destFile
|
pure destFile
|
||||||
dl = do
|
dl = do
|
||||||
let uri' = E.decodeUtf8 (serializeURIRef' (view dlUri dli))
|
let uri' = decUTF8Safe (serializeURIRef' (view dlUri dli))
|
||||||
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
|
let destFile = getDestFile
|
||||||
|
|
||||||
-- download
|
-- download
|
||||||
flip onException
|
flip onException
|
||||||
(liftIO $ hideError doesNotExistErrorType $ deleteFile destFile)
|
(liftIO $ hideError doesNotExistErrorType $ rmFile destFile)
|
||||||
$ catchAllE @_ @'[ProcessError, DownloadFailed, UnsupportedScheme]
|
$ catchAllE @_ @'[ProcessError, DownloadFailed, UnsupportedScheme]
|
||||||
(\e ->
|
(\e ->
|
||||||
(liftIO $ hideError doesNotExistErrorType $ deleteFile destFile)
|
liftIO (hideError doesNotExistErrorType $ rmFile destFile)
|
||||||
>> (throwE . DownloadFailed $ e)
|
>> (throwE . DownloadFailed $ e)
|
||||||
) $ do
|
) $ do
|
||||||
#if defined(CURL)
|
case downloader of
|
||||||
liftE $ lEM @_ @'[ProcessError] $ liftIO $ exec "curl" True
|
Curl -> do
|
||||||
["-sSfL", "-o", toFilePath destFile , serializeURIRef' $ view dlUri dli] Nothing Nothing
|
o' <- liftIO getCurlOpts
|
||||||
#else
|
liftE $ lEM @_ @'[ProcessError] $ exec "curl"
|
||||||
(https, host, fullPath, port) <- liftE $ uriToQuadruple (view dlUri dli)
|
(o' ++ ["-fL", "-o", destFile, (T.unpack . decUTF8Safe) $ serializeURIRef' $ view dlUri dli]) Nothing Nothing
|
||||||
liftE $ downloadToFile https host fullPath port destFile
|
Wget -> do
|
||||||
|
o' <- liftIO getWgetOpts
|
||||||
|
liftE $ lEM @_ @'[ProcessError] $ exec "wget"
|
||||||
|
(o' ++ ["-O", destFile , (T.unpack . decUTF8Safe) $ serializeURIRef' $ view dlUri dli]) Nothing Nothing
|
||||||
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
|
Internal -> do
|
||||||
|
(https, host, fullPath, port) <- liftE $ uriToQuadruple (view dlUri dli)
|
||||||
|
liftE $ downloadToFile https host fullPath port destFile
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
liftE $ checkDigest dli destFile
|
liftE $ checkDigest settings dli destFile
|
||||||
pure destFile
|
pure destFile
|
||||||
|
|
||||||
-- Manage to find a file we can write the body into.
|
-- Manage to find a file we can write the body into.
|
||||||
getDestFile :: MonadThrow m => m (Path Abs)
|
getDestFile :: FilePath
|
||||||
getDestFile = maybe (urlBaseName path <&> (dest </>)) (pure . (dest </>)) mfn
|
getDestFile = maybe (dest </> T.unpack (decUTF8Safe (urlBaseName path))) (dest </>) mfn
|
||||||
|
|
||||||
path = view (dlUri % pathL') dli
|
path = view (dlUri % pathL') dli
|
||||||
|
|
||||||
@@ -318,27 +394,41 @@ downloadCached :: ( MonadMask m
|
|||||||
, MonadThrow m
|
, MonadThrow m
|
||||||
, MonadLogger m
|
, MonadLogger m
|
||||||
, MonadIO m
|
, MonadIO m
|
||||||
, MonadReader Settings m
|
, MonadUnliftIO m
|
||||||
)
|
)
|
||||||
=> DownloadInfo
|
=> Settings
|
||||||
-> Maybe (Path Rel) -- ^ optional filename
|
-> Dirs
|
||||||
-> Excepts '[DigestError , DownloadFailed] m (Path Abs)
|
-> DownloadInfo
|
||||||
downloadCached dli mfn = do
|
-> Maybe FilePath -- ^ optional filename
|
||||||
cache <- lift getCache
|
-> Excepts '[DigestError , DownloadFailed] m FilePath
|
||||||
|
downloadCached settings@Settings{ cache } dirs dli mfn = do
|
||||||
case cache of
|
case cache of
|
||||||
True -> do
|
True -> downloadCached' settings dirs dli mfn
|
||||||
cachedir <- liftIO $ ghcupCacheDir
|
|
||||||
fn <- maybe (urlBaseName $ view (dlUri % pathL') dli) pure mfn
|
|
||||||
let cachfile = cachedir </> fn
|
|
||||||
fileExists <- liftIO $ doesFileExist cachfile
|
|
||||||
if
|
|
||||||
| fileExists -> do
|
|
||||||
liftE $ checkDigest dli cachfile
|
|
||||||
pure $ cachfile
|
|
||||||
| otherwise -> liftE $ download dli cachedir mfn
|
|
||||||
False -> do
|
False -> do
|
||||||
tmp <- lift withGHCupTmpDir
|
tmp <- lift withGHCupTmpDir
|
||||||
liftE $ download dli tmp mfn
|
liftE $ download settings dli tmp mfn
|
||||||
|
|
||||||
|
|
||||||
|
downloadCached' :: ( MonadMask m
|
||||||
|
, MonadThrow m
|
||||||
|
, MonadLogger m
|
||||||
|
, MonadIO m
|
||||||
|
, MonadUnliftIO m
|
||||||
|
)
|
||||||
|
=> Settings
|
||||||
|
-> Dirs
|
||||||
|
-> DownloadInfo
|
||||||
|
-> Maybe FilePath -- ^ optional filename
|
||||||
|
-> Excepts '[DigestError , DownloadFailed] m FilePath
|
||||||
|
downloadCached' settings Dirs{..} dli mfn = do
|
||||||
|
let fn = fromMaybe ((T.unpack . decUTF8Safe) $ urlBaseName $ view (dlUri % pathL') dli) mfn
|
||||||
|
let cachfile = cacheDir </> fn
|
||||||
|
fileExists <- liftIO $ doesFileExist cachfile
|
||||||
|
if
|
||||||
|
| fileExists -> do
|
||||||
|
liftE $ checkDigest settings dli cachfile
|
||||||
|
pure cachfile
|
||||||
|
| otherwise -> liftE $ download settings dli cacheDir mfn
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -351,8 +441,9 @@ downloadCached dli mfn = do
|
|||||||
|
|
||||||
|
|
||||||
-- | This is used for downloading the JSON.
|
-- | This is used for downloading the JSON.
|
||||||
downloadBS :: (MonadCatch m, MonadIO m)
|
downloadBS :: (MonadCatch m, MonadIO m, MonadLogger m)
|
||||||
=> URI
|
=> Downloader
|
||||||
|
-> URI
|
||||||
-> Excepts
|
-> Excepts
|
||||||
'[ FileDoesNotExistError
|
'[ FileDoesNotExistError
|
||||||
, HTTPStatusError
|
, HTTPStatusError
|
||||||
@@ -364,45 +455,83 @@ downloadBS :: (MonadCatch m, MonadIO m)
|
|||||||
]
|
]
|
||||||
m
|
m
|
||||||
L.ByteString
|
L.ByteString
|
||||||
downloadBS uri'
|
downloadBS downloader uri'
|
||||||
| scheme == "https"
|
| scheme == "https"
|
||||||
= dl True
|
= dl True
|
||||||
| scheme == "http"
|
| scheme == "http"
|
||||||
= dl False
|
= dl False
|
||||||
| scheme == "file"
|
| scheme == "file"
|
||||||
= liftIOException doesNotExistErrorType (FileDoesNotExistError path)
|
= liftIOException doesNotExistErrorType (FileDoesNotExistError $ T.unpack $ decUTF8Safe path)
|
||||||
$ (liftIO $ RD.readFile path)
|
(liftIO $ L.readFile (T.unpack $ decUTF8Safe path))
|
||||||
| otherwise
|
| otherwise
|
||||||
= throwE UnsupportedScheme
|
= throwE UnsupportedScheme
|
||||||
|
|
||||||
where
|
where
|
||||||
scheme = view (uriSchemeL' % schemeBSL') uri'
|
scheme = view (uriSchemeL' % schemeBSL') uri'
|
||||||
path = view pathL' uri'
|
path = view pathL' uri'
|
||||||
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
dl https = do
|
dl https = do
|
||||||
#if defined(CURL)
|
|
||||||
let exe = [rel|curl|]
|
|
||||||
args = ["-sSfL", serializeURIRef' uri']
|
|
||||||
liftIO (executeOut exe args Nothing) >>= \case
|
|
||||||
CapturedProcess ExitSuccess stdout _ -> do
|
|
||||||
pure $ L.fromStrict stdout
|
|
||||||
CapturedProcess (ExitFailure i') _ _ -> throwE $ NonZeroExit i' (toFilePath exe) args
|
|
||||||
#else
|
#else
|
||||||
(_, host', fullPath', port') <- liftE $ uriToQuadruple uri'
|
dl _ = do
|
||||||
liftE $ downloadBS' https host' fullPath' port'
|
#endif
|
||||||
|
lift $ $(logDebug) [i|downloading: #{serializeURIRef' uri'}|]
|
||||||
|
case downloader of
|
||||||
|
Curl -> do
|
||||||
|
o' <- liftIO getCurlOpts
|
||||||
|
let exe = "curl"
|
||||||
|
args = o' ++ ["-sSfL", T.unpack $ decUTF8Safe $ serializeURIRef' uri']
|
||||||
|
lift (executeOut exe args Nothing) >>= \case
|
||||||
|
CapturedProcess ExitSuccess stdout _ -> do
|
||||||
|
pure stdout
|
||||||
|
CapturedProcess (ExitFailure i') _ _ -> throwE $ NonZeroExit i' exe args
|
||||||
|
Wget -> do
|
||||||
|
o' <- liftIO getWgetOpts
|
||||||
|
let exe = "wget"
|
||||||
|
args = o' ++ ["-qO-", T.unpack $ decUTF8Safe $ serializeURIRef' uri']
|
||||||
|
lift (executeOut exe args Nothing) >>= \case
|
||||||
|
CapturedProcess ExitSuccess stdout _ -> do
|
||||||
|
pure stdout
|
||||||
|
CapturedProcess (ExitFailure i') _ _ -> throwE $ NonZeroExit i' exe args
|
||||||
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
|
Internal -> do
|
||||||
|
(_, host', fullPath', port') <- liftE $ uriToQuadruple uri'
|
||||||
|
liftE $ downloadBS' https host' fullPath' port'
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
checkDigest :: (MonadIO m, MonadLogger m, MonadReader Settings m)
|
checkDigest :: (MonadIO m, MonadThrow m, MonadLogger m)
|
||||||
=> DownloadInfo
|
=> Settings
|
||||||
-> Path Abs
|
-> DownloadInfo
|
||||||
|
-> FilePath
|
||||||
-> Excepts '[DigestError] m ()
|
-> Excepts '[DigestError] m ()
|
||||||
checkDigest dli file = do
|
checkDigest Settings{ noVerify } dli file = do
|
||||||
verify <- lift ask <&> (not . noVerify)
|
let verify = not noVerify
|
||||||
when verify $ do
|
when verify $ do
|
||||||
let p' = toFilePath file
|
let p' = takeFileName file
|
||||||
lift $ $(logInfo) [i|verifying digest of: #{p'}|]
|
lift $ $(logInfo) [i|verifying digest of: #{p'}|]
|
||||||
c <- liftIO $ readFile file
|
c <- liftIO $ L.readFile file
|
||||||
let cDigest = E.decodeUtf8 . toHex . digest (digestByName "sha256") $ c
|
cDigest <- throwEither . E.decodeUtf8' . B16.encode . SHA256.hashlazy $ c
|
||||||
eDigest = view dlHash dli
|
let eDigest = view dlHash dli
|
||||||
when ((cDigest /= eDigest) && verify) $ throwE (DigestError cDigest eDigest)
|
when ((cDigest /= eDigest) && verify) $ throwE (DigestError cDigest eDigest)
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get additional curl args from env. This is an undocumented option.
|
||||||
|
getCurlOpts :: IO [String]
|
||||||
|
getCurlOpts =
|
||||||
|
lookupEnv "GHCUP_CURL_OPTS" >>= \case
|
||||||
|
Just r -> pure $ splitOn " " r
|
||||||
|
Nothing -> pure []
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get additional wget args from env. This is an undocumented option.
|
||||||
|
getWgetOpts :: IO [String]
|
||||||
|
getWgetOpts =
|
||||||
|
lookupEnv "GHCUP_WGET_OPTS" >>= \case
|
||||||
|
Just r -> pure $ splitOn " " r
|
||||||
|
Nothing -> pure []
|
||||||
|
|
||||||
|
|
||||||
|
urlBaseName :: ByteString -- ^ the url path (without scheme and host)
|
||||||
|
-> ByteString
|
||||||
|
urlBaseName = snd . B.breakEnd (== _slash) . urlDecode False
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,6 @@
|
|||||||
{-# LANGUAGE DataKinds #-}
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE DeriveGeneric #-}
|
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
|
||||||
{-# LANGUAGE TemplateHaskell #-}
|
|
||||||
{-# LANGUAGE TypeApplications #-}
|
|
||||||
{-# LANGUAGE TypeFamilies #-}
|
{-# LANGUAGE TypeFamilies #-}
|
||||||
|
|
||||||
|
|
||||||
@@ -13,6 +9,7 @@ module GHCup.Download.IOStreams where
|
|||||||
|
|
||||||
import GHCup.Download.Utils
|
import GHCup.Download.Utils
|
||||||
import GHCup.Errors
|
import GHCup.Errors
|
||||||
|
import GHCup.Types.Optics
|
||||||
import GHCup.Types.JSON ( )
|
import GHCup.Types.JSON ( )
|
||||||
import GHCup.Utils.File
|
import GHCup.Utils.File
|
||||||
import GHCup.Utils.Prelude
|
import GHCup.Utils.Prelude
|
||||||
@@ -27,8 +24,6 @@ import Data.CaseInsensitive ( CI )
|
|||||||
import Data.IORef
|
import Data.IORef
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.Text.Read
|
import Data.Text.Read
|
||||||
import HPath
|
|
||||||
import HPath.IO as HIO
|
|
||||||
import Haskus.Utils.Variant.Excepts
|
import Haskus.Utils.Variant.Excepts
|
||||||
import Network.Http.Client hiding ( URL )
|
import Network.Http.Client hiding ( URL )
|
||||||
import Optics
|
import Optics
|
||||||
@@ -36,17 +31,13 @@ import Prelude hiding ( abs
|
|||||||
, readFile
|
, readFile
|
||||||
, writeFile
|
, writeFile
|
||||||
)
|
)
|
||||||
import "unix" System.Posix.IO.ByteString
|
|
||||||
hiding ( fdWrite )
|
|
||||||
import "unix-bytestring" System.Posix.IO.ByteString
|
|
||||||
( fdWrite )
|
|
||||||
import System.ProgressBar
|
import System.ProgressBar
|
||||||
|
import System.IO
|
||||||
import URI.ByteString
|
import URI.ByteString
|
||||||
|
|
||||||
import qualified Data.ByteString as BS
|
import qualified Data.ByteString as BS
|
||||||
import qualified Data.ByteString.Lazy as L
|
import qualified Data.ByteString.Lazy as L
|
||||||
import qualified Data.Map.Strict as M
|
import qualified Data.Map.Strict as M
|
||||||
import qualified Data.Text.Encoding as E
|
|
||||||
import qualified System.IO.Streams as Streams
|
import qualified System.IO.Streams as Streams
|
||||||
|
|
||||||
|
|
||||||
@@ -72,7 +63,7 @@ downloadBS' :: MonadIO m
|
|||||||
, TooManyRedirs
|
, TooManyRedirs
|
||||||
]
|
]
|
||||||
m
|
m
|
||||||
(L.ByteString)
|
L.ByteString
|
||||||
downloadBS' https host path port = do
|
downloadBS' https host path port = do
|
||||||
bref <- liftIO $ newIORef (mempty :: Builder)
|
bref <- liftIO $ newIORef (mempty :: Builder)
|
||||||
let stepper bs = modifyIORef bref (<> byteString bs)
|
let stepper bs = modifyIORef bref (<> byteString bs)
|
||||||
@@ -85,12 +76,12 @@ downloadToFile :: (MonadMask m, MonadIO m)
|
|||||||
-> ByteString -- ^ host (e.g. "www.example.com")
|
-> ByteString -- ^ host (e.g. "www.example.com")
|
||||||
-> ByteString -- ^ path (e.g. "/my/file") including query
|
-> ByteString -- ^ path (e.g. "/my/file") including query
|
||||||
-> Maybe Int -- ^ optional port (e.g. 3000)
|
-> Maybe Int -- ^ optional port (e.g. 3000)
|
||||||
-> Path Abs -- ^ destination file to create and write to
|
-> FilePath -- ^ destination file to create and write to
|
||||||
-> Excepts '[DownloadFailed] m ()
|
-> Excepts '[DownloadFailed] m ()
|
||||||
downloadToFile https host fullPath port destFile = do
|
downloadToFile https host fullPath port destFile = do
|
||||||
fd <- liftIO $ createRegularFileFd newFilePerms destFile
|
fd <- liftIO $ openFile destFile WriteMode
|
||||||
let stepper = fdWrite fd
|
let stepper = BS.hPut fd
|
||||||
flip finally (liftIO $ closeFd fd)
|
flip finally (liftIO $ hClose fd)
|
||||||
$ reThrowAll DownloadFailed $ downloadInternal True https host fullPath port stepper
|
$ reThrowAll DownloadFailed $ downloadInternal True https host fullPath port stepper
|
||||||
|
|
||||||
|
|
||||||
@@ -132,7 +123,7 @@ downloadInternal = go (5 :: Int)
|
|||||||
if
|
if
|
||||||
| scode >= 200 && scode < 300 -> downloadStream r i' >> pure Nothing
|
| scode >= 200 && scode < 300 -> downloadStream r i' >> pure Nothing
|
||||||
| scode >= 300 && scode < 400 -> case getHeader r "Location" of
|
| scode >= 300 && scode < 400 -> case getHeader r "Location" of
|
||||||
Just r' -> pure $ Just $ r'
|
Just r' -> pure $ Just r'
|
||||||
Nothing -> throwE NoLocationHeader
|
Nothing -> throwE NoLocationHeader
|
||||||
| otherwise -> throwE $ HTTPStatusError scode
|
| otherwise -> throwE $ HTTPStatusError scode
|
||||||
)
|
)
|
||||||
@@ -145,13 +136,13 @@ downloadInternal = go (5 :: Int)
|
|||||||
|
|
||||||
downloadStream r i' = do
|
downloadStream r i' = do
|
||||||
let size = case getHeader r "Content-Length" of
|
let size = case getHeader r "Content-Length" of
|
||||||
Just x' -> case decimal $ E.decodeUtf8 x' of
|
Just x' -> case decimal $ decUTF8Safe x' of
|
||||||
Left _ -> 0
|
Left _ -> 0
|
||||||
Right (r', _) -> r'
|
Right (r', _) -> r'
|
||||||
Nothing -> 0
|
Nothing -> 0
|
||||||
|
|
||||||
mpb <- if progressBar
|
mpb <- if progressBar
|
||||||
then Just <$> (liftIO $ newProgressBar defStyle 10 (Progress 0 size ()))
|
then Just <$> liftIO (newProgressBar defStyle 10 (Progress 0 size ()))
|
||||||
else pure Nothing
|
else pure Nothing
|
||||||
|
|
||||||
outStream <- liftIO $ Streams.makeOutputStream
|
outStream <- liftIO $ Streams.makeOutputStream
|
||||||
@@ -224,9 +215,9 @@ headInternal = go (5 :: Int)
|
|||||||
if
|
if
|
||||||
| scode >= 200 && scode < 300 -> do
|
| scode >= 200 && scode < 300 -> do
|
||||||
let headers = getHeaderMap r
|
let headers = getHeaderMap r
|
||||||
pure $ Right $ headers
|
pure $ Right headers
|
||||||
| scode >= 300 && scode < 400 -> case getHeader r "Location" of
|
| scode >= 300 && scode < 400 -> case getHeader r "Location" of
|
||||||
Just r' -> pure $ Left $ r'
|
Just r' -> pure $ Left r'
|
||||||
Nothing -> throwE NoLocationHeader
|
Nothing -> throwE NoLocationHeader
|
||||||
| otherwise -> throwE $ HTTPStatusError scode
|
| otherwise -> throwE $ HTTPStatusError scode
|
||||||
)
|
)
|
||||||
@@ -243,7 +234,7 @@ withConnection' :: Bool
|
|||||||
-> Maybe Int
|
-> Maybe Int
|
||||||
-> (Connection -> IO a)
|
-> (Connection -> IO a)
|
||||||
-> IO a
|
-> IO a
|
||||||
withConnection' https host port action = bracket acquire closeConnection action
|
withConnection' https host port = bracket acquire closeConnection
|
||||||
|
|
||||||
where
|
where
|
||||||
acquire = case https of
|
acquire = case https of
|
||||||
|
|||||||
@@ -1,10 +1,6 @@
|
|||||||
{-# LANGUAGE DataKinds #-}
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE DeriveGeneric #-}
|
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
|
||||||
{-# LANGUAGE TemplateHaskell #-}
|
|
||||||
{-# LANGUAGE TypeApplications #-}
|
|
||||||
{-# LANGUAGE TypeFamilies #-}
|
{-# LANGUAGE TypeFamilies #-}
|
||||||
|
|
||||||
|
|
||||||
@@ -55,7 +51,7 @@ uriToQuadruple URI {..} = do
|
|||||||
let queryBS =
|
let queryBS =
|
||||||
BS.intercalate "&"
|
BS.intercalate "&"
|
||||||
. fmap (\(x, y) -> encodeQuery x <> "=" <> encodeQuery y)
|
. fmap (\(x, y) -> encodeQuery x <> "=" <> encodeQuery y)
|
||||||
$ (queryPairs uriQuery)
|
$ queryPairs uriQuery
|
||||||
port =
|
port =
|
||||||
preview (_Just % authorityPortL' % _Just % portNumberL') uriAuthority
|
preview (_Just % authorityPortL' % _Just % portNumberL') uriAuthority
|
||||||
fullpath = if BS.null queryBS then uriPath else uriPath <> "?" <> queryBS
|
fullpath = if BS.null queryBS then uriPath else uriPath <> "?" <> queryBS
|
||||||
|
|||||||
@@ -1,18 +1,39 @@
|
|||||||
|
{-# OPTIONS_GHC -Wno-orphans #-}
|
||||||
|
{-# LANGUAGE CPP #-}
|
||||||
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE ExistentialQuantification #-}
|
{-# LANGUAGE ExistentialQuantification #-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE StandaloneDeriving #-}
|
{-# LANGUAGE StandaloneDeriving #-}
|
||||||
{-# LANGUAGE DataKinds #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
{-# LANGUAGE TypeOperators #-}
|
||||||
|
{-# LANGUAGE FlexibleInstances #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Errors
|
||||||
|
Description : GHCup error types
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
module GHCup.Errors where
|
module GHCup.Errors where
|
||||||
|
|
||||||
import GHCup.Types
|
import GHCup.Types
|
||||||
|
|
||||||
|
#if !defined(TAR)
|
||||||
|
import Codec.Archive
|
||||||
|
#else
|
||||||
|
import qualified Codec.Archive.Tar as Tar
|
||||||
|
#endif
|
||||||
import Control.Exception.Safe
|
import Control.Exception.Safe
|
||||||
import Data.ByteString ( ByteString )
|
import Data.String.Interpolate
|
||||||
import Data.Text ( Text )
|
import Data.Text ( Text )
|
||||||
import Data.Versions
|
import Data.Versions
|
||||||
import Haskus.Utils.Variant
|
import Haskus.Utils.Variant
|
||||||
import HPath
|
import Text.PrettyPrint
|
||||||
|
import Text.PrettyPrint.HughesPJClass
|
||||||
|
import URI.ByteString
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -26,72 +47,191 @@ import HPath
|
|||||||
data NoCompatiblePlatform = NoCompatiblePlatform String -- the platform we got
|
data NoCompatiblePlatform = NoCompatiblePlatform String -- the platform we got
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NoCompatiblePlatform where
|
||||||
|
pPrint (NoCompatiblePlatform str') =
|
||||||
|
text ("Could not find a compatible platform. Got: " ++ str')
|
||||||
|
|
||||||
-- | Unable to find a download for the requested versio/distro.
|
-- | Unable to find a download for the requested versio/distro.
|
||||||
data NoDownload = NoDownload
|
data NoDownload = NoDownload
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NoDownload where
|
||||||
|
pPrint NoDownload =
|
||||||
|
text "Unable to find a download for the requested version/distro."
|
||||||
|
|
||||||
|
-- | No update available or necessary.
|
||||||
|
data NoUpdate = NoUpdate
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NoUpdate where
|
||||||
|
pPrint NoUpdate = text "No update available or necessary."
|
||||||
|
|
||||||
-- | The Architecture is unknown and unsupported.
|
-- | The Architecture is unknown and unsupported.
|
||||||
data NoCompatibleArch = NoCompatibleArch String
|
data NoCompatibleArch = NoCompatibleArch String
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NoCompatibleArch where
|
||||||
|
pPrint (NoCompatibleArch arch) =
|
||||||
|
text ("The Architecture is unknown or unsupported. Got: " ++ arch)
|
||||||
|
|
||||||
-- | Unable to figure out the distribution of the host.
|
-- | Unable to figure out the distribution of the host.
|
||||||
data DistroNotFound = DistroNotFound
|
data DistroNotFound = DistroNotFound
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty DistroNotFound where
|
||||||
|
pPrint DistroNotFound =
|
||||||
|
text "Unable to figure out the distribution of the host."
|
||||||
|
|
||||||
-- | The archive format is unknown. We don't know how to extract it.
|
-- | The archive format is unknown. We don't know how to extract it.
|
||||||
data UnknownArchive = UnknownArchive ByteString
|
data UnknownArchive = UnknownArchive FilePath
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty UnknownArchive where
|
||||||
|
pPrint (UnknownArchive file) =
|
||||||
|
text [i|The archive format is unknown. We don't know how to extract the file "#{file}"|]
|
||||||
|
|
||||||
-- | The scheme is not supported (such as ftp).
|
-- | The scheme is not supported (such as ftp).
|
||||||
data UnsupportedScheme = UnsupportedScheme
|
data UnsupportedScheme = UnsupportedScheme
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty UnsupportedScheme where
|
||||||
|
pPrint UnsupportedScheme = text "The scheme is not supported (such as ftp)."
|
||||||
|
|
||||||
-- | Unable to copy a file.
|
-- | Unable to copy a file.
|
||||||
data CopyError = CopyError String
|
data CopyError = CopyError String
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty CopyError where
|
||||||
|
pPrint (CopyError reason) =
|
||||||
|
text ("Unable to copy a file. Reason was: " ++ reason)
|
||||||
|
|
||||||
-- | Unable to find a tag of a tool.
|
-- | Unable to find a tag of a tool.
|
||||||
data TagNotFound = TagNotFound Tag Tool
|
data TagNotFound = TagNotFound Tag Tool
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty TagNotFound where
|
||||||
|
pPrint (TagNotFound tag tool) =
|
||||||
|
text "Unable to find tag" <+> pPrint tag <+> text [i|of tool "#{tool}"|]
|
||||||
|
|
||||||
|
-- | Unable to find the next version of a tool (the one after the currently
|
||||||
|
-- set one).
|
||||||
|
data NextVerNotFound = NextVerNotFound Tool
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NextVerNotFound where
|
||||||
|
pPrint (NextVerNotFound tool) =
|
||||||
|
text [i|Unable to find next (the one after the currently set one) version of tool "#{tool}"|]
|
||||||
|
|
||||||
-- | The tool (such as GHC) is already installed with that version.
|
-- | The tool (such as GHC) is already installed with that version.
|
||||||
data AlreadyInstalled = AlreadyInstalled Tool Version
|
data AlreadyInstalled = AlreadyInstalled Tool Version
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty AlreadyInstalled where
|
||||||
|
pPrint (AlreadyInstalled tool ver') =
|
||||||
|
text [i|#{tool}-#{prettyShow ver'} is already installed|]
|
||||||
|
|
||||||
-- | The tool is not installed. Some operations rely on a tool
|
-- | The tool is not installed. Some operations rely on a tool
|
||||||
-- to be installed (such as setting the current GHC version).
|
-- to be installed (such as setting the current GHC version).
|
||||||
data NotInstalled = NotInstalled Tool Version
|
data NotInstalled = NotInstalled Tool GHCTargetVersion
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NotInstalled where
|
||||||
|
pPrint (NotInstalled tool ver) =
|
||||||
|
text [i|The version "#{prettyShow ver}" of the tool "#{tool}" is not installed.|]
|
||||||
|
|
||||||
|
-- | An executable was expected to be in PATH, but was not found.
|
||||||
|
data NotFoundInPATH = NotFoundInPATH FilePath
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NotFoundInPATH where
|
||||||
|
pPrint (NotFoundInPATH exe) =
|
||||||
|
text [i|The exe "#{exe}" was not found in PATH.|]
|
||||||
|
|
||||||
-- | JSON decoding failed.
|
-- | JSON decoding failed.
|
||||||
data JSONError = JSONDecodeError String
|
data JSONError = JSONDecodeError String
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty JSONError where
|
||||||
|
pPrint (JSONDecodeError err) =
|
||||||
|
text [i|JSON decoding failed with: #{err}|]
|
||||||
|
|
||||||
-- | A file that is supposed to exist does not exist
|
-- | A file that is supposed to exist does not exist
|
||||||
-- (e.g. when we use file scheme to "download" something).
|
-- (e.g. when we use file scheme to "download" something).
|
||||||
data FileDoesNotExistError = FileDoesNotExistError ByteString
|
data FileDoesNotExistError = FileDoesNotExistError FilePath
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty FileDoesNotExistError where
|
||||||
|
pPrint (FileDoesNotExistError file) =
|
||||||
|
text [i|File "#{file}" does not exist.|]
|
||||||
|
|
||||||
|
data TarDirDoesNotExist = TarDirDoesNotExist TarDir
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty TarDirDoesNotExist where
|
||||||
|
pPrint (TarDirDoesNotExist dir) =
|
||||||
|
text "Tar directory does not exist:" <+> pPrint dir
|
||||||
|
|
||||||
-- | File digest verification failed.
|
-- | File digest verification failed.
|
||||||
data DigestError = DigestError Text Text
|
data DigestError = DigestError Text Text
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty DigestError where
|
||||||
|
pPrint (DigestError currentDigest expectedDigest) =
|
||||||
|
text [i|Digest error: expected "#{expectedDigest}", but got "#{currentDigest}"|]
|
||||||
|
|
||||||
-- | Unexpected HTTP status.
|
-- | Unexpected HTTP status.
|
||||||
data HTTPStatusError = HTTPStatusError Int
|
data HTTPStatusError = HTTPStatusError Int
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty HTTPStatusError where
|
||||||
|
pPrint (HTTPStatusError status) =
|
||||||
|
text [i|Unexpected HTTP status: #{status}|]
|
||||||
|
|
||||||
-- | The 'Location' header was expected during a 3xx redirect, but not found.
|
-- | The 'Location' header was expected during a 3xx redirect, but not found.
|
||||||
data NoLocationHeader = NoLocationHeader
|
data NoLocationHeader = NoLocationHeader
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NoLocationHeader where
|
||||||
|
pPrint NoLocationHeader =
|
||||||
|
text [i|The 'Location' header was expected during a 3xx redirect, but not found.|]
|
||||||
|
|
||||||
-- | Too many redirects.
|
-- | Too many redirects.
|
||||||
data TooManyRedirs = TooManyRedirs
|
data TooManyRedirs = TooManyRedirs
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty TooManyRedirs where
|
||||||
|
pPrint TooManyRedirs =
|
||||||
|
text [i|Too many redirections.|]
|
||||||
|
|
||||||
-- | A patch could not be applied.
|
-- | A patch could not be applied.
|
||||||
data PatchFailed = PatchFailed
|
data PatchFailed = PatchFailed
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty PatchFailed where
|
||||||
|
pPrint PatchFailed =
|
||||||
|
text [i|A patch could not be applied.|]
|
||||||
|
|
||||||
|
-- | The tool requirements could not be found.
|
||||||
|
data NoToolRequirements = NoToolRequirements
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NoToolRequirements where
|
||||||
|
pPrint NoToolRequirements =
|
||||||
|
text [i|The Tool requirements could not be found.|]
|
||||||
|
|
||||||
|
data InvalidBuildConfig = InvalidBuildConfig Text
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty InvalidBuildConfig where
|
||||||
|
pPrint (InvalidBuildConfig reason) =
|
||||||
|
text [i|The build config is invalid. Reason was: #{reason}|]
|
||||||
|
|
||||||
|
data NoToolVersionSet = NoToolVersionSet Tool
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty NoToolVersionSet where
|
||||||
|
pPrint (NoToolVersionSet tool) =
|
||||||
|
text [i|No version is set for tool "#{tool}".|]
|
||||||
|
|
||||||
|
|
||||||
-------------------------
|
-------------------------
|
||||||
@@ -99,13 +239,21 @@ data PatchFailed = PatchFailed
|
|||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
-- | A download failed. The underlying error is encapsulated.
|
-- | A download failed. The underlying error is encapsulated.
|
||||||
data DownloadFailed = forall es . Show (V es) => DownloadFailed (V es)
|
data DownloadFailed = forall x xs . (Show x, Show (V xs), Pretty x, Pretty (V xs)) => DownloadFailed (V (x ': xs))
|
||||||
|
|
||||||
|
instance Pretty DownloadFailed where
|
||||||
|
pPrint (DownloadFailed reason) =
|
||||||
|
text "Download failed:" <+> pPrint reason
|
||||||
|
|
||||||
deriving instance Show DownloadFailed
|
deriving instance Show DownloadFailed
|
||||||
|
|
||||||
|
|
||||||
-- | A build failed.
|
-- | A build failed.
|
||||||
data BuildFailed = forall es . Show (V es) => BuildFailed (Path Abs) (V es)
|
data BuildFailed = forall es . Show (V es) => BuildFailed FilePath (V es)
|
||||||
|
|
||||||
|
instance Pretty BuildFailed where
|
||||||
|
pPrint (BuildFailed path reason) =
|
||||||
|
text [i|BuildFailed failed in dir "#{path}": #{reason}|]
|
||||||
|
|
||||||
deriving instance Show BuildFailed
|
deriving instance Show BuildFailed
|
||||||
|
|
||||||
@@ -113,6 +261,10 @@ deriving instance Show BuildFailed
|
|||||||
-- | Setting the current GHC version failed.
|
-- | Setting the current GHC version failed.
|
||||||
data GHCupSetError = forall es . Show (V es) => GHCupSetError (V es)
|
data GHCupSetError = forall es . Show (V es) => GHCupSetError (V es)
|
||||||
|
|
||||||
|
instance Pretty GHCupSetError where
|
||||||
|
pPrint (GHCupSetError reason) =
|
||||||
|
text [i|Setting the current GHC version failed: #{reason}|]
|
||||||
|
|
||||||
deriving instance Show GHCupSetError
|
deriving instance Show GHCupSetError
|
||||||
|
|
||||||
|
|
||||||
@@ -125,4 +277,75 @@ deriving instance Show GHCupSetError
|
|||||||
data ParseError = ParseError String
|
data ParseError = ParseError String
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty ParseError where
|
||||||
|
pPrint (ParseError reason) =
|
||||||
|
text [i|Parsing failed: #{reason}|]
|
||||||
|
|
||||||
instance Exception ParseError
|
instance Exception ParseError
|
||||||
|
|
||||||
|
|
||||||
|
data UnexpectedListLength = UnexpectedListLength String
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty UnexpectedListLength where
|
||||||
|
pPrint (UnexpectedListLength reason) =
|
||||||
|
text [i|List length unexpected: #{reason}|]
|
||||||
|
|
||||||
|
instance Exception UnexpectedListLength
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
------------------------
|
||||||
|
--[ orphan instances ]--
|
||||||
|
------------------------
|
||||||
|
|
||||||
|
instance Pretty (V '[]) where
|
||||||
|
{-# INLINABLE pPrint #-}
|
||||||
|
pPrint _ = undefined
|
||||||
|
|
||||||
|
instance
|
||||||
|
( Pretty x
|
||||||
|
, Pretty (V xs)
|
||||||
|
) => Pretty (V (x ': xs))
|
||||||
|
where
|
||||||
|
pPrint v = case popVariantHead v of
|
||||||
|
Right x -> pPrint x
|
||||||
|
Left xs -> pPrint xs
|
||||||
|
|
||||||
|
instance Pretty URIParseError where
|
||||||
|
pPrint (MalformedScheme reason) =
|
||||||
|
text [i|Failed to parse URI. Malformed scheme: #{reason}|]
|
||||||
|
pPrint MalformedUserInfo =
|
||||||
|
text [i|Failed to parse URI. Malformed user info.|]
|
||||||
|
pPrint MalformedQuery =
|
||||||
|
text [i|Failed to parse URI. Malformed query.|]
|
||||||
|
pPrint MalformedFragment =
|
||||||
|
text [i|Failed to parse URI. Malformed fragment.|]
|
||||||
|
pPrint MalformedHost =
|
||||||
|
text [i|Failed to parse URI. Malformed host.|]
|
||||||
|
pPrint MalformedPort =
|
||||||
|
text [i|Failed to parse URI. Malformed port.|]
|
||||||
|
pPrint MalformedPath =
|
||||||
|
text [i|Failed to parse URI. Malformed path.|]
|
||||||
|
pPrint (OtherError err) =
|
||||||
|
text [i|Failed to parse URI: #{err}|]
|
||||||
|
|
||||||
|
#if !defined(TAR)
|
||||||
|
instance Pretty ArchiveResult where
|
||||||
|
pPrint ArchiveFatal = text "Archive result: fatal"
|
||||||
|
pPrint ArchiveFailed = text "Archive result: failed"
|
||||||
|
pPrint ArchiveWarn = text "Archive result: warning"
|
||||||
|
pPrint ArchiveRetry = text "Archive result: retry"
|
||||||
|
pPrint ArchiveOk = text "Archive result: Ok"
|
||||||
|
pPrint ArchiveEOF = text "Archive result: EOF"
|
||||||
|
#else
|
||||||
|
instance Pretty Tar.FormatError where
|
||||||
|
pPrint Tar.TruncatedArchive = text "Truncated archive"
|
||||||
|
pPrint Tar.ShortTrailer = text "Short trailer"
|
||||||
|
pPrint Tar.BadTrailer = text "Bad trailer"
|
||||||
|
pPrint Tar.TrailingJunk = text "Trailing junk"
|
||||||
|
pPrint Tar.ChecksumIncorrect = text "Checksum incorrect"
|
||||||
|
pPrint Tar.NotTarFormat = text "Not a tar format"
|
||||||
|
pPrint Tar.UnrecognisedTarFormat = text "Unrecognised tar format"
|
||||||
|
pPrint Tar.HeaderBadNumericEncoding = text "Header has bad numeric encoding"
|
||||||
|
#endif
|
||||||
|
|||||||
@@ -6,13 +6,21 @@
|
|||||||
{-# LANGUAGE TemplateHaskell #-}
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Plaform
|
||||||
|
Description : Retrieving platform information
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
module GHCup.Platform where
|
module GHCup.Platform where
|
||||||
|
|
||||||
|
|
||||||
import GHCup.Errors
|
import GHCup.Errors
|
||||||
import GHCup.Types
|
import GHCup.Types
|
||||||
import GHCup.Types.JSON ( )
|
import GHCup.Types.JSON ( )
|
||||||
import GHCup.Utils.Bash
|
|
||||||
import GHCup.Utils.File
|
import GHCup.Utils.File
|
||||||
import GHCup.Utils.Prelude
|
import GHCup.Utils.Prelude
|
||||||
import GHCup.Utils.String.QQ
|
import GHCup.Utils.String.QQ
|
||||||
@@ -28,35 +36,56 @@ import Data.Maybe
|
|||||||
import Data.String.Interpolate
|
import Data.String.Interpolate
|
||||||
import Data.Text ( Text )
|
import Data.Text ( Text )
|
||||||
import Data.Versions
|
import Data.Versions
|
||||||
import HPath
|
|
||||||
import HPath.IO
|
|
||||||
import Haskus.Utils.Variant.Excepts
|
import Haskus.Utils.Variant.Excepts
|
||||||
import Prelude hiding ( abs
|
import Prelude hiding ( abs
|
||||||
, readFile
|
, readFile
|
||||||
, writeFile
|
, writeFile
|
||||||
)
|
)
|
||||||
import System.Info
|
import System.Info
|
||||||
|
import System.Directory
|
||||||
|
import System.OsRelease
|
||||||
|
import Text.PrettyPrint.HughesPJClass ( prettyShow )
|
||||||
import Text.Regex.Posix
|
import Text.Regex.Posix
|
||||||
|
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
import qualified Data.Text.Encoding as E
|
import qualified Data.Text.IO as T
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--------------------------
|
--------------------------
|
||||||
--[ Platform detection ]--
|
--[ Platform detection ]--
|
||||||
--------------------------
|
--------------------------
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get the full platform request, consisting of architecture, distro, ...
|
||||||
|
platformRequest :: (Alternative m, MonadFail m, MonadLogger m, MonadCatch m, MonadIO m)
|
||||||
|
=> Excepts
|
||||||
|
'[NoCompatiblePlatform, NoCompatibleArch, DistroNotFound]
|
||||||
|
m
|
||||||
|
PlatformRequest
|
||||||
|
platformRequest = do
|
||||||
|
(PlatformResult rp rv) <- liftE getPlatform
|
||||||
|
ar <- lE getArchitecture
|
||||||
|
pure $ PlatformRequest ar rp rv
|
||||||
|
|
||||||
|
|
||||||
getArchitecture :: Either NoCompatibleArch Architecture
|
getArchitecture :: Either NoCompatibleArch Architecture
|
||||||
getArchitecture = case arch of
|
getArchitecture = case arch of
|
||||||
"x86_64" -> Right A_64
|
"x86_64" -> Right A_64
|
||||||
"i386" -> Right A_32
|
"i386" -> Right A_32
|
||||||
what -> Left (NoCompatibleArch what)
|
"powerpc" -> Right A_PowerPC
|
||||||
|
"powerpc64" -> Right A_PowerPC64
|
||||||
|
"powerpc64le" -> Right A_PowerPC64
|
||||||
|
"sparc" -> Right A_Sparc
|
||||||
|
"sparc64" -> Right A_Sparc64
|
||||||
|
"arm" -> Right A_ARM
|
||||||
|
"aarch64" -> Right A_ARM64
|
||||||
|
what -> Left (NoCompatibleArch what)
|
||||||
|
|
||||||
|
|
||||||
|
getPlatform :: (Alternative m, MonadLogger m, MonadCatch m, MonadIO m, MonadFail m)
|
||||||
getPlatform :: (MonadLogger m, MonadCatch m, MonadIO m)
|
|
||||||
=> Excepts
|
=> Excepts
|
||||||
'[NoCompatiblePlatform , DistroNotFound]
|
'[NoCompatiblePlatform, DistroNotFound]
|
||||||
m
|
m
|
||||||
PlatformResult
|
PlatformResult
|
||||||
getPlatform = do
|
getPlatform = do
|
||||||
@@ -64,28 +93,39 @@ getPlatform = do
|
|||||||
"linux" -> do
|
"linux" -> do
|
||||||
(distro, ver) <- liftE getLinuxDistro
|
(distro, ver) <- liftE getLinuxDistro
|
||||||
pure $ PlatformResult { _platform = Linux distro, _distroVersion = ver }
|
pure $ PlatformResult { _platform = Linux distro, _distroVersion = ver }
|
||||||
-- TODO: these are not verified
|
"darwin" -> do
|
||||||
"darwin" ->
|
ver <-
|
||||||
pure $ PlatformResult { _platform = Darwin, _distroVersion = Nothing }
|
either (const Nothing) Just
|
||||||
|
. versioning
|
||||||
|
-- TODO: maybe do this somewhere else
|
||||||
|
. decUTF8Safe'
|
||||||
|
<$> getDarwinVersion
|
||||||
|
pure $ PlatformResult { _platform = Darwin, _distroVersion = ver }
|
||||||
"freebsd" -> do
|
"freebsd" -> do
|
||||||
ver <- getFreeBSDVersion
|
ver <-
|
||||||
|
either (const Nothing) Just . versioning . decUTF8Safe'
|
||||||
|
<$> getFreeBSDVersion
|
||||||
pure $ PlatformResult { _platform = FreeBSD, _distroVersion = ver }
|
pure $ PlatformResult { _platform = FreeBSD, _distroVersion = ver }
|
||||||
|
"mingw32" -> pure PlatformResult { _platform = Windows, _distroVersion = Nothing }
|
||||||
what -> throwE $ NoCompatiblePlatform what
|
what -> throwE $ NoCompatiblePlatform what
|
||||||
lift $ $(logDebug) [i|Identified Platform as: #{pfr}|]
|
lift $ $(logDebug) [i|Identified Platform as: #{prettyShow pfr}|]
|
||||||
pure pfr
|
pure pfr
|
||||||
where getFreeBSDVersion = pure Nothing
|
where
|
||||||
|
getFreeBSDVersion = lift $ fmap _stdOut $ executeOut "freebsd-version" [] Nothing
|
||||||
|
getDarwinVersion = lift $ fmap _stdOut $ executeOut "sw_vers"
|
||||||
|
["-productVersion"]
|
||||||
|
Nothing
|
||||||
|
|
||||||
|
|
||||||
getLinuxDistro :: (MonadCatch m, MonadIO m)
|
getLinuxDistro :: (Alternative m, MonadCatch m, MonadIO m, MonadFail m)
|
||||||
=> Excepts '[DistroNotFound] m (LinuxDistro, Maybe Versioning)
|
=> Excepts '[DistroNotFound] m (LinuxDistro, Maybe Versioning)
|
||||||
getLinuxDistro = do
|
getLinuxDistro = do
|
||||||
-- TODO: don't do alternative on IO, because it hides bugs
|
-- TODO: don't do alternative on IO, because it hides bugs
|
||||||
(name, ver) <- handleIO (\_ -> throwE DistroNotFound) $ liftIO $ asum
|
(name, ver) <- handleIO (\_ -> throwE DistroNotFound) $ lift $ asum
|
||||||
[ try_os_release
|
[ liftIO try_os_release
|
||||||
, try_lsb_release_cmd
|
, try_lsb_release_cmd
|
||||||
, try_lsb_release
|
, liftIO try_redhat_release
|
||||||
, try_redhat_release
|
, liftIO try_debian_version
|
||||||
, try_debian_version
|
|
||||||
]
|
]
|
||||||
let parsedVer = ver >>= either (const Nothing) Just . versioning
|
let parsedVer = ver >>= either (const Nothing) Just . versioning
|
||||||
distro = if
|
distro = if
|
||||||
@@ -108,43 +148,34 @@ getLinuxDistro = do
|
|||||||
where
|
where
|
||||||
regex x = makeRegexOpts compIgnoreCase execBlank ([s|\<|] ++ x ++ [s|\>|])
|
regex x = makeRegexOpts compIgnoreCase execBlank ([s|\<|] ++ x ++ [s|\>|])
|
||||||
|
|
||||||
os_release :: Path Abs
|
lsb_release_cmd :: FilePath
|
||||||
os_release = [abs|/etc/os-release|]
|
lsb_release_cmd = "lsb-release"
|
||||||
lsb_release :: Path Abs
|
redhat_release :: FilePath
|
||||||
lsb_release = [abs|/etc/lsb-release|]
|
redhat_release = "/etc/redhat-release"
|
||||||
lsb_release_cmd :: Path Rel
|
debian_version :: FilePath
|
||||||
lsb_release_cmd = [rel|lsb-release|]
|
debian_version = "/etc/debian_version"
|
||||||
redhat_release :: Path Abs
|
|
||||||
redhat_release = [abs|/etc/redhat-release|]
|
|
||||||
debian_version :: Path Abs
|
|
||||||
debian_version = [abs|/etc/debian_version|]
|
|
||||||
|
|
||||||
try_os_release :: IO (Text, Maybe Text)
|
try_os_release :: IO (Text, Maybe Text)
|
||||||
try_os_release = do
|
try_os_release = do
|
||||||
(Just name) <- getAssignmentValueFor os_release "NAME"
|
Just OsRelease{ name = name, version_id = version_id } <-
|
||||||
ver <- getAssignmentValueFor os_release "VERSION_ID"
|
fmap osRelease <$> parseOsRelease
|
||||||
pure (T.pack name, fmap T.pack ver)
|
pure (T.pack name, fmap T.pack version_id)
|
||||||
|
|
||||||
try_lsb_release_cmd :: IO (Text, Maybe Text)
|
try_lsb_release_cmd :: (MonadFail m, MonadIO m)
|
||||||
|
=> m (Text, Maybe Text)
|
||||||
try_lsb_release_cmd = do
|
try_lsb_release_cmd = do
|
||||||
(Just _) <- findExecutable lsb_release_cmd
|
(Just _) <- liftIO $ findExecutable lsb_release_cmd
|
||||||
name <- fmap _stdOut $ executeOut lsb_release_cmd ["-si"] Nothing
|
name <- fmap _stdOut $ executeOut lsb_release_cmd ["-si"] Nothing
|
||||||
ver <- fmap _stdOut $ executeOut lsb_release_cmd ["-sr"] Nothing
|
ver <- fmap _stdOut $ executeOut lsb_release_cmd ["-sr"] Nothing
|
||||||
pure (E.decodeUtf8 name, Just $ E.decodeUtf8 ver)
|
pure (decUTF8Safe' name, Just $ decUTF8Safe' ver)
|
||||||
|
|
||||||
try_lsb_release :: IO (Text, Maybe Text)
|
|
||||||
try_lsb_release = do
|
|
||||||
(Just name) <- getAssignmentValueFor lsb_release "DISTRIB_ID"
|
|
||||||
ver <- getAssignmentValueFor lsb_release "DISTRIB_RELEASE"
|
|
||||||
pure (T.pack name, fmap T.pack ver)
|
|
||||||
|
|
||||||
try_redhat_release :: IO (Text, Maybe Text)
|
try_redhat_release :: IO (Text, Maybe Text)
|
||||||
try_redhat_release = do
|
try_redhat_release = do
|
||||||
t <- fmap lBS2sT $ readFile redhat_release
|
t <- T.readFile redhat_release
|
||||||
let nameRegex n =
|
let nameRegex n =
|
||||||
makeRegexOpts compIgnoreCase
|
makeRegexOpts compIgnoreCase
|
||||||
execBlank
|
execBlank
|
||||||
(([s|\<|] <> fS n <> [s|\>|] :: ByteString)) :: Regex
|
([s|\<|] <> fS n <> [s|\>|] :: ByteString) :: Regex
|
||||||
let verRegex =
|
let verRegex =
|
||||||
makeRegexOpts compIgnoreCase
|
makeRegexOpts compIgnoreCase
|
||||||
execBlank
|
execBlank
|
||||||
@@ -162,5 +193,5 @@ getLinuxDistro = do
|
|||||||
|
|
||||||
try_debian_version :: IO (Text, Maybe Text)
|
try_debian_version :: IO (Text, Maybe Text)
|
||||||
try_debian_version = do
|
try_debian_version = do
|
||||||
ver <- readFile debian_version
|
ver <- T.readFile debian_version
|
||||||
pure (T.pack "debian", Just $ lBS2sT ver)
|
pure (T.pack "debian", Just ver)
|
||||||
|
|||||||
68
lib/GHCup/Requirements.hs
Normal file
68
lib/GHCup/Requirements.hs
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Requirements
|
||||||
|
Description : Requirements utilities
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
|
module GHCup.Requirements where
|
||||||
|
|
||||||
|
import GHCup.Types
|
||||||
|
import GHCup.Types.JSON ( )
|
||||||
|
import GHCup.Types.Optics
|
||||||
|
import GHCup.Version
|
||||||
|
|
||||||
|
import Control.Applicative
|
||||||
|
import Data.List ( find )
|
||||||
|
import Data.Maybe
|
||||||
|
import Optics
|
||||||
|
import Prelude hiding ( abs
|
||||||
|
, readFile
|
||||||
|
, writeFile
|
||||||
|
)
|
||||||
|
|
||||||
|
import qualified Data.Map.Strict as M
|
||||||
|
import qualified Data.Text as T
|
||||||
|
|
||||||
|
|
||||||
|
-- | Get the requirements. Right now this combines GHC and cabal
|
||||||
|
-- and doesn't do fine-grained distinction. However, the 'ToolRequirements'
|
||||||
|
-- type allows it.
|
||||||
|
getCommonRequirements :: PlatformResult
|
||||||
|
-> ToolRequirements
|
||||||
|
-> Maybe Requirements
|
||||||
|
getCommonRequirements pr tr =
|
||||||
|
with_distro <|> without_distro_ver <|> without_distro
|
||||||
|
where
|
||||||
|
with_distro = distro_preview _platform _distroVersion
|
||||||
|
without_distro_ver = distro_preview _platform (const Nothing)
|
||||||
|
without_distro = distro_preview (set _Linux UnknownLinux . _platform) (const Nothing)
|
||||||
|
|
||||||
|
distro_preview f g =
|
||||||
|
let platformVersionSpec =
|
||||||
|
preview (ix GHC % ix Nothing % ix (f pr)) tr
|
||||||
|
mv' = g pr
|
||||||
|
in fmap snd
|
||||||
|
. find
|
||||||
|
(\(mverRange, _) -> maybe
|
||||||
|
(isNothing mv')
|
||||||
|
(\range -> maybe False (`versionRange` range) mv')
|
||||||
|
mverRange
|
||||||
|
)
|
||||||
|
. M.toList
|
||||||
|
=<< platformVersionSpec
|
||||||
|
|
||||||
|
|
||||||
|
prettyRequirements :: Requirements -> T.Text
|
||||||
|
prettyRequirements Requirements {..} =
|
||||||
|
let d = if not . null $ _distroPKGs
|
||||||
|
then
|
||||||
|
"\n Please install the following distro packages: "
|
||||||
|
<> T.intercalate " " _distroPKGs
|
||||||
|
else ""
|
||||||
|
n = if not . T.null $ _notes then "\n Note: " <> _notes else ""
|
||||||
|
in "System requirements " <> d <> n
|
||||||
@@ -1,16 +1,86 @@
|
|||||||
{-# LANGUAGE DeriveGeneric #-}
|
{-# OPTIONS_GHC -Wno-orphans #-}
|
||||||
|
{-# LANGUAGE CPP #-}
|
||||||
|
{-# LANGUAGE DeriveGeneric #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
module GHCup.Types where
|
{-|
|
||||||
|
Module : GHCup.Types
|
||||||
|
Description : GHCup types
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
|
module GHCup.Types
|
||||||
|
( module GHCup.Types
|
||||||
|
#if defined(BRICK)
|
||||||
|
, Key(..)
|
||||||
|
#endif
|
||||||
|
)
|
||||||
|
where
|
||||||
|
|
||||||
|
import Control.Applicative
|
||||||
|
import Control.Monad.Logger
|
||||||
import Data.Map.Strict ( Map )
|
import Data.Map.Strict ( Map )
|
||||||
|
import Data.List.NonEmpty ( NonEmpty (..) )
|
||||||
import Data.Text ( Text )
|
import Data.Text ( Text )
|
||||||
import Data.Versions
|
import Data.Versions
|
||||||
import HPath
|
import Haskus.Utils.Variant.Excepts
|
||||||
|
import Text.PrettyPrint.HughesPJClass (Pretty, pPrint, text)
|
||||||
import URI.ByteString
|
import URI.ByteString
|
||||||
|
#if defined(BRICK)
|
||||||
|
import Graphics.Vty ( Key(..) )
|
||||||
|
#endif
|
||||||
|
|
||||||
|
import qualified Control.Monad.Trans.Class as Trans
|
||||||
|
import qualified Data.Text as T
|
||||||
import qualified GHC.Generics as GHC
|
import qualified GHC.Generics as GHC
|
||||||
|
|
||||||
|
|
||||||
|
#if !defined(BRICK)
|
||||||
|
data Key = KEsc | KChar Char | KBS | KEnter
|
||||||
|
| KLeft | KRight | KUp | KDown
|
||||||
|
| KUpLeft | KUpRight | KDownLeft | KDownRight | KCenter
|
||||||
|
| KFun Int | KBackTab | KPrtScr | KPause | KIns
|
||||||
|
| KHome | KPageUp | KDel | KEnd | KPageDown | KBegin | KMenu
|
||||||
|
deriving (Eq,Show,Read,Ord,GHC.Generic)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
--------------------
|
||||||
|
--[ GHCInfo Tree ]--
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
|
||||||
|
data GHCupInfo = GHCupInfo
|
||||||
|
{ _toolRequirements :: ToolRequirements
|
||||||
|
, _ghcupDownloads :: GHCupDownloads
|
||||||
|
, _globalTools :: Map GlobalTool DownloadInfo
|
||||||
|
}
|
||||||
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-------------------------
|
||||||
|
--[ Requirements Tree ]--
|
||||||
|
-------------------------
|
||||||
|
|
||||||
|
|
||||||
|
type ToolRequirements = Map Tool ToolReqVersionSpec
|
||||||
|
type ToolReqVersionSpec = Map (Maybe Version) PlatformReqSpec
|
||||||
|
type PlatformReqSpec = Map Platform PlatformReqVersionSpec
|
||||||
|
type PlatformReqVersionSpec = Map (Maybe VersionRange) Requirements
|
||||||
|
|
||||||
|
|
||||||
|
data Requirements = Requirements
|
||||||
|
{ _distroPKGs :: [Text]
|
||||||
|
, _notes :: Text
|
||||||
|
}
|
||||||
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
---------------------
|
---------------------
|
||||||
@@ -24,44 +94,102 @@ type GHCupDownloads = Map Tool ToolVersionSpec
|
|||||||
type ToolVersionSpec = Map Version VersionInfo
|
type ToolVersionSpec = Map Version VersionInfo
|
||||||
type ArchitectureSpec = Map Architecture PlatformSpec
|
type ArchitectureSpec = Map Architecture PlatformSpec
|
||||||
type PlatformSpec = Map Platform PlatformVersionSpec
|
type PlatformSpec = Map Platform PlatformVersionSpec
|
||||||
type PlatformVersionSpec = Map (Maybe Versioning) DownloadInfo
|
type PlatformVersionSpec = Map (Maybe VersionRange) DownloadInfo
|
||||||
|
|
||||||
|
|
||||||
-- | An installable tool.
|
-- | An installable tool.
|
||||||
data Tool = GHC
|
data Tool = GHC
|
||||||
| Cabal
|
| Cabal
|
||||||
| GHCup
|
| GHCup
|
||||||
deriving (Eq, GHC.Generic, Ord, Show)
|
| HLS
|
||||||
|
| Stack
|
||||||
|
deriving (Eq, GHC.Generic, Ord, Show, Enum, Bounded)
|
||||||
|
|
||||||
|
data GlobalTool = ShimGen
|
||||||
|
deriving (Eq, GHC.Generic, Ord, Show, Enum, Bounded)
|
||||||
|
|
||||||
|
|
||||||
-- | All necessary information of a tool version, including
|
-- | All necessary information of a tool version, including
|
||||||
-- source download and per-architecture downloads.
|
-- source download and per-architecture downloads.
|
||||||
data VersionInfo = VersionInfo
|
data VersionInfo = VersionInfo
|
||||||
{ _viTags :: [Tag] -- ^ version specific tag
|
{ _viTags :: [Tag] -- ^ version specific tag
|
||||||
, _viSourceDL :: Maybe DownloadInfo -- ^ source tarball
|
, _viChangeLog :: Maybe URI
|
||||||
, _viArch :: ArchitectureSpec -- ^ descend for binary downloads per arch
|
, _viSourceDL :: Maybe DownloadInfo -- ^ source tarball
|
||||||
|
, _viArch :: ArchitectureSpec -- ^ descend for binary downloads per arch
|
||||||
|
-- informative messages
|
||||||
|
, _viPostInstall :: Maybe Text
|
||||||
|
, _viPostRemove :: Maybe Text
|
||||||
|
, _viPreCompile :: Maybe Text
|
||||||
}
|
}
|
||||||
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.
|
||||||
data Tag = Latest
|
data Tag = Latest
|
||||||
| Recommended
|
| Recommended
|
||||||
deriving (Ord, Eq, Show)
|
| Prerelease
|
||||||
|
| Base PVP
|
||||||
|
| Old -- ^ old version are hidden by default in TUI
|
||||||
|
| UnknownTag String -- ^ used for upwardscompat
|
||||||
|
deriving (Ord, Eq, GHC.Generic, Show) -- FIXME: manual JSON instance
|
||||||
|
|
||||||
|
tagToString :: Tag -> String
|
||||||
|
tagToString Recommended = "recommended"
|
||||||
|
tagToString Latest = "latest"
|
||||||
|
tagToString Prerelease = "prerelease"
|
||||||
|
tagToString (Base pvp'') = "base-" ++ T.unpack (prettyPVP pvp'')
|
||||||
|
tagToString (UnknownTag t ) = t
|
||||||
|
tagToString Old = ""
|
||||||
|
|
||||||
|
instance Pretty Tag where
|
||||||
|
pPrint Recommended = text "recommended"
|
||||||
|
pPrint Latest = text "latest"
|
||||||
|
pPrint Prerelease = text "prerelease"
|
||||||
|
pPrint (Base pvp'') = text ("base-" ++ T.unpack (prettyPVP pvp''))
|
||||||
|
pPrint (UnknownTag t ) = text t
|
||||||
|
pPrint Old = mempty
|
||||||
|
|
||||||
data Architecture = A_64
|
data Architecture = A_64
|
||||||
| A_32
|
| A_32
|
||||||
|
| A_PowerPC
|
||||||
|
| A_PowerPC64
|
||||||
|
| A_Sparc
|
||||||
|
| A_Sparc64
|
||||||
|
| A_ARM
|
||||||
|
| A_ARM64
|
||||||
deriving (Eq, GHC.Generic, Ord, Show)
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
archToString :: Architecture -> String
|
||||||
|
archToString A_64 = "x86_64"
|
||||||
|
archToString A_32 = "i386"
|
||||||
|
archToString A_PowerPC = "powerpc"
|
||||||
|
archToString A_PowerPC64 = "powerpc64"
|
||||||
|
archToString A_Sparc = "sparc"
|
||||||
|
archToString A_Sparc64 = "sparc64"
|
||||||
|
archToString A_ARM = "arm"
|
||||||
|
archToString A_ARM64 = "aarch64"
|
||||||
|
|
||||||
|
instance Pretty Architecture where
|
||||||
|
pPrint = text . archToString
|
||||||
|
|
||||||
data Platform = Linux LinuxDistro
|
data Platform = Linux LinuxDistro
|
||||||
-- ^ must exit
|
-- ^ must exit
|
||||||
| Darwin
|
| Darwin
|
||||||
-- ^ must exit
|
-- ^ must exit
|
||||||
| FreeBSD
|
| FreeBSD
|
||||||
|
| Windows
|
||||||
|
-- ^ must exit
|
||||||
deriving (Eq, GHC.Generic, Ord, Show)
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
platformToString :: Platform -> String
|
||||||
|
platformToString (Linux distro) = "linux-" ++ distroToString distro
|
||||||
|
platformToString Darwin = "darwin"
|
||||||
|
platformToString FreeBSD = "freebsd"
|
||||||
|
platformToString Windows = "windows"
|
||||||
|
|
||||||
|
instance Pretty Platform where
|
||||||
|
pPrint = text . platformToString
|
||||||
|
|
||||||
data LinuxDistro = Debian
|
data LinuxDistro = Debian
|
||||||
| Ubuntu
|
| Ubuntu
|
||||||
| Mint
|
| Mint
|
||||||
@@ -78,15 +206,31 @@ data LinuxDistro = Debian
|
|||||||
-- ^ must exit
|
-- ^ must exit
|
||||||
deriving (Eq, GHC.Generic, Ord, Show)
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
distroToString :: LinuxDistro -> String
|
||||||
|
distroToString Debian = "debian"
|
||||||
|
distroToString Ubuntu = "ubuntu"
|
||||||
|
distroToString Mint= "mint"
|
||||||
|
distroToString Fedora = "fedora"
|
||||||
|
distroToString CentOS = "centos"
|
||||||
|
distroToString RedHat = "redhat"
|
||||||
|
distroToString Alpine = "alpine"
|
||||||
|
distroToString AmazonLinux = "amazon"
|
||||||
|
distroToString Gentoo = "gentoo"
|
||||||
|
distroToString Exherbo = "exherbo"
|
||||||
|
distroToString UnknownLinux = "unknown"
|
||||||
|
|
||||||
|
instance Pretty LinuxDistro where
|
||||||
|
pPrint = text . distroToString
|
||||||
|
|
||||||
|
|
||||||
-- | 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.
|
||||||
data DownloadInfo = DownloadInfo
|
data DownloadInfo = DownloadInfo
|
||||||
{ _dlUri :: URI
|
{ _dlUri :: URI
|
||||||
, _dlSubdir :: Maybe (Path Rel)
|
, _dlSubdir :: Maybe TarDir
|
||||||
, _dlHash :: Text
|
, _dlHash :: Text
|
||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, Ord, GHC.Generic, Show)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -96,25 +240,121 @@ data DownloadInfo = DownloadInfo
|
|||||||
--------------
|
--------------
|
||||||
|
|
||||||
|
|
||||||
|
-- | How to descend into a tar archive.
|
||||||
|
data TarDir = RealDir FilePath
|
||||||
|
| RegexDir String -- ^ will be compiled to regex, the first match will "win"
|
||||||
|
deriving (Eq, Ord, GHC.Generic, Show)
|
||||||
|
|
||||||
|
instance Pretty TarDir where
|
||||||
|
pPrint (RealDir path) = text path
|
||||||
|
pPrint (RegexDir regex) = text regex
|
||||||
|
|
||||||
|
|
||||||
-- | Where to fetch GHCupDownloads from.
|
-- | Where to fetch GHCupDownloads from.
|
||||||
data URLSource = GHCupURL
|
data URLSource = GHCupURL
|
||||||
| OwnSource URI
|
| OwnSource URI
|
||||||
| OwnSpec GHCupDownloads
|
| 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 Key
|
||||||
|
, kDown :: Maybe Key
|
||||||
|
, kQuit :: Maybe Key
|
||||||
|
, kInstall :: Maybe Key
|
||||||
|
, kUninstall :: Maybe Key
|
||||||
|
, kSet :: Maybe Key
|
||||||
|
, kChangelog :: Maybe Key
|
||||||
|
, kShowAll :: Maybe Key
|
||||||
|
, kShowAllTools :: Maybe Key
|
||||||
|
}
|
||||||
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
|
data KeyBindings = KeyBindings
|
||||||
|
{ bUp :: Key
|
||||||
|
, bDown :: Key
|
||||||
|
, bQuit :: Key
|
||||||
|
, bInstall :: Key
|
||||||
|
, bUninstall :: Key
|
||||||
|
, bSet :: Key
|
||||||
|
, bChangelog :: Key
|
||||||
|
, bShowAllVersions :: Key
|
||||||
|
, bShowAllTools :: Key
|
||||||
|
}
|
||||||
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
|
defaultKeyBindings :: KeyBindings
|
||||||
|
defaultKeyBindings = KeyBindings
|
||||||
|
{ bUp = KUp
|
||||||
|
, bDown = KDown
|
||||||
|
, bQuit = KChar 'q'
|
||||||
|
, bInstall = KChar 'i'
|
||||||
|
, bUninstall = KChar 'u'
|
||||||
|
, bSet = KChar 's'
|
||||||
|
, bChangelog = KChar 'c'
|
||||||
|
, bShowAllVersions = KChar 'a'
|
||||||
|
, bShowAllTools = KChar 't'
|
||||||
|
}
|
||||||
|
|
||||||
|
data AppState = AppState
|
||||||
|
{ settings :: Settings
|
||||||
|
, dirs :: Dirs
|
||||||
|
, keyBindings :: KeyBindings
|
||||||
|
, ghcupInfo :: GHCupInfo
|
||||||
|
, pfreq :: PlatformRequest
|
||||||
|
} deriving (Show)
|
||||||
|
|
||||||
data Settings = Settings
|
data Settings = Settings
|
||||||
{ cache :: Bool
|
{ cache :: Bool
|
||||||
, noVerify :: Bool
|
, noVerify :: Bool
|
||||||
|
, keepDirs :: KeepDirs
|
||||||
|
, downloader :: Downloader
|
||||||
|
, verbose :: Bool
|
||||||
|
, urlSource :: URLSource
|
||||||
|
}
|
||||||
|
deriving (Show, GHC.Generic)
|
||||||
|
|
||||||
|
data Dirs = Dirs
|
||||||
|
{ baseDir :: FilePath
|
||||||
|
, binDir :: FilePath
|
||||||
|
, cacheDir :: FilePath
|
||||||
|
, logsDir :: FilePath
|
||||||
|
, confDir :: FilePath
|
||||||
}
|
}
|
||||||
deriving Show
|
deriving Show
|
||||||
|
|
||||||
|
data KeepDirs = Always
|
||||||
|
| Errors
|
||||||
|
| Never
|
||||||
|
deriving (Eq, Show, Ord)
|
||||||
|
|
||||||
|
data Downloader = Curl
|
||||||
|
| Wget
|
||||||
|
#if defined(INTERNAL_DOWNLOADER)
|
||||||
|
| Internal
|
||||||
|
#endif
|
||||||
|
deriving (Eq, Show, Ord)
|
||||||
|
|
||||||
data DebugInfo = DebugInfo
|
data DebugInfo = DebugInfo
|
||||||
{ diBaseDir :: Path Abs
|
{ diBaseDir :: FilePath
|
||||||
, diBinDir :: Path Abs
|
, diBinDir :: FilePath
|
||||||
, diGHCDir :: Path Abs
|
, diGHCDir :: FilePath
|
||||||
, diCacheDir :: Path Abs
|
, diCacheDir :: FilePath
|
||||||
, diArch :: Architecture
|
, diArch :: Architecture
|
||||||
, diPlatform :: PlatformResult
|
, diPlatform :: PlatformResult
|
||||||
}
|
}
|
||||||
@@ -133,9 +373,90 @@ data PlatformResult = PlatformResult
|
|||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, Show)
|
||||||
|
|
||||||
|
platResToString :: PlatformResult -> String
|
||||||
|
platResToString PlatformResult { _platform = plat, _distroVersion = Just v' }
|
||||||
|
= show plat <> ", " <> T.unpack (prettyV v')
|
||||||
|
platResToString PlatformResult { _platform = plat, _distroVersion = Nothing }
|
||||||
|
= show plat
|
||||||
|
|
||||||
|
instance Pretty PlatformResult where
|
||||||
|
pPrint = text . platResToString
|
||||||
|
|
||||||
data PlatformRequest = PlatformRequest
|
data PlatformRequest = PlatformRequest
|
||||||
{ _rArch :: Architecture
|
{ _rArch :: Architecture
|
||||||
, _rPlatform :: Platform
|
, _rPlatform :: Platform
|
||||||
, _rVersion :: Maybe Versioning
|
, _rVersion :: Maybe Versioning
|
||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, Show)
|
||||||
|
|
||||||
|
pfReqToString :: PlatformRequest -> String
|
||||||
|
pfReqToString (PlatformRequest arch plat ver) =
|
||||||
|
archToString arch ++ "-" ++ platformToString plat ++ pver
|
||||||
|
where
|
||||||
|
pver = case ver of
|
||||||
|
Just v' -> "-" ++ T.unpack (prettyV v')
|
||||||
|
Nothing -> ""
|
||||||
|
|
||||||
|
instance Pretty PlatformRequest where
|
||||||
|
pPrint = text . pfReqToString
|
||||||
|
|
||||||
|
-- | A GHC identified by the target platform triple
|
||||||
|
-- and the version.
|
||||||
|
data GHCTargetVersion = GHCTargetVersion
|
||||||
|
{ _tvTarget :: Maybe Text
|
||||||
|
, _tvVersion :: Version
|
||||||
|
}
|
||||||
|
deriving (Ord, Eq, Show)
|
||||||
|
|
||||||
|
data GitBranch = GitBranch
|
||||||
|
{ ref :: String
|
||||||
|
, repo :: Maybe String
|
||||||
|
}
|
||||||
|
deriving (Ord, Eq, Show)
|
||||||
|
|
||||||
|
mkTVer :: Version -> GHCTargetVersion
|
||||||
|
mkTVer = GHCTargetVersion Nothing
|
||||||
|
|
||||||
|
tVerToText :: GHCTargetVersion -> Text
|
||||||
|
tVerToText (GHCTargetVersion (Just t) v') = t <> "-" <> prettyVer v'
|
||||||
|
tVerToText (GHCTargetVersion Nothing v') = prettyVer v'
|
||||||
|
|
||||||
|
-- | Assembles a path of the form: <target-triple>-<version>
|
||||||
|
instance Pretty GHCTargetVersion where
|
||||||
|
pPrint = text . T.unpack . tVerToText
|
||||||
|
|
||||||
|
|
||||||
|
-- | A comparator and a version.
|
||||||
|
data VersionCmp = VR_gt Versioning
|
||||||
|
| VR_gteq Versioning
|
||||||
|
| VR_lt Versioning
|
||||||
|
| VR_lteq Versioning
|
||||||
|
| VR_eq Versioning
|
||||||
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
|
||||||
|
-- | A version range. Supports && and ||, but not arbitrary
|
||||||
|
-- combinations. This is a little simplified.
|
||||||
|
data VersionRange = SimpleRange (NonEmpty VersionCmp) -- And
|
||||||
|
| OrRange (NonEmpty VersionCmp) VersionRange
|
||||||
|
deriving (Eq, GHC.Generic, Ord, Show)
|
||||||
|
|
||||||
|
|
||||||
|
instance Pretty Versioning where
|
||||||
|
pPrint = text . T.unpack . prettyV
|
||||||
|
|
||||||
|
instance Pretty Version where
|
||||||
|
pPrint = text . T.unpack . prettyVer
|
||||||
|
|
||||||
|
|
||||||
|
instance (Monad m, Alternative m) => Alternative (LoggingT m) where
|
||||||
|
empty = Trans.lift empty
|
||||||
|
{-# INLINE empty #-}
|
||||||
|
m <|> n = LoggingT $ \ r -> runLoggingT m r <|> runLoggingT n r
|
||||||
|
{-# INLINE (<|>) #-}
|
||||||
|
|
||||||
|
|
||||||
|
instance MonadLogger m => MonadLogger (Excepts e m) where
|
||||||
|
monadLoggerLog a b c d = Trans.lift $ monadLoggerLog a b c d
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -10,39 +10,72 @@
|
|||||||
{-# LANGUAGE TemplateHaskell #-}
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
{-# LANGUAGE TypeFamilies #-}
|
{-# LANGUAGE TypeFamilies #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Types.JSON
|
||||||
|
Description : GHCup JSON types/instances
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
module GHCup.Types.JSON where
|
module GHCup.Types.JSON where
|
||||||
|
|
||||||
import GHCup.Types
|
import GHCup.Types
|
||||||
|
import GHCup.Utils.MegaParsec
|
||||||
import GHCup.Utils.Prelude
|
import GHCup.Utils.Prelude
|
||||||
|
|
||||||
|
import Control.Applicative ( (<|>) )
|
||||||
import Data.Aeson
|
import Data.Aeson
|
||||||
import Data.Aeson.TH
|
import Data.Aeson.TH
|
||||||
import Data.Aeson.Types
|
import Data.Aeson.Types
|
||||||
|
import Data.List.NonEmpty ( NonEmpty(..) )
|
||||||
import Data.Text.Encoding as E
|
import Data.Text.Encoding as E
|
||||||
import Data.Versions
|
import Data.Versions
|
||||||
import Data.Word8
|
import Data.Void
|
||||||
import HPath
|
|
||||||
import URI.ByteString
|
import URI.ByteString
|
||||||
|
import Text.Casing
|
||||||
|
|
||||||
import qualified Data.ByteString as BS
|
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.Char as MPC
|
||||||
|
|
||||||
|
|
||||||
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 } ''Platform
|
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''SemVer
|
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Tool
|
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''VSep
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''VSep
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''VUnit
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''VUnit
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''VersionInfo
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''MChunk
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Tag
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Platform
|
||||||
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''DownloadInfo
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Mess
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''SemVer
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Tool
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''GlobalTool
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''KeepDirs
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Downloader
|
||||||
|
|
||||||
|
instance ToJSON Tag where
|
||||||
|
toJSON Latest = String "Latest"
|
||||||
|
toJSON Recommended = String "Recommended"
|
||||||
|
toJSON Prerelease = String "Prerelease"
|
||||||
|
toJSON Old = String "old"
|
||||||
|
toJSON (Base pvp'') = String ("base-" <> prettyPVP pvp'')
|
||||||
|
toJSON (UnknownTag x ) = String (T.pack x)
|
||||||
|
|
||||||
|
instance FromJSON Tag where
|
||||||
|
parseJSON = withText "Tag" $ \t -> case T.unpack t of
|
||||||
|
"Latest" -> pure Latest
|
||||||
|
"Recommended" -> pure Recommended
|
||||||
|
"Prerelease" -> pure Prerelease
|
||||||
|
"old" -> pure Old
|
||||||
|
('b' : 'a' : 's' : 'e' : '-' : ver') -> case pvp (T.pack ver') of
|
||||||
|
Right x -> pure $ Base x
|
||||||
|
Left e -> fail . show $ e
|
||||||
|
x -> pure (UnknownTag x)
|
||||||
|
|
||||||
instance ToJSON URI where
|
instance ToJSON URI where
|
||||||
toJSON = toJSON . decodeUtf8 . serializeURIRef'
|
toJSON = toJSON . decUTF8Safe . serializeURIRef'
|
||||||
|
|
||||||
instance FromJSON URI where
|
instance FromJSON URI where
|
||||||
parseJSON = withText "URL" $ \t ->
|
parseJSON = withText "URL" $ \t ->
|
||||||
@@ -69,14 +102,14 @@ instance FromJSONKey Versioning where
|
|||||||
instance ToJSONKey (Maybe Versioning) where
|
instance ToJSONKey (Maybe Versioning) where
|
||||||
toJSONKey = toJSONKeyText $ \case
|
toJSONKey = toJSONKeyText $ \case
|
||||||
Just x -> prettyV x
|
Just x -> prettyV x
|
||||||
Nothing -> T.pack "unknown_version"
|
Nothing -> T.pack "unknown_versioning"
|
||||||
|
|
||||||
instance FromJSONKey (Maybe Versioning) where
|
instance FromJSONKey (Maybe Versioning) where
|
||||||
fromJSONKey = FromJSONKeyTextParser $ \t ->
|
fromJSONKey = FromJSONKeyTextParser $ \t ->
|
||||||
if t == T.pack "unknown_version" then pure Nothing else pure $ just t
|
if t == T.pack "unknown_versioning" then pure Nothing else just t
|
||||||
where
|
where
|
||||||
just t = case versioning t of
|
just t = case versioning t of
|
||||||
Right x -> pure x
|
Right x -> pure $ Just x
|
||||||
Left e -> fail $ "Failure in (Maybe Versioning) (FromJSONKey)" <> show e
|
Left e -> fail $ "Failure in (Maybe Versioning) (FromJSONKey)" <> show e
|
||||||
|
|
||||||
instance ToJSONKey Platform where
|
instance ToJSONKey Platform where
|
||||||
@@ -84,11 +117,13 @@ instance ToJSONKey Platform where
|
|||||||
Darwin -> T.pack "Darwin"
|
Darwin -> T.pack "Darwin"
|
||||||
FreeBSD -> T.pack "FreeBSD"
|
FreeBSD -> T.pack "FreeBSD"
|
||||||
Linux d -> T.pack ("Linux_" <> show d)
|
Linux d -> T.pack ("Linux_" <> show d)
|
||||||
|
Windows -> T.pack "Windows"
|
||||||
|
|
||||||
instance FromJSONKey Platform where
|
instance FromJSONKey Platform where
|
||||||
fromJSONKey = FromJSONKeyTextParser $ \t -> if
|
fromJSONKey = FromJSONKeyTextParser $ \t -> if
|
||||||
| T.pack "Darwin" == t -> pure Darwin
|
| T.pack "Darwin" == t -> pure Darwin
|
||||||
| T.pack "FreeBSD" == t -> pure FreeBSD
|
| T.pack "FreeBSD" == t -> pure FreeBSD
|
||||||
|
| T.pack "Windows" == t -> pure Windows
|
||||||
| T.pack "Linux_" `T.isPrefixOf` t -> case
|
| T.pack "Linux_" `T.isPrefixOf` t -> case
|
||||||
T.stripPrefix (T.pack "Linux_") t
|
T.stripPrefix (T.pack "Linux_") t
|
||||||
of
|
of
|
||||||
@@ -104,7 +139,7 @@ instance FromJSONKey Platform where
|
|||||||
$ "Unexpected failure in decoding LinuxDistro: "
|
$ "Unexpected failure in decoding LinuxDistro: "
|
||||||
<> show dstr
|
<> show dstr
|
||||||
Nothing -> fail "Unexpected failure in Platform stripPrefix"
|
Nothing -> fail "Unexpected failure in Platform stripPrefix"
|
||||||
| otherwise -> fail $ "Failure in Platform (FromJSONKey)"
|
| otherwise -> fail "Failure in Platform (FromJSONKey)"
|
||||||
|
|
||||||
instance ToJSONKey Architecture where
|
instance ToJSONKey Architecture where
|
||||||
toJSONKey = genericToJSONKey defaultJSONKeyOptions
|
toJSONKey = genericToJSONKey defaultJSONKeyOptions
|
||||||
@@ -112,6 +147,19 @@ instance ToJSONKey Architecture where
|
|||||||
instance FromJSONKey Architecture where
|
instance FromJSONKey Architecture where
|
||||||
fromJSONKey = genericFromJSONKey defaultJSONKeyOptions
|
fromJSONKey = genericFromJSONKey defaultJSONKeyOptions
|
||||||
|
|
||||||
|
instance ToJSONKey (Maybe Version) where
|
||||||
|
toJSONKey = toJSONKeyText $ \case
|
||||||
|
Just x -> prettyVer x
|
||||||
|
Nothing -> T.pack "unknown_version"
|
||||||
|
|
||||||
|
instance FromJSONKey (Maybe Version) where
|
||||||
|
fromJSONKey = FromJSONKeyTextParser $ \t ->
|
||||||
|
if t == T.pack "unknown_version" then pure Nothing else just t
|
||||||
|
where
|
||||||
|
just t = case version t of
|
||||||
|
Right x -> pure $ Just x
|
||||||
|
Left e -> fail $ "Failure in (Maybe Version) (FromJSONKey)" <> show e
|
||||||
|
|
||||||
instance ToJSON Version where
|
instance ToJSON Version where
|
||||||
toJSON = toJSON . prettyVer
|
toJSON = toJSON . prettyVer
|
||||||
|
|
||||||
@@ -128,21 +176,143 @@ instance FromJSONKey Version where
|
|||||||
Right x -> pure x
|
Right x -> pure x
|
||||||
Left e -> fail $ "Failure in Version (FromJSONKey)" <> show e
|
Left e -> fail $ "Failure in Version (FromJSONKey)" <> show e
|
||||||
|
|
||||||
|
instance ToJSON PVP where
|
||||||
|
toJSON = toJSON . prettyPVP
|
||||||
|
|
||||||
|
instance FromJSON PVP where
|
||||||
|
parseJSON = withText "PVP" $ \t -> case pvp t of
|
||||||
|
Right x -> pure x
|
||||||
|
Left e -> fail $ "Failure in PVP (FromJSON)" <> show e
|
||||||
|
|
||||||
instance ToJSONKey Tool where
|
instance ToJSONKey Tool where
|
||||||
toJSONKey = genericToJSONKey defaultJSONKeyOptions
|
toJSONKey = genericToJSONKey defaultJSONKeyOptions
|
||||||
|
|
||||||
instance FromJSONKey Tool where
|
instance FromJSONKey Tool where
|
||||||
fromJSONKey = genericFromJSONKey defaultJSONKeyOptions
|
fromJSONKey = genericFromJSONKey defaultJSONKeyOptions
|
||||||
|
|
||||||
instance ToJSON (Path Rel) where
|
instance ToJSONKey GlobalTool where
|
||||||
toJSON p = case and . fmap isAscii . BS.unpack $ fp of
|
toJSONKey = genericToJSONKey defaultJSONKeyOptions
|
||||||
True -> toJSON . E.decodeUtf8 $ fp
|
|
||||||
False -> String "/not/a/valid/path"
|
|
||||||
where fp = toFilePath p
|
|
||||||
|
|
||||||
instance FromJSON (Path Rel) where
|
instance FromJSONKey GlobalTool where
|
||||||
parseJSON = withText "HPath Rel" $ \t -> do
|
fromJSONKey = genericFromJSONKey defaultJSONKeyOptions
|
||||||
let d = encodeUtf8 t
|
|
||||||
case parseRel d of
|
instance ToJSON TarDir where
|
||||||
Right x -> pure x
|
toJSON (RealDir p) = toJSON p
|
||||||
Left e -> fail $ "Failure in HPath Rel (FromJSON)" <> show e
|
toJSON (RegexDir r) = object ["RegexDir" .= r]
|
||||||
|
|
||||||
|
instance FromJSON TarDir where
|
||||||
|
parseJSON v = realDir v <|> regexDir v
|
||||||
|
where
|
||||||
|
realDir = withText "TarDir" $ \t -> do
|
||||||
|
fp <- parseJSON (String t)
|
||||||
|
pure (RealDir fp)
|
||||||
|
regexDir = withObject "TarDir" $ \o -> do
|
||||||
|
r <- o .: "RegexDir"
|
||||||
|
pure $ RegexDir r
|
||||||
|
|
||||||
|
|
||||||
|
instance ToJSON VersionCmp where
|
||||||
|
toJSON = String . versionCmpToText
|
||||||
|
|
||||||
|
instance FromJSON VersionCmp where
|
||||||
|
parseJSON = withText "VersionCmp" $ \t -> do
|
||||||
|
case MP.parse versionCmpP "" t of
|
||||||
|
Right r -> pure r
|
||||||
|
Left e -> fail (MP.errorBundlePretty e)
|
||||||
|
|
||||||
|
versionCmpToText :: VersionCmp -> T.Text
|
||||||
|
versionCmpToText (VR_gt ver') = "> " <> prettyV ver'
|
||||||
|
versionCmpToText (VR_gteq ver') = ">= " <> prettyV ver'
|
||||||
|
versionCmpToText (VR_lt ver') = "< " <> prettyV ver'
|
||||||
|
versionCmpToText (VR_lteq ver') = "<= " <> prettyV ver'
|
||||||
|
versionCmpToText (VR_eq ver') = "== " <> prettyV ver'
|
||||||
|
|
||||||
|
versionCmpP :: MP.Parsec Void T.Text VersionCmp
|
||||||
|
versionCmpP =
|
||||||
|
fmap VR_gt (MP.try $ MPC.space *> MP.chunk ">" *> MPC.space *> versioningEnd)
|
||||||
|
<|> fmap
|
||||||
|
VR_gteq
|
||||||
|
(MP.try $ MPC.space *> MP.chunk ">=" *> MPC.space *> versioningEnd)
|
||||||
|
<|> fmap
|
||||||
|
VR_lt
|
||||||
|
(MP.try $ MPC.space *> MP.chunk "<" *> MPC.space *> versioningEnd)
|
||||||
|
<|> fmap
|
||||||
|
VR_lteq
|
||||||
|
(MP.try $ MPC.space *> MP.chunk "<=" *> MPC.space *> versioningEnd)
|
||||||
|
<|> fmap
|
||||||
|
VR_eq
|
||||||
|
(MP.try $ MPC.space *> MP.chunk "==" *> MPC.space *> versioningEnd)
|
||||||
|
<|> fmap
|
||||||
|
VR_eq
|
||||||
|
(MP.try $ MPC.space *> versioningEnd)
|
||||||
|
|
||||||
|
instance ToJSON VersionRange where
|
||||||
|
toJSON = String . verRangeToText
|
||||||
|
|
||||||
|
verRangeToText :: VersionRange -> T.Text
|
||||||
|
verRangeToText (SimpleRange cmps) =
|
||||||
|
let inner = foldr1 (\x y -> x <> " && " <> y)
|
||||||
|
(versionCmpToText <$> NE.toList cmps)
|
||||||
|
in "( " <> inner <> " )"
|
||||||
|
verRangeToText (OrRange cmps range) =
|
||||||
|
let left = verRangeToText (SimpleRange cmps)
|
||||||
|
right = verRangeToText range
|
||||||
|
in left <> " || " <> right
|
||||||
|
|
||||||
|
instance FromJSON VersionRange where
|
||||||
|
parseJSON = withText "VersionRange" $ \t -> do
|
||||||
|
case MP.parse versionRangeP "" t of
|
||||||
|
Right r -> pure r
|
||||||
|
Left e -> fail (MP.errorBundlePretty e)
|
||||||
|
|
||||||
|
versionRangeP :: MP.Parsec Void T.Text VersionRange
|
||||||
|
versionRangeP = go <* MP.eof
|
||||||
|
where
|
||||||
|
go =
|
||||||
|
MP.try orParse
|
||||||
|
<|> MP.try (fmap SimpleRange andParse)
|
||||||
|
<|> fmap (SimpleRange . pure) versionCmpP
|
||||||
|
|
||||||
|
orParse :: MP.Parsec Void T.Text VersionRange
|
||||||
|
orParse =
|
||||||
|
(\a o -> OrRange a o)
|
||||||
|
<$> (MP.try andParse <|> fmap pure versionCmpP)
|
||||||
|
<*> (MPC.space *> MP.chunk "||" *> MPC.space *> go)
|
||||||
|
|
||||||
|
andParse :: MP.Parsec Void T.Text (NonEmpty VersionCmp)
|
||||||
|
andParse =
|
||||||
|
fmap (\h t -> h :| t)
|
||||||
|
(MPC.space *> MP.chunk "(" *> MPC.space *> versionCmpP)
|
||||||
|
<*> MP.try (MP.many (MPC.space *> MP.chunk "&&" *> MPC.space *> versionCmpP))
|
||||||
|
<* MPC.space
|
||||||
|
<* MP.chunk ")"
|
||||||
|
<* MPC.space
|
||||||
|
|
||||||
|
versioningEnd :: MP.Parsec Void T.Text Versioning
|
||||||
|
versioningEnd =
|
||||||
|
MP.try (verP (MP.chunk " " <|> MP.chunk ")" <|> MP.chunk "&&") <* MPC.space)
|
||||||
|
<|> versioning'
|
||||||
|
|
||||||
|
instance ToJSONKey (Maybe VersionRange) where
|
||||||
|
toJSONKey = toJSONKeyText $ \case
|
||||||
|
Just x -> verRangeToText x
|
||||||
|
Nothing -> "unknown_versioning"
|
||||||
|
|
||||||
|
instance FromJSONKey (Maybe VersionRange) where
|
||||||
|
fromJSONKey = FromJSONKeyTextParser $ \t ->
|
||||||
|
if t == T.pack "unknown_versioning" then pure Nothing else just t
|
||||||
|
where
|
||||||
|
just t = case MP.parse versionRangeP "" t of
|
||||||
|
Right x -> pure $ Just x
|
||||||
|
Left e -> fail $ "Failure in (Maybe VersionRange) (FromJSONKey)" <> MP.errorBundlePretty e
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''Requirements
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''DownloadInfo
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''VersionInfo
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = removeLensFieldLabel } ''GHCupInfo
|
||||||
|
deriveJSON defaultOptions { sumEncoding = ObjectWithSingleField } ''URLSource
|
||||||
|
deriveJSON defaultOptions { sumEncoding = ObjectWithSingleField } ''Key
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = \str' -> maybe str' T.unpack . T.stripPrefix (T.pack "k-") . T.pack . kebab $ str' } ''UserKeyBindings
|
||||||
|
deriveJSON defaultOptions { fieldLabelModifier = \str' -> maybe str' T.unpack . T.stripPrefix (T.pack "u-") . T.pack . kebab $ str' } ''UserSettings
|
||||||
|
|||||||
@@ -1,5 +1,14 @@
|
|||||||
{-# LANGUAGE TemplateHaskell #-}
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Types.Optics
|
||||||
|
Description : GHCup optics
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
module GHCup.Types.Optics where
|
module GHCup.Types.Optics where
|
||||||
|
|
||||||
import GHCup.Types
|
import GHCup.Types
|
||||||
@@ -19,6 +28,9 @@ makeLenses ''DownloadInfo
|
|||||||
makeLenses ''Tag
|
makeLenses ''Tag
|
||||||
makeLenses ''VersionInfo
|
makeLenses ''VersionInfo
|
||||||
|
|
||||||
|
makeLenses ''GHCTargetVersion
|
||||||
|
|
||||||
|
makeLenses ''GHCupInfo
|
||||||
|
|
||||||
uriSchemeL' :: Lens' (URIRef Absolute) Scheme
|
uriSchemeL' :: Lens' (URIRef Absolute) Scheme
|
||||||
uriSchemeL' = lensVL uriSchemeL
|
uriSchemeL' = lensVL uriSchemeL
|
||||||
|
|||||||
1107
lib/GHCup/Utils.hs
1107
lib/GHCup/Utils.hs
File diff suppressed because it is too large
Load Diff
4
lib/GHCup/Utils.hs-boot
Normal file
4
lib/GHCup/Utils.hs-boot
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
module GHCup.Utils where
|
||||||
|
|
||||||
|
getLinkTarget :: FilePath -> IO FilePath
|
||||||
|
pathIsLink :: FilePath -> IO Bool
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
module GHCup.Utils.Bash
|
|
||||||
( findAssignment
|
|
||||||
, equalsAssignmentWith
|
|
||||||
, getRValue
|
|
||||||
, getAssignmentValueFor
|
|
||||||
)
|
|
||||||
where
|
|
||||||
|
|
||||||
import Control.Monad
|
|
||||||
import Data.ByteString.UTF8 ( toString )
|
|
||||||
import Data.List
|
|
||||||
import Data.Maybe
|
|
||||||
import HPath
|
|
||||||
import HPath.IO
|
|
||||||
import Language.Bash.Parse
|
|
||||||
import Language.Bash.Syntax
|
|
||||||
import Language.Bash.Word
|
|
||||||
import Prelude hiding ( readFile )
|
|
||||||
|
|
||||||
import qualified Data.ByteString.Lazy.UTF8 as UTF8
|
|
||||||
|
|
||||||
|
|
||||||
extractAssignments :: List -> [Assign]
|
|
||||||
extractAssignments (List stms) = join $ fmap getAssign $ getCommands stms
|
|
||||||
where
|
|
||||||
getCommands :: [Statement] -> [Command]
|
|
||||||
getCommands = join . fmap commands . catMaybes . fmap findPipes
|
|
||||||
where
|
|
||||||
findPipes (Statement (Last p@(Pipeline{})) Sequential) = Just p
|
|
||||||
findPipes _ = Nothing
|
|
||||||
|
|
||||||
getAssign :: Command -> [Assign]
|
|
||||||
getAssign (Command (SimpleCommand ass _) _) = ass
|
|
||||||
getAssign _ = []
|
|
||||||
|
|
||||||
|
|
||||||
-- | Find an assignment matching the predicate in the given file.
|
|
||||||
findAssignment :: Path b -> (Assign -> Bool) -> IO (Maybe Assign)
|
|
||||||
findAssignment p predicate = do
|
|
||||||
fileContents <- readFile p
|
|
||||||
-- TODO: this should accept bytestring:
|
|
||||||
-- https://github.com/knrafto/language-bash/issues/37
|
|
||||||
case parse (toString . toFilePath $ p) (UTF8.toString fileContents) of
|
|
||||||
Left e -> fail $ show e
|
|
||||||
Right l -> pure $ find predicate (extractAssignments $ l)
|
|
||||||
|
|
||||||
|
|
||||||
-- | Check that the assignment is of the form Foo= ignoring the
|
|
||||||
-- right hand-side.
|
|
||||||
equalsAssignmentWith :: String -> Assign -> Bool
|
|
||||||
equalsAssignmentWith n ass = case ass of
|
|
||||||
(Assign (Parameter name' Nothing) Equals _) -> n == name'
|
|
||||||
_ -> False
|
|
||||||
|
|
||||||
|
|
||||||
-- | This pretty-prints the right hand of an Equals assignment, removing
|
|
||||||
-- quotations. No evaluation is performed.
|
|
||||||
getRValue :: Assign -> Maybe String
|
|
||||||
getRValue ass = case ass of
|
|
||||||
(Assign (Parameter _ _) Equals (RValue w)) -> Just $ unquote w
|
|
||||||
_ -> Nothing
|
|
||||||
|
|
||||||
|
|
||||||
-- | Given a bash assignment such as Foo="Bar" in the given file,
|
|
||||||
-- will return "Bar" (without quotations).
|
|
||||||
getAssignmentValueFor :: Path b -> String -> IO (Maybe String)
|
|
||||||
getAssignmentValueFor p n = do
|
|
||||||
mass <- findAssignment p (equalsAssignmentWith n)
|
|
||||||
pure (mass >>= getRValue)
|
|
||||||
@@ -1,79 +1,281 @@
|
|||||||
|
{-# LANGUAGE CPP #-}
|
||||||
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
{-# LANGUAGE ViewPatterns #-}
|
||||||
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
|
||||||
module GHCup.Utils.Dirs where
|
{-|
|
||||||
|
Module : GHCup.Utils.Dirs
|
||||||
|
Description : Definition of GHCup directories
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
|
module GHCup.Utils.Dirs
|
||||||
|
( getDirs
|
||||||
|
, ghcupBaseDir
|
||||||
|
, ghcupConfigFile
|
||||||
|
, ghcupCacheDir
|
||||||
|
, ghcupGHCBaseDir
|
||||||
|
, ghcupGHCDir
|
||||||
|
, mkGhcupTmpDir
|
||||||
|
, parseGHCupGHCDir
|
||||||
|
, relativeSymlink
|
||||||
|
, withGHCupTmpDir
|
||||||
|
, getConfigFilePath
|
||||||
|
#if !defined(IS_WINDOWS)
|
||||||
|
, useXDG
|
||||||
|
#endif
|
||||||
|
)
|
||||||
|
where
|
||||||
|
|
||||||
|
|
||||||
|
import GHCup.Errors
|
||||||
|
import GHCup.Types
|
||||||
import GHCup.Types.JSON ( )
|
import GHCup.Types.JSON ( )
|
||||||
|
import GHCup.Utils.MegaParsec
|
||||||
import GHCup.Utils.Prelude
|
import GHCup.Utils.Prelude
|
||||||
|
|
||||||
import Control.Applicative
|
|
||||||
import Control.Exception.Safe
|
import Control.Exception.Safe
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
|
import Control.Monad.IO.Unlift
|
||||||
|
import Control.Monad.Logger
|
||||||
import Control.Monad.Reader
|
import Control.Monad.Reader
|
||||||
import Control.Monad.Trans.Resource
|
import Control.Monad.Trans.Resource hiding (throwM)
|
||||||
|
import Data.Bifunctor
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.Versions
|
import Data.String.Interpolate
|
||||||
import HPath
|
import GHC.IO.Exception ( IOErrorType(NoSuchThing) )
|
||||||
import HPath.IO
|
import Haskus.Utils.Variant.Excepts
|
||||||
import Optics
|
import Optics
|
||||||
import Prelude hiding ( abs
|
#if !defined(IS_WINDOWS)
|
||||||
, readFile
|
import System.Directory
|
||||||
, writeFile
|
#endif
|
||||||
)
|
import System.DiskSpace
|
||||||
import System.Posix.Env.ByteString ( getEnv
|
import System.Environment
|
||||||
, getEnvDefault
|
import System.FilePath
|
||||||
)
|
import System.IO.Temp
|
||||||
import System.Posix.Temp.ByteString ( mkdtemp )
|
|
||||||
|
|
||||||
import qualified Data.ByteString.UTF8 as UTF8
|
import qualified Data.ByteString as BS
|
||||||
import qualified System.Posix.FilePath as FP
|
import qualified Data.Text as T
|
||||||
import qualified System.Posix.User as PU
|
import qualified Data.Yaml as Y
|
||||||
|
import qualified Text.Megaparsec as MP
|
||||||
|
import Control.Concurrent (threadDelay)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
------------------------------
|
||||||
|
--[ GHCup base directories ]--
|
||||||
|
------------------------------
|
||||||
|
|
||||||
|
|
||||||
|
-- | ~/.ghcup by default
|
||||||
|
--
|
||||||
|
-- If 'GHCUP_USE_XDG_DIRS' is set (to anything),
|
||||||
|
-- then uses 'XDG_DATA_HOME/ghcup' as per xdg spec.
|
||||||
|
ghcupBaseDir :: IO FilePath
|
||||||
|
ghcupBaseDir = do
|
||||||
|
#if defined(IS_WINDOWS)
|
||||||
|
bdir <- fromMaybe "C:\\" <$> lookupEnv "GHCUP_INSTALL_BASE_PREFIX"
|
||||||
|
pure (bdir </> "ghcup")
|
||||||
|
#else
|
||||||
|
xdg <- useXDG
|
||||||
|
if xdg
|
||||||
|
then do
|
||||||
|
bdir <- lookupEnv "XDG_DATA_HOME" >>= \case
|
||||||
|
Just r -> pure r
|
||||||
|
Nothing -> do
|
||||||
|
home <- liftIO getHomeDirectory
|
||||||
|
pure (home </> ".local" </> "share")
|
||||||
|
pure (bdir </> "ghcup")
|
||||||
|
else do
|
||||||
|
bdir <- lookupEnv "GHCUP_INSTALL_BASE_PREFIX" >>= \case
|
||||||
|
Just r -> pure r
|
||||||
|
Nothing -> liftIO getHomeDirectory
|
||||||
|
pure (bdir </> ".ghcup")
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
-- | ~/.ghcup by default
|
||||||
|
--
|
||||||
|
-- If 'GHCUP_USE_XDG_DIRS' is set (to anything),
|
||||||
|
-- then uses 'XDG_CONFIG_HOME/ghcup' as per xdg spec.
|
||||||
|
ghcupConfigDir :: IO FilePath
|
||||||
|
ghcupConfigDir = do
|
||||||
|
#if defined(IS_WINDOWS)
|
||||||
|
ghcupBaseDir
|
||||||
|
#else
|
||||||
|
xdg <- useXDG
|
||||||
|
if xdg
|
||||||
|
then do
|
||||||
|
bdir <- lookupEnv "XDG_CONFIG_HOME" >>= \case
|
||||||
|
Just r -> pure r
|
||||||
|
Nothing -> do
|
||||||
|
home <- liftIO getHomeDirectory
|
||||||
|
pure (home </> ".config")
|
||||||
|
pure (bdir </> "ghcup")
|
||||||
|
else do
|
||||||
|
bdir <- lookupEnv "GHCUP_INSTALL_BASE_PREFIX" >>= \case
|
||||||
|
Just r -> pure r
|
||||||
|
Nothing -> liftIO getHomeDirectory
|
||||||
|
pure (bdir </> ".ghcup")
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
-- | If 'GHCUP_USE_XDG_DIRS' is set (to anything),
|
||||||
|
-- then uses 'XDG_BIN_HOME' env var or defaults to '~/.local/bin'
|
||||||
|
-- (which, sadly is not strictly xdg spec).
|
||||||
|
ghcupBinDir :: IO FilePath
|
||||||
|
ghcupBinDir = do
|
||||||
|
#if defined(IS_WINDOWS)
|
||||||
|
ghcupBaseDir <&> (</> "bin")
|
||||||
|
#else
|
||||||
|
xdg <- useXDG
|
||||||
|
if xdg
|
||||||
|
then do
|
||||||
|
lookupEnv "XDG_BIN_HOME" >>= \case
|
||||||
|
Just r -> pure r
|
||||||
|
Nothing -> do
|
||||||
|
home <- liftIO getHomeDirectory
|
||||||
|
pure (home </> ".local" </> "bin")
|
||||||
|
else ghcupBaseDir <&> (</> "bin")
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
-- | Defaults to '~/.ghcup/cache'.
|
||||||
|
--
|
||||||
|
-- If 'GHCUP_USE_XDG_DIRS' is set (to anything),
|
||||||
|
-- then uses 'XDG_CACHE_HOME/ghcup' as per xdg spec.
|
||||||
|
ghcupCacheDir :: IO FilePath
|
||||||
|
ghcupCacheDir = do
|
||||||
|
#if defined(IS_WINDOWS)
|
||||||
|
ghcupBaseDir <&> (</> "cache")
|
||||||
|
#else
|
||||||
|
xdg <- useXDG
|
||||||
|
if xdg
|
||||||
|
then do
|
||||||
|
bdir <- lookupEnv "XDG_CACHE_HOME" >>= \case
|
||||||
|
Just r -> pure r
|
||||||
|
Nothing -> do
|
||||||
|
home <- liftIO getHomeDirectory
|
||||||
|
pure (home </> ".cache")
|
||||||
|
pure (bdir </> "ghcup")
|
||||||
|
else ghcupBaseDir <&> (</> "cache")
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
-- | Defaults to '~/.ghcup/logs'.
|
||||||
|
--
|
||||||
|
-- If 'GHCUP_USE_XDG_DIRS' is set (to anything),
|
||||||
|
-- then uses 'XDG_CACHE_HOME/ghcup/logs' as per xdg spec.
|
||||||
|
ghcupLogsDir :: IO FilePath
|
||||||
|
ghcupLogsDir = do
|
||||||
|
#if defined(IS_WINDOWS)
|
||||||
|
ghcupBaseDir <&> (</> "logs")
|
||||||
|
#else
|
||||||
|
xdg <- useXDG
|
||||||
|
if xdg
|
||||||
|
then do
|
||||||
|
bdir <- lookupEnv "XDG_CACHE_HOME" >>= \case
|
||||||
|
Just r -> pure r
|
||||||
|
Nothing -> do
|
||||||
|
home <- liftIO getHomeDirectory
|
||||||
|
pure (home </> ".cache")
|
||||||
|
pure (bdir </> "ghcup" </> "logs")
|
||||||
|
else ghcupBaseDir <&> (</> "logs")
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
getDirs :: IO Dirs
|
||||||
|
getDirs = do
|
||||||
|
baseDir <- ghcupBaseDir
|
||||||
|
binDir <- ghcupBinDir
|
||||||
|
cacheDir <- ghcupCacheDir
|
||||||
|
logsDir <- ghcupLogsDir
|
||||||
|
confDir <- ghcupConfigDir
|
||||||
|
pure Dirs { .. }
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-------------------
|
||||||
|
--[ GHCup files ]--
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
getConfigFilePath :: (MonadIO m) => m FilePath
|
||||||
|
getConfigFilePath = do
|
||||||
|
confDir <- liftIO ghcupConfigDir
|
||||||
|
pure $ confDir </> "config.yaml"
|
||||||
|
|
||||||
|
ghcupConfigFile :: (MonadIO m)
|
||||||
|
=> Excepts '[JSONError] m UserSettings
|
||||||
|
ghcupConfigFile = do
|
||||||
|
filepath <- getConfigFilePath
|
||||||
|
contents <- liftIO $ handleIO' NoSuchThing (\_ -> pure Nothing) $ Just <$> BS.readFile filepath
|
||||||
|
case contents of
|
||||||
|
Nothing -> pure defaultUserSettings
|
||||||
|
Just contents' -> lE' JSONDecodeError . first show . Y.decodeEither' $ contents'
|
||||||
|
|
||||||
|
|
||||||
-------------------------
|
-------------------------
|
||||||
--[ GHCup directories ]--
|
--[ GHCup directories ]--
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
|
|
||||||
ghcupBaseDir :: IO (Path Abs)
|
-- | ~/.ghcup/ghc by default.
|
||||||
ghcupBaseDir = do
|
ghcupGHCBaseDir :: (MonadReader AppState m) => m FilePath
|
||||||
getEnv "GHCUP_INSTALL_BASE_PREFIX" >>= \case
|
ghcupGHCBaseDir = do
|
||||||
Just r -> parseAbs r
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
Nothing -> do
|
pure (baseDir </> "ghc")
|
||||||
home <- liftIO getHomeDirectory
|
|
||||||
pure (home </> [rel|.ghcup|])
|
|
||||||
|
|
||||||
ghcupGHCBaseDir :: IO (Path Abs)
|
|
||||||
ghcupGHCBaseDir = ghcupBaseDir <&> (</> [rel|ghc|])
|
|
||||||
|
|
||||||
ghcupGHCDir :: Version -> IO (Path Abs)
|
-- | Gets '~/.ghcup/ghc/<ghcupGHCDir>'.
|
||||||
|
-- The dir may be of the form
|
||||||
|
-- * armv7-unknown-linux-gnueabihf-8.8.3
|
||||||
|
-- * 8.8.4
|
||||||
|
ghcupGHCDir :: (MonadReader AppState m, MonadThrow m)
|
||||||
|
=> GHCTargetVersion
|
||||||
|
-> m FilePath
|
||||||
ghcupGHCDir ver = do
|
ghcupGHCDir ver = do
|
||||||
ghcbasedir <- ghcupGHCBaseDir
|
ghcbasedir <- ghcupGHCBaseDir
|
||||||
verdir <- parseRel (verToBS ver)
|
let verdir = T.unpack $ tVerToText ver
|
||||||
pure (ghcbasedir </> verdir)
|
pure (ghcbasedir </> verdir)
|
||||||
|
|
||||||
|
|
||||||
ghcupBinDir :: IO (Path Abs)
|
-- | See 'ghcupToolParser'.
|
||||||
ghcupBinDir = ghcupBaseDir <&> (</> [rel|bin|])
|
parseGHCupGHCDir :: MonadThrow m => FilePath -> m GHCTargetVersion
|
||||||
|
parseGHCupGHCDir (T.pack -> fp) =
|
||||||
ghcupCacheDir :: IO (Path Abs)
|
throwEither $ MP.parse ghcTargetVerP "" fp
|
||||||
ghcupCacheDir = ghcupBaseDir <&> (</> [rel|cache|])
|
|
||||||
|
|
||||||
ghcupLogsDir :: IO (Path Abs)
|
|
||||||
ghcupLogsDir = ghcupBaseDir <&> (</> [rel|logs|])
|
|
||||||
|
|
||||||
|
|
||||||
mkGhcupTmpDir :: (MonadThrow m, MonadIO m) => m (Path Abs)
|
mkGhcupTmpDir :: (MonadUnliftIO m, MonadLogger m, MonadCatch m, MonadThrow m, MonadIO m) => m FilePath
|
||||||
mkGhcupTmpDir = do
|
mkGhcupTmpDir = do
|
||||||
tmpdir <- liftIO $ getEnvDefault "TMPDIR" "/tmp"
|
tmpdir <- liftIO getCanonicalTemporaryDirectory
|
||||||
tmp <- liftIO $ mkdtemp $ (tmpdir FP.</> "ghcup-")
|
|
||||||
parseAbs tmp
|
let minSpace = 5000 -- a rough guess, aight?
|
||||||
|
space <- handleIO (\_ -> pure Nothing) $ fmap Just $ liftIO $ getAvailSpace tmpdir
|
||||||
|
when (maybe False (toBytes minSpace >) space) $ do
|
||||||
|
$(logWarn) [i|Possibly insufficient disk space on #{tmpdir}. At least #{minSpace} MB are recommended, but only #{toMB (fromJust space)} are free. Consider freeing up disk space or setting TMPDIR env variable.|]
|
||||||
|
$(logWarn)
|
||||||
|
"...waiting for 10 seconds before continuing anyway, you can still abort..."
|
||||||
|
liftIO $ threadDelay 10000000 -- give the user a sec to intervene
|
||||||
|
|
||||||
|
liftIO $ createTempDirectory tmpdir "ghcup"
|
||||||
|
where
|
||||||
|
toBytes mb = mb * 1024 * 1024
|
||||||
|
toMB b = show (truncate' (fromIntegral b / (1024 * 1024) :: Double) 2)
|
||||||
|
truncate' :: Double -> Int -> Double
|
||||||
|
truncate' x n = fromIntegral (floor (x * t) :: Integer) / t
|
||||||
|
where t = 10^n
|
||||||
|
|
||||||
|
|
||||||
|
withGHCupTmpDir :: (MonadUnliftIO m, MonadLogger m, MonadCatch m, MonadResource m, MonadThrow m, MonadIO m) => m FilePath
|
||||||
|
withGHCupTmpDir = snd <$> withRunInIO (\run -> run $ allocate (run mkGhcupTmpDir) rmPath)
|
||||||
|
|
||||||
|
|
||||||
withGHCupTmpDir :: (MonadResource m, MonadThrow m, MonadIO m) => m (Path Abs)
|
|
||||||
withGHCupTmpDir = snd <$> allocate mkGhcupTmpDir deleteDirRecursive
|
|
||||||
|
|
||||||
|
|
||||||
--------------
|
--------------
|
||||||
@@ -81,11 +283,21 @@ withGHCupTmpDir = snd <$> allocate mkGhcupTmpDir deleteDirRecursive
|
|||||||
--------------
|
--------------
|
||||||
|
|
||||||
|
|
||||||
getHomeDirectory :: IO (Path Abs)
|
#if !defined(IS_WINDOWS)
|
||||||
getHomeDirectory = do
|
useXDG :: IO Bool
|
||||||
e <- getEnv "HOME"
|
useXDG = isJust <$> lookupEnv "GHCUP_USE_XDG_DIRS"
|
||||||
case e of
|
#endif
|
||||||
Just fp -> parseAbs fp
|
|
||||||
Nothing -> do
|
|
||||||
h <- PU.homeDirectory <$> (PU.getEffectiveUserID >>= PU.getUserEntryForID)
|
relativeSymlink :: FilePath -- ^ the path in which to create the symlink
|
||||||
parseAbs $ UTF8.fromString h -- this is a guess
|
-> FilePath -- ^ the symlink destination
|
||||||
|
-> FilePath
|
||||||
|
relativeSymlink p1 p2 =
|
||||||
|
let d1 = splitDirectories p1
|
||||||
|
d2 = splitDirectories p2
|
||||||
|
common = takeWhile (\(x, y) -> x == y) $ zip d1 d2
|
||||||
|
cPrefix = drop (length common) d1
|
||||||
|
in joinPath (replicate (length cPrefix) "..")
|
||||||
|
<> joinPath ([pathSeparator] : drop (length common) d2)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,390 +1,17 @@
|
|||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE CPP #-}
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
|
||||||
{-# LANGUAGE TemplateHaskell #-}
|
|
||||||
|
|
||||||
module GHCup.Utils.File where
|
module GHCup.Utils.File (
|
||||||
|
module GHCup.Utils.File.Common,
|
||||||
|
#if IS_WINDOWS
|
||||||
|
module GHCup.Utils.File.Windows
|
||||||
|
#else
|
||||||
|
module GHCup.Utils.File.Posix
|
||||||
|
#endif
|
||||||
|
) where
|
||||||
|
|
||||||
import GHCup.Utils.Dirs
|
import GHCup.Utils.File.Common
|
||||||
import GHCup.Utils.Prelude
|
#if IS_WINDOWS
|
||||||
|
import GHCup.Utils.File.Windows
|
||||||
import Control.Concurrent
|
#else
|
||||||
import Control.Exception ( evaluate )
|
import GHCup.Utils.File.Posix
|
||||||
import Control.Exception.Safe
|
#endif
|
||||||
import Control.Monad
|
|
||||||
import Data.ByteString ( ByteString )
|
|
||||||
import Data.ByteString.Unsafe ( unsafeUseAsCStringLen )
|
|
||||||
import Data.Char
|
|
||||||
import Data.Foldable
|
|
||||||
import Data.Functor
|
|
||||||
import Data.IORef
|
|
||||||
import Data.Maybe
|
|
||||||
import GHC.Foreign ( peekCStringLen )
|
|
||||||
import GHC.IO.Encoding ( getLocaleEncoding )
|
|
||||||
import GHC.IO.Exception
|
|
||||||
import HPath
|
|
||||||
import HPath.IO
|
|
||||||
import Optics
|
|
||||||
import Streamly
|
|
||||||
import Streamly.External.ByteString
|
|
||||||
import Streamly.External.ByteString.Lazy
|
|
||||||
import System.Console.Pretty
|
|
||||||
import System.Console.Regions
|
|
||||||
import System.IO
|
|
||||||
import System.IO.Error
|
|
||||||
import System.Posix.Directory.ByteString
|
|
||||||
import System.Posix.FD as FD
|
|
||||||
import System.Posix.FilePath hiding ( (</>) )
|
|
||||||
import System.Posix.Foreign ( oExcl )
|
|
||||||
import "unix" System.Posix.IO.ByteString
|
|
||||||
hiding ( openFd )
|
|
||||||
import System.Posix.Process ( ProcessStatus(..) )
|
|
||||||
import System.Posix.Types
|
|
||||||
|
|
||||||
|
|
||||||
import qualified Control.Exception as EX
|
|
||||||
import qualified Data.Text as T
|
|
||||||
import qualified Data.Text.Encoding as E
|
|
||||||
import qualified Data.Text.Encoding.Error as E
|
|
||||||
import qualified System.Posix.Process.ByteString
|
|
||||||
as SPPB
|
|
||||||
import Streamly.External.Posix.DirStream
|
|
||||||
import qualified Streamly.Internal.Memory.ArrayStream
|
|
||||||
as AS
|
|
||||||
import qualified Streamly.FileSystem.Handle as FH
|
|
||||||
import qualified Streamly.Internal.Data.Unfold as SU
|
|
||||||
import qualified Streamly.Prelude as S
|
|
||||||
import qualified Data.ByteString as BS
|
|
||||||
import qualified Data.ByteString.Lazy as L
|
|
||||||
import qualified "unix-bytestring" System.Posix.IO.ByteString
|
|
||||||
as SPIB
|
|
||||||
|
|
||||||
|
|
||||||
data StopThread = StopThread Bool
|
|
||||||
deriving Show
|
|
||||||
|
|
||||||
instance Exception StopThread
|
|
||||||
|
|
||||||
|
|
||||||
data ProcessError = NonZeroExit Int ByteString [ByteString]
|
|
||||||
| PTerminated ByteString [ByteString]
|
|
||||||
| PStopped ByteString [ByteString]
|
|
||||||
| NoSuchPid ByteString [ByteString]
|
|
||||||
deriving Show
|
|
||||||
|
|
||||||
|
|
||||||
data CapturedProcess = CapturedProcess
|
|
||||||
{ _exitCode :: ExitCode
|
|
||||||
, _stdOut :: ByteString
|
|
||||||
, _stdErr :: ByteString
|
|
||||||
}
|
|
||||||
deriving (Eq, Show)
|
|
||||||
|
|
||||||
makeLenses ''CapturedProcess
|
|
||||||
|
|
||||||
|
|
||||||
readFd :: Fd -> IO L.ByteString
|
|
||||||
readFd fd = do
|
|
||||||
handle' <- fdToHandle fd
|
|
||||||
fromChunksIO $ (S.unfold (SU.finallyIO hClose FH.readChunks) handle')
|
|
||||||
|
|
||||||
|
|
||||||
-- | Read the lines of a file into a stream. The stream holds
|
|
||||||
-- a file handle as a resource and will close it once the stream
|
|
||||||
-- terminates (either through exception or because it's drained).
|
|
||||||
readFileLines :: Path b -> IO (SerialT IO ByteString)
|
|
||||||
readFileLines p = do
|
|
||||||
stream <- readFileStream p
|
|
||||||
pure
|
|
||||||
. (fmap fromArray)
|
|
||||||
. AS.splitOn (fromIntegral $ ord '\n')
|
|
||||||
. (fmap toArray)
|
|
||||||
$ stream
|
|
||||||
|
|
||||||
|
|
||||||
-- | Find the given executable by searching all *absolute* PATH components.
|
|
||||||
-- Relative paths in PATH are ignored.
|
|
||||||
--
|
|
||||||
-- This shouldn't throw IO exceptions, unless getting the environment variable
|
|
||||||
-- PATH does.
|
|
||||||
findExecutable :: Path Rel -> IO (Maybe (Path Abs))
|
|
||||||
findExecutable ex = do
|
|
||||||
sPaths <- fmap catMaybes . (fmap . fmap) parseAbs $ getSearchPath
|
|
||||||
-- We don't want exceptions to mess up our result. If we can't
|
|
||||||
-- figure out if a file exists, then treat it as a negative result.
|
|
||||||
asum $ fmap (handleIO (\_ -> pure Nothing)) $ fmap
|
|
||||||
-- asum for short-circuiting behavior
|
|
||||||
(\s' -> (isExecutable (s' </> ex) >>= guard) $> (Just (s' </> ex)))
|
|
||||||
sPaths
|
|
||||||
|
|
||||||
|
|
||||||
-- | Execute the given command and collect the stdout, stderr and the exit code.
|
|
||||||
-- The command is run in a subprocess.
|
|
||||||
executeOut :: Path b -- ^ command as filename, e.g. 'ls'
|
|
||||||
-> [ByteString] -- ^ arguments to the command
|
|
||||||
-> Maybe (Path Abs) -- ^ chdir to this path
|
|
||||||
-> IO CapturedProcess
|
|
||||||
executeOut path args chdir = captureOutStreams $ do
|
|
||||||
maybe (pure ()) (changeWorkingDirectory . toFilePath) chdir
|
|
||||||
SPPB.executeFile (toFilePath path) True args Nothing
|
|
||||||
|
|
||||||
|
|
||||||
execLogged :: ByteString -- ^ thing to execute
|
|
||||||
-> Bool -- ^ whether to search PATH for the thing
|
|
||||||
-> [ByteString] -- ^ args for the thing
|
|
||||||
-> Path Rel -- ^ log filename
|
|
||||||
-> Maybe (Path Abs) -- ^ optionally chdir into this
|
|
||||||
-> Maybe [(ByteString, ByteString)] -- ^ optional environment
|
|
||||||
-> IO (Either ProcessError ())
|
|
||||||
execLogged exe spath args lfile chdir env = do
|
|
||||||
ldir <- ghcupLogsDir
|
|
||||||
logfile <- (ldir </>) <$> parseRel (toFilePath lfile <> ".log")
|
|
||||||
bracket (createFile (toFilePath logfile) newFilePerms) closeFd action
|
|
||||||
where
|
|
||||||
action fd = do
|
|
||||||
actionWithPipes $ \(stdoutRead, stdoutWrite) -> do
|
|
||||||
-- start the thread that logs to stdout in a region
|
|
||||||
done <- newEmptyMVar
|
|
||||||
tid <-
|
|
||||||
forkIO
|
|
||||||
$ EX.handle (\(_ :: StopThread) -> pure ())
|
|
||||||
$ EX.handle (\(_ :: IOException) -> pure ())
|
|
||||||
$ flip finally (putMVar done ())
|
|
||||||
$ printToRegion fd stdoutRead 6
|
|
||||||
|
|
||||||
-- fork our subprocess
|
|
||||||
pid <- SPPB.forkProcess $ do
|
|
||||||
void $ dupTo stdoutWrite stdOutput
|
|
||||||
void $ dupTo stdoutWrite stdError
|
|
||||||
closeFd stdoutWrite
|
|
||||||
closeFd stdoutRead
|
|
||||||
|
|
||||||
-- execute the action
|
|
||||||
maybe (pure ()) (changeWorkingDirectory . toFilePath) chdir
|
|
||||||
SPPB.executeFile exe spath args env
|
|
||||||
|
|
||||||
closeFd stdoutWrite
|
|
||||||
|
|
||||||
-- wait for the subprocess to finish
|
|
||||||
e <- SPPB.getProcessStatus True True pid >>= \case
|
|
||||||
i@(Just (SPPB.Exited _)) -> pure $ toProcessError exe args i
|
|
||||||
i -> pure $ toProcessError exe args i
|
|
||||||
|
|
||||||
-- make sure the logging thread stops
|
|
||||||
case e of
|
|
||||||
Left _ -> EX.throwTo tid (StopThread False)
|
|
||||||
Right _ -> EX.throwTo tid (StopThread True)
|
|
||||||
takeMVar done
|
|
||||||
|
|
||||||
closeFd stdoutRead
|
|
||||||
pure e
|
|
||||||
|
|
||||||
-- Reads fdIn and logs the output in a continous scrolling area
|
|
||||||
-- of 'size' terminal lines. Also writes to a log file.
|
|
||||||
printToRegion fileFd fdIn size = do
|
|
||||||
ref <- newIORef ([] :: [ByteString])
|
|
||||||
displayConsoleRegions $ do
|
|
||||||
rs <- sequence . replicate size . openConsoleRegion $ Linear
|
|
||||||
flip finally (readTilEOF (lineAction ref rs) fdIn) -- make sure the last few lines don't get cut off
|
|
||||||
$ handle
|
|
||||||
(\(StopThread b) -> do
|
|
||||||
when b (forM_ rs closeConsoleRegion)
|
|
||||||
EX.throw (StopThread b)
|
|
||||||
)
|
|
||||||
$ readForever (lineAction ref rs) fdIn
|
|
||||||
|
|
||||||
where
|
|
||||||
-- action to perform line by line
|
|
||||||
lineAction ref rs bs' = do
|
|
||||||
modifyIORef' ref (swapRegs bs')
|
|
||||||
regs <- readIORef ref
|
|
||||||
forM (zip regs rs) $ \(bs, r) -> do
|
|
||||||
setConsoleRegion r $ do
|
|
||||||
w <- consoleWidth
|
|
||||||
return
|
|
||||||
. T.pack
|
|
||||||
. color Blue
|
|
||||||
. T.unpack
|
|
||||||
. E.decodeUtf8With E.lenientDecode
|
|
||||||
. trim w
|
|
||||||
. (\b -> "[ " <> toFilePath lfile <> " ] " <> b)
|
|
||||||
$ bs
|
|
||||||
SPIB.fdWrite fileFd (bs <> "\n")
|
|
||||||
|
|
||||||
|
|
||||||
swapRegs bs regs | length regs < size = regs ++ [bs]
|
|
||||||
| otherwise = tail regs ++ [bs]
|
|
||||||
|
|
||||||
-- trim output line to terminal width
|
|
||||||
trim w bs | BS.length bs > w && w > 5 = BS.take (w - 4) bs <> "..."
|
|
||||||
| otherwise = bs
|
|
||||||
|
|
||||||
-- read an entire line from the file descriptor (removes the newline char)
|
|
||||||
readLine fd' = do
|
|
||||||
bs <-
|
|
||||||
handle
|
|
||||||
(\(e :: IOError) -> do
|
|
||||||
if isEOFError e then threadDelay 1000 >> pure "" else throw e
|
|
||||||
)
|
|
||||||
$ SPIB.fdRead fd' 1
|
|
||||||
if
|
|
||||||
| bs == "\n" -> pure ""
|
|
||||||
| bs == "" -> pure ""
|
|
||||||
| otherwise -> fmap (bs <>) $ readLine fd'
|
|
||||||
|
|
||||||
readForever action' fd' = do
|
|
||||||
bs <- readLine fd'
|
|
||||||
if not $ BS.null bs
|
|
||||||
then action' bs >> readForever action' fd'
|
|
||||||
else readForever action' fd'
|
|
||||||
|
|
||||||
readTilEOF action' fd' = do
|
|
||||||
bs <- readLine fd'
|
|
||||||
when (not $ BS.null bs) (action' bs >> readTilEOF action' fd')
|
|
||||||
|
|
||||||
|
|
||||||
-- | Capture the stdout and stderr of the given action, which
|
|
||||||
-- is run in a subprocess. Stdin is closed. You might want to
|
|
||||||
-- 'race' this to make sure it terminates.
|
|
||||||
captureOutStreams :: IO a
|
|
||||||
-- ^ the action to execute in a subprocess
|
|
||||||
-> IO CapturedProcess
|
|
||||||
captureOutStreams action = do
|
|
||||||
actionWithPipes $ \(parentStdoutRead, childStdoutWrite) ->
|
|
||||||
actionWithPipes $ \(parentStderrRead, childStderrWrite) -> do
|
|
||||||
pid <- SPPB.forkProcess $ do
|
|
||||||
-- dup stdout
|
|
||||||
void $ dupTo childStdoutWrite stdOutput
|
|
||||||
closeFd childStdoutWrite
|
|
||||||
closeFd parentStdoutRead
|
|
||||||
|
|
||||||
-- dup stderr
|
|
||||||
void $ dupTo childStderrWrite stdError
|
|
||||||
closeFd childStderrWrite
|
|
||||||
closeFd parentStderrRead
|
|
||||||
|
|
||||||
-- execute the action
|
|
||||||
a <- action
|
|
||||||
void $ evaluate a
|
|
||||||
|
|
||||||
-- close everything we don't need
|
|
||||||
closeFd childStdoutWrite
|
|
||||||
closeFd childStderrWrite
|
|
||||||
|
|
||||||
-- start thread that writes the output
|
|
||||||
refOut <- newIORef BS.empty
|
|
||||||
refErr <- newIORef BS.empty
|
|
||||||
done <- newEmptyMVar
|
|
||||||
_ <-
|
|
||||||
forkIO
|
|
||||||
$ EX.handle (\(_ :: StopThread) -> pure ())
|
|
||||||
$ EX.handle (\(_ :: IOException) -> pure ())
|
|
||||||
$ flip finally (putMVar done ())
|
|
||||||
$ writeStds parentStdoutRead parentStderrRead refOut refErr
|
|
||||||
|
|
||||||
status <- SPPB.getProcessStatus True True pid
|
|
||||||
takeMVar done
|
|
||||||
|
|
||||||
case status of
|
|
||||||
-- readFd will take care of closing the fd
|
|
||||||
Just (SPPB.Exited es) -> do
|
|
||||||
stdout' <- readIORef refOut
|
|
||||||
stderr' <- readIORef refErr
|
|
||||||
pure $ CapturedProcess { _exitCode = es
|
|
||||||
, _stdOut = stdout'
|
|
||||||
, _stdErr = stderr'
|
|
||||||
}
|
|
||||||
|
|
||||||
_ -> throwIO $ userError $ ("No such PID " ++ show pid)
|
|
||||||
|
|
||||||
where
|
|
||||||
writeStds pout perr rout rerr = do
|
|
||||||
doneOut <- newEmptyMVar
|
|
||||||
void
|
|
||||||
$ forkIO
|
|
||||||
$ EX.handle (\(_ :: IOException) -> pure ())
|
|
||||||
$ flip finally (putMVar doneOut ())
|
|
||||||
$ readTilEOF (\x -> modifyIORef' rout (<> x)) pout
|
|
||||||
doneErr <- newEmptyMVar
|
|
||||||
void
|
|
||||||
$ forkIO
|
|
||||||
$ EX.handle (\(_ :: IOException) -> pure ())
|
|
||||||
$ flip finally (putMVar doneErr ())
|
|
||||||
$ readTilEOF (\x -> modifyIORef' rerr (<> x)) perr
|
|
||||||
takeMVar doneOut
|
|
||||||
takeMVar doneErr
|
|
||||||
|
|
||||||
readTilEOF action' fd' = do
|
|
||||||
bs <- SPIB.fdRead fd' 512
|
|
||||||
when (not $ BS.null bs) (action' bs >> readTilEOF action' fd')
|
|
||||||
|
|
||||||
|
|
||||||
actionWithPipes :: ((Fd, Fd) -> IO b) -> IO b
|
|
||||||
actionWithPipes a =
|
|
||||||
createPipe >>= \(p1, p2) -> (flip finally) (cleanup [p1, p2]) $ a (p1, p2)
|
|
||||||
|
|
||||||
cleanup :: [Fd] -> IO ()
|
|
||||||
cleanup fds = for_ fds $ \fd -> handleIO (\_ -> pure ()) $ closeFd fd
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
-- | Create a new regular file in write-only mode. The file must not exist.
|
|
||||||
createRegularFileFd :: FileMode -> Path b -> IO Fd
|
|
||||||
createRegularFileFd fm dest =
|
|
||||||
FD.openFd (toFilePath dest) WriteOnly [oExcl] (Just fm)
|
|
||||||
|
|
||||||
|
|
||||||
-- | Thin wrapper around `executeFile`.
|
|
||||||
exec :: ByteString -- ^ thing to execute
|
|
||||||
-> Bool -- ^ whether to search PATH for the thing
|
|
||||||
-> [ByteString] -- ^ args for the thing
|
|
||||||
-> Maybe (Path Abs) -- ^ optionally chdir into this
|
|
||||||
-> Maybe [(ByteString, ByteString)] -- ^ optional environment
|
|
||||||
-> IO (Either ProcessError ())
|
|
||||||
exec exe spath args chdir env = do
|
|
||||||
pid <- SPPB.forkProcess $ do
|
|
||||||
maybe (pure ()) (changeWorkingDirectory . toFilePath) chdir
|
|
||||||
SPPB.executeFile exe spath args env
|
|
||||||
|
|
||||||
fmap (toProcessError exe args) $ SPPB.getProcessStatus True True pid
|
|
||||||
|
|
||||||
|
|
||||||
toProcessError :: ByteString
|
|
||||||
-> [ByteString]
|
|
||||||
-> Maybe ProcessStatus
|
|
||||||
-> Either ProcessError ()
|
|
||||||
toProcessError exe args mps = case mps of
|
|
||||||
Just (SPPB.Exited (ExitFailure i)) -> Left $ NonZeroExit i exe args
|
|
||||||
Just (SPPB.Exited ExitSuccess ) -> Right ()
|
|
||||||
Just (Terminated _ _ ) -> Left $ PTerminated exe args
|
|
||||||
Just (Stopped _ ) -> Left $ PStopped exe args
|
|
||||||
Nothing -> Left $ NoSuchPid exe args
|
|
||||||
|
|
||||||
|
|
||||||
-- | Convert the String to a ByteString with the current
|
|
||||||
-- system encoding.
|
|
||||||
unsafePathToString :: Path b -> IO FilePath
|
|
||||||
unsafePathToString p = do
|
|
||||||
enc <- getLocaleEncoding
|
|
||||||
unsafeUseAsCStringLen (toFilePath p) (peekCStringLen enc)
|
|
||||||
|
|
||||||
|
|
||||||
-- | Search for a file in the search paths.
|
|
||||||
--
|
|
||||||
-- Catches `PermissionDenied` and `NoSuchThing` and returns `Nothing`.
|
|
||||||
searchPath :: [Path Abs] -> Path Rel -> IO (Maybe (Path Abs))
|
|
||||||
searchPath paths needle = go paths
|
|
||||||
where
|
|
||||||
go [] = pure Nothing
|
|
||||||
go (x : xs) =
|
|
||||||
hideErrorDefM PermissionDenied (go xs)
|
|
||||||
$ hideErrorDefM NoSuchThing (go xs)
|
|
||||||
$ do
|
|
||||||
dirStream <- openDirStream (toFilePath x)
|
|
||||||
S.findM (\(_, p) -> isMatch x p) (dirContentsStream dirStream)
|
|
||||||
>>= \case
|
|
||||||
Just _ -> pure $ Just (x </> needle)
|
|
||||||
Nothing -> go xs
|
|
||||||
isMatch basedir p = do
|
|
||||||
if p == toFilePath needle
|
|
||||||
then isExecutable (basedir </> needle)
|
|
||||||
else pure False
|
|
||||||
|
|||||||
106
lib/GHCup/Utils/File/Common.hs
Normal file
106
lib/GHCup/Utils/File/Common.hs
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
{-# LANGUAGE ViewPatterns #-}
|
||||||
|
|
||||||
|
module GHCup.Utils.File.Common where
|
||||||
|
|
||||||
|
import GHCup.Utils.Prelude
|
||||||
|
|
||||||
|
import Control.Monad.Extra
|
||||||
|
import Control.Monad.Reader
|
||||||
|
import Data.Maybe
|
||||||
|
import Data.String.Interpolate
|
||||||
|
import GHC.IO.Exception
|
||||||
|
import Optics hiding ((<|), (|>))
|
||||||
|
import System.Directory
|
||||||
|
import System.FilePath
|
||||||
|
import Text.PrettyPrint.HughesPJClass hiding ( (<>) )
|
||||||
|
import Text.Regex.Posix
|
||||||
|
|
||||||
|
import qualified Data.ByteString.Lazy as BL
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
data ProcessError = NonZeroExit Int FilePath [String]
|
||||||
|
| PTerminated FilePath [String]
|
||||||
|
| PStopped FilePath [String]
|
||||||
|
| NoSuchPid FilePath [String]
|
||||||
|
deriving Show
|
||||||
|
|
||||||
|
instance Pretty ProcessError where
|
||||||
|
pPrint (NonZeroExit e exe args) =
|
||||||
|
text [i|Process "#{exe}" with arguments #{args} failed with exit code #{e}.|]
|
||||||
|
pPrint (PTerminated exe args) =
|
||||||
|
text [i|Process "#{exe}" with arguments #{args} terminated.|]
|
||||||
|
pPrint (PStopped exe args) =
|
||||||
|
text [i|Process "#{exe}" with arguments #{args} stopped.|]
|
||||||
|
pPrint (NoSuchPid exe args) =
|
||||||
|
text [i|Could not find PID for process running "#{exe}" with arguments #{args}.|]
|
||||||
|
|
||||||
|
data CapturedProcess = CapturedProcess
|
||||||
|
{ _exitCode :: ExitCode
|
||||||
|
, _stdOut :: BL.ByteString
|
||||||
|
, _stdErr :: BL.ByteString
|
||||||
|
}
|
||||||
|
deriving (Eq, Show)
|
||||||
|
|
||||||
|
makeLenses ''CapturedProcess
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-- | Search for a file in the search paths.
|
||||||
|
--
|
||||||
|
-- Catches `PermissionDenied` and `NoSuchThing` and returns `Nothing`.
|
||||||
|
searchPath :: [FilePath] -> FilePath -> IO (Maybe FilePath)
|
||||||
|
searchPath paths needle = go paths
|
||||||
|
where
|
||||||
|
go [] = pure Nothing
|
||||||
|
go (x : xs) =
|
||||||
|
hideErrorDefM [InappropriateType, PermissionDenied, NoSuchThing] (go xs)
|
||||||
|
$ do
|
||||||
|
contents <- listDirectory x
|
||||||
|
findM (isMatch x) contents >>= \case
|
||||||
|
Just _ -> pure $ Just (x </> needle)
|
||||||
|
Nothing -> go xs
|
||||||
|
isMatch basedir p = do
|
||||||
|
if p == needle
|
||||||
|
then isExecutable (basedir </> needle)
|
||||||
|
else pure False
|
||||||
|
|
||||||
|
isExecutable :: FilePath -> IO Bool
|
||||||
|
isExecutable file = executable <$> getPermissions file
|
||||||
|
|
||||||
|
|
||||||
|
-- | Check wether a binary is shadowed by another one that comes before
|
||||||
|
-- it in PATH. Returns the path to said binary, if any.
|
||||||
|
isShadowed :: FilePath -> IO (Maybe FilePath)
|
||||||
|
isShadowed p = do
|
||||||
|
let dir = takeDirectory p
|
||||||
|
let fn = takeFileName p
|
||||||
|
spaths <- liftIO getSearchPath
|
||||||
|
if dir `elem` spaths
|
||||||
|
then do
|
||||||
|
let shadowPaths = takeWhile (/= dir) spaths
|
||||||
|
searchPath shadowPaths fn
|
||||||
|
else pure Nothing
|
||||||
|
|
||||||
|
|
||||||
|
-- | Check whether the binary is in PATH. This returns only `True`
|
||||||
|
-- if the directory containing the binary is part of PATH.
|
||||||
|
isInPath :: FilePath -> IO Bool
|
||||||
|
isInPath p = do
|
||||||
|
let dir = takeDirectory p
|
||||||
|
let fn = takeFileName p
|
||||||
|
spaths <- liftIO getSearchPath
|
||||||
|
if dir `elem` spaths
|
||||||
|
then isJust <$> searchPath [dir] fn
|
||||||
|
else pure False
|
||||||
|
|
||||||
|
|
||||||
|
findFiles :: FilePath -> Regex -> IO [FilePath]
|
||||||
|
findFiles path regex = do
|
||||||
|
contents <- listDirectory path
|
||||||
|
pure $ filter (match regex) contents
|
||||||
|
|
||||||
386
lib/GHCup/Utils/File/Posix.hs
Normal file
386
lib/GHCup/Utils/File/Posix.hs
Normal file
@@ -0,0 +1,386 @@
|
|||||||
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
{-# LANGUAGE ViewPatterns #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Utils.File.Posix
|
||||||
|
Description : File and unix APIs
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : POSIX
|
||||||
|
|
||||||
|
This module handles file and executable handling.
|
||||||
|
Some of these functions use sophisticated logging.
|
||||||
|
-}
|
||||||
|
module GHCup.Utils.File.Posix where
|
||||||
|
|
||||||
|
import GHCup.Utils.File.Common
|
||||||
|
import GHCup.Utils.Prelude
|
||||||
|
import GHCup.Types
|
||||||
|
|
||||||
|
import Control.Concurrent
|
||||||
|
import Control.Concurrent.Async
|
||||||
|
import Control.Exception ( evaluate )
|
||||||
|
import Control.Exception.Safe
|
||||||
|
import Control.Monad
|
||||||
|
import Control.Monad.Logger
|
||||||
|
import Control.Monad.Reader
|
||||||
|
import Control.Monad.Trans.State.Strict
|
||||||
|
import Data.ByteString ( ByteString )
|
||||||
|
import Data.Foldable
|
||||||
|
import Data.IORef
|
||||||
|
import Data.Sequence ( Seq, (|>) )
|
||||||
|
import Data.String.Interpolate
|
||||||
|
import Data.List
|
||||||
|
import Data.Word8
|
||||||
|
import GHC.IO.Exception
|
||||||
|
import System.Console.Pretty hiding ( Pretty )
|
||||||
|
import System.Console.Regions
|
||||||
|
import System.IO.Error
|
||||||
|
import System.FilePath
|
||||||
|
import System.Directory
|
||||||
|
import System.Posix.Directory
|
||||||
|
import System.Posix.Files
|
||||||
|
import System.Posix.IO
|
||||||
|
import System.Posix.Process ( ProcessStatus(..) )
|
||||||
|
import System.Posix.Types
|
||||||
|
|
||||||
|
|
||||||
|
import qualified Control.Exception as EX
|
||||||
|
import qualified Data.Sequence as Sq
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Data.Text.Encoding as E
|
||||||
|
import qualified System.Posix.Process as SPP
|
||||||
|
import qualified Data.ByteString as BS
|
||||||
|
import qualified Data.ByteString.Lazy as BL
|
||||||
|
import qualified "unix-bytestring" System.Posix.IO.ByteString
|
||||||
|
as SPIB
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-- | Execute the given command and collect the stdout, stderr and the exit code.
|
||||||
|
-- The command is run in a subprocess.
|
||||||
|
executeOut :: MonadIO m
|
||||||
|
=> FilePath -- ^ command as filename, e.g. 'ls'
|
||||||
|
-> [String] -- ^ arguments to the command
|
||||||
|
-> Maybe FilePath -- ^ chdir to this path
|
||||||
|
-> m CapturedProcess
|
||||||
|
executeOut path args chdir = liftIO $ captureOutStreams $ do
|
||||||
|
maybe (pure ()) changeWorkingDirectory chdir
|
||||||
|
SPP.executeFile path True args Nothing
|
||||||
|
|
||||||
|
|
||||||
|
execLogged :: (MonadReader AppState m, MonadIO m, MonadThrow m)
|
||||||
|
=> FilePath -- ^ thing to execute
|
||||||
|
-> [String] -- ^ args for the thing
|
||||||
|
-> Maybe FilePath -- ^ optionally chdir into this
|
||||||
|
-> FilePath -- ^ log filename (opened in append mode)
|
||||||
|
-> Maybe [(String, String)] -- ^ optional environment
|
||||||
|
-> m (Either ProcessError ())
|
||||||
|
execLogged exe args chdir lfile env = do
|
||||||
|
AppState { settings = Settings {..}, dirs = Dirs {..} } <- ask
|
||||||
|
let logfile = logsDir </> lfile <> ".log"
|
||||||
|
liftIO $ bracket (openFd logfile WriteOnly (Just newFilePerms) defaultFileFlags{ append = True })
|
||||||
|
closeFd
|
||||||
|
(action verbose)
|
||||||
|
where
|
||||||
|
action verbose fd = do
|
||||||
|
actionWithPipes $ \(stdoutRead, stdoutWrite) -> do
|
||||||
|
-- start the thread that logs to stdout
|
||||||
|
pState <- newEmptyMVar
|
||||||
|
done <- newEmptyMVar
|
||||||
|
void
|
||||||
|
$ forkIO
|
||||||
|
$ EX.handle (\(_ :: IOException) -> pure ())
|
||||||
|
$ EX.finally
|
||||||
|
(if verbose
|
||||||
|
then tee fd stdoutRead
|
||||||
|
else printToRegion fd stdoutRead 6 pState
|
||||||
|
)
|
||||||
|
(putMVar done ())
|
||||||
|
|
||||||
|
-- fork the subprocess
|
||||||
|
pid <- SPP.forkProcess $ do
|
||||||
|
void $ dupTo stdoutWrite stdOutput
|
||||||
|
void $ dupTo stdoutWrite stdError
|
||||||
|
closeFd stdoutRead
|
||||||
|
closeFd stdoutWrite
|
||||||
|
|
||||||
|
-- execute the action
|
||||||
|
maybe (pure ()) changeWorkingDirectory chdir
|
||||||
|
void $ SPP.executeFile exe (not ("./" `isPrefixOf` exe)) args env
|
||||||
|
|
||||||
|
closeFd stdoutWrite
|
||||||
|
|
||||||
|
-- wait for the subprocess to finish
|
||||||
|
e <- toProcessError exe args <$!> SPP.getProcessStatus True True pid
|
||||||
|
putMVar pState (either (const False) (const True) e)
|
||||||
|
|
||||||
|
void $ race (takeMVar done) (threadDelay (1000000 * 3))
|
||||||
|
closeFd stdoutRead
|
||||||
|
|
||||||
|
pure e
|
||||||
|
|
||||||
|
tee :: Fd -> Fd -> IO ()
|
||||||
|
tee fileFd fdIn = readTilEOF lineAction fdIn
|
||||||
|
|
||||||
|
where
|
||||||
|
lineAction :: ByteString -> IO ()
|
||||||
|
lineAction bs' = do
|
||||||
|
void $ SPIB.fdWrite fileFd (bs' <> "\n")
|
||||||
|
void $ SPIB.fdWrite stdOutput (bs' <> "\n")
|
||||||
|
|
||||||
|
-- Reads fdIn and logs the output in a continous scrolling area
|
||||||
|
-- of 'size' terminal lines. Also writes to a log file.
|
||||||
|
printToRegion :: Fd -> Fd -> Int -> MVar Bool -> IO ()
|
||||||
|
printToRegion fileFd fdIn size pState = do
|
||||||
|
void $ displayConsoleRegions $ do
|
||||||
|
rs <-
|
||||||
|
liftIO
|
||||||
|
. fmap Sq.fromList
|
||||||
|
. sequence
|
||||||
|
. replicate size
|
||||||
|
. openConsoleRegion
|
||||||
|
$ Linear
|
||||||
|
flip runStateT mempty
|
||||||
|
$ handle
|
||||||
|
(\(ex :: SomeException) -> do
|
||||||
|
ps <- liftIO $ takeMVar pState
|
||||||
|
when ps (forM_ rs (liftIO . closeConsoleRegion))
|
||||||
|
throw ex
|
||||||
|
)
|
||||||
|
$ readTilEOF (lineAction rs) fdIn
|
||||||
|
|
||||||
|
where
|
||||||
|
-- action to perform line by line
|
||||||
|
-- TODO: do this with vty for efficiency
|
||||||
|
lineAction :: (MonadMask m, MonadIO m)
|
||||||
|
=> Seq ConsoleRegion
|
||||||
|
-> ByteString
|
||||||
|
-> StateT (Seq ByteString) m ()
|
||||||
|
lineAction rs = \bs' -> do
|
||||||
|
void $ liftIO $ SPIB.fdWrite fileFd (bs' <> "\n")
|
||||||
|
modify (swapRegs bs')
|
||||||
|
regs <- get
|
||||||
|
liftIO $ forM_ (Sq.zip regs rs) $ \(bs, r) -> setConsoleRegion r $ do
|
||||||
|
w <- consoleWidth
|
||||||
|
return
|
||||||
|
. T.pack
|
||||||
|
. color Blue
|
||||||
|
. T.unpack
|
||||||
|
. decUTF8Safe
|
||||||
|
. trim w
|
||||||
|
. (\b -> "[ " <> E.encodeUtf8 (T.pack lfile) <> " ] " <> b)
|
||||||
|
$ bs
|
||||||
|
|
||||||
|
swapRegs :: a -> Seq a -> Seq a
|
||||||
|
swapRegs bs = \regs -> if
|
||||||
|
| Sq.length regs < size -> regs |> bs
|
||||||
|
| otherwise -> Sq.drop 1 regs |> bs
|
||||||
|
|
||||||
|
-- trim output line to terminal width
|
||||||
|
trim :: Int -> ByteString -> ByteString
|
||||||
|
trim w = \bs -> if
|
||||||
|
| BS.length bs > w && w > 5 -> BS.take (w - 4) bs <> "..."
|
||||||
|
| otherwise -> bs
|
||||||
|
|
||||||
|
-- Consecutively read from Fd in 512 chunks until we hit
|
||||||
|
-- newline or EOF.
|
||||||
|
readLine :: MonadIO m
|
||||||
|
=> Fd -- ^ input file descriptor
|
||||||
|
-> ByteString -- ^ rest buffer (read across newline)
|
||||||
|
-> m (ByteString, ByteString, Bool) -- ^ (full line, rest, eof)
|
||||||
|
readLine fd = go
|
||||||
|
where
|
||||||
|
go inBs = do
|
||||||
|
-- if buffer is not empty, process it first
|
||||||
|
mbs <- if BS.length inBs == 0
|
||||||
|
-- otherwise attempt read
|
||||||
|
then liftIO
|
||||||
|
$ handleIO (\e -> if isEOFError e then pure Nothing else ioError e)
|
||||||
|
$ fmap Just
|
||||||
|
$ SPIB.fdRead fd 512
|
||||||
|
else pure $ Just inBs
|
||||||
|
case mbs of
|
||||||
|
Nothing -> pure ("", "", True)
|
||||||
|
Just bs -> do
|
||||||
|
-- split on newline
|
||||||
|
let (line, rest) = BS.span (/= _lf) bs
|
||||||
|
if
|
||||||
|
| BS.length rest /= 0 -> pure (line, BS.tail rest, False)
|
||||||
|
-- if rest is empty, then there was no newline, process further
|
||||||
|
| otherwise -> (\(l, r, b) -> (line <> l, r, b)) <$!> go mempty
|
||||||
|
|
||||||
|
readTilEOF :: MonadIO m => (ByteString -> m a) -> Fd -> m ()
|
||||||
|
readTilEOF ~action' fd' = go mempty
|
||||||
|
where
|
||||||
|
go bs' = do
|
||||||
|
(bs, rest, eof) <- readLine fd' bs'
|
||||||
|
if eof
|
||||||
|
then liftIO $ ioError (mkIOError eofErrorType "" Nothing Nothing)
|
||||||
|
else void (action' bs) >> go rest
|
||||||
|
|
||||||
|
|
||||||
|
-- | Capture the stdout and stderr of the given action, which
|
||||||
|
-- is run in a subprocess. Stdin is closed. You might want to
|
||||||
|
-- 'race' this to make sure it terminates.
|
||||||
|
captureOutStreams :: IO a
|
||||||
|
-- ^ the action to execute in a subprocess
|
||||||
|
-> IO CapturedProcess
|
||||||
|
captureOutStreams action = do
|
||||||
|
actionWithPipes $ \(parentStdoutRead, childStdoutWrite) ->
|
||||||
|
actionWithPipes $ \(parentStderrRead, childStderrWrite) -> do
|
||||||
|
pid <- SPP.forkProcess $ do
|
||||||
|
-- dup stdout
|
||||||
|
void $ dupTo childStdoutWrite stdOutput
|
||||||
|
closeFd childStdoutWrite
|
||||||
|
closeFd parentStdoutRead
|
||||||
|
|
||||||
|
-- dup stderr
|
||||||
|
void $ dupTo childStderrWrite stdError
|
||||||
|
closeFd childStderrWrite
|
||||||
|
closeFd parentStderrRead
|
||||||
|
|
||||||
|
-- execute the action
|
||||||
|
a <- action
|
||||||
|
void $ evaluate a
|
||||||
|
|
||||||
|
-- close everything we don't need
|
||||||
|
closeFd childStdoutWrite
|
||||||
|
closeFd childStderrWrite
|
||||||
|
|
||||||
|
-- start thread that writes the output
|
||||||
|
refOut <- newIORef BL.empty
|
||||||
|
refErr <- newIORef BL.empty
|
||||||
|
done <- newEmptyMVar
|
||||||
|
_ <-
|
||||||
|
forkIO
|
||||||
|
$ EX.handle (\(_ :: IOException) -> pure ())
|
||||||
|
$ flip EX.finally (putMVar done ())
|
||||||
|
$ writeStds parentStdoutRead parentStderrRead refOut refErr
|
||||||
|
|
||||||
|
status <- SPP.getProcessStatus True True pid
|
||||||
|
void $ race (takeMVar done) (threadDelay (1000000 * 3))
|
||||||
|
|
||||||
|
case status of
|
||||||
|
-- readFd will take care of closing the fd
|
||||||
|
Just (SPP.Exited es) -> do
|
||||||
|
stdout' <- readIORef refOut
|
||||||
|
stderr' <- readIORef refErr
|
||||||
|
pure $ CapturedProcess { _exitCode = es
|
||||||
|
, _stdOut = stdout'
|
||||||
|
, _stdErr = stderr'
|
||||||
|
}
|
||||||
|
|
||||||
|
_ -> throwIO $ userError ("No such PID " ++ show pid)
|
||||||
|
|
||||||
|
where
|
||||||
|
writeStds :: Fd -> Fd -> IORef BL.ByteString -> IORef BL.ByteString -> IO ()
|
||||||
|
writeStds pout perr rout rerr = do
|
||||||
|
doneOut <- newEmptyMVar
|
||||||
|
void
|
||||||
|
$ forkIO
|
||||||
|
$ hideError eofErrorType
|
||||||
|
$ flip EX.finally (putMVar doneOut ())
|
||||||
|
$ readTilEOF (\x -> modifyIORef' rout (<> BL.fromStrict x)) pout
|
||||||
|
doneErr <- newEmptyMVar
|
||||||
|
void
|
||||||
|
$ forkIO
|
||||||
|
$ hideError eofErrorType
|
||||||
|
$ flip EX.finally (putMVar doneErr ())
|
||||||
|
$ readTilEOF (\x -> modifyIORef' rerr (<> BL.fromStrict x)) perr
|
||||||
|
takeMVar doneOut
|
||||||
|
takeMVar doneErr
|
||||||
|
|
||||||
|
readTilEOF ~action' fd' = do
|
||||||
|
bs <- SPIB.fdRead fd' 512
|
||||||
|
void $ action' bs
|
||||||
|
readTilEOF action' fd'
|
||||||
|
|
||||||
|
|
||||||
|
actionWithPipes :: ((Fd, Fd) -> IO b) -> IO b
|
||||||
|
actionWithPipes a =
|
||||||
|
createPipe >>= \(p1, p2) -> flip finally (cleanup [p1, p2]) $ a (p1, p2)
|
||||||
|
|
||||||
|
cleanup :: [Fd] -> IO ()
|
||||||
|
cleanup fds = for_ fds $ \fd -> handleIO (\_ -> pure ()) $ closeFd fd
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-- | Create a new regular file in write-only mode. The file must not exist.
|
||||||
|
createRegularFileFd :: FileMode -> FilePath -> IO Fd
|
||||||
|
createRegularFileFd fm dest =
|
||||||
|
openFd dest WriteOnly (Just fm) defaultFileFlags{ exclusive = True }
|
||||||
|
|
||||||
|
|
||||||
|
-- | Thin wrapper around `executeFile`.
|
||||||
|
exec :: MonadIO m
|
||||||
|
=> String -- ^ thing to execute
|
||||||
|
-> [String] -- ^ args for the thing
|
||||||
|
-> Maybe FilePath -- ^ optionally chdir into this
|
||||||
|
-> Maybe [(String, String)] -- ^ optional environment
|
||||||
|
-> m (Either ProcessError ())
|
||||||
|
exec exe args chdir env = liftIO $ do
|
||||||
|
pid <- SPP.forkProcess $ do
|
||||||
|
maybe (pure ()) changeWorkingDirectory chdir
|
||||||
|
SPP.executeFile exe (not ("./" `isPrefixOf` exe)) args env
|
||||||
|
|
||||||
|
fmap (toProcessError exe args) $ SPP.getProcessStatus True True pid
|
||||||
|
|
||||||
|
|
||||||
|
toProcessError :: FilePath
|
||||||
|
-> [String]
|
||||||
|
-> Maybe ProcessStatus
|
||||||
|
-> Either ProcessError ()
|
||||||
|
toProcessError exe args mps = case mps of
|
||||||
|
Just (SPP.Exited (ExitFailure xi)) -> Left $ NonZeroExit xi exe args
|
||||||
|
Just (SPP.Exited ExitSuccess ) -> Right ()
|
||||||
|
Just (Terminated _ _ ) -> Left $ PTerminated exe args
|
||||||
|
Just (Stopped _ ) -> Left $ PStopped exe args
|
||||||
|
Nothing -> Left $ NoSuchPid exe args
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
chmod_755 :: (MonadLogger m, MonadIO m) => FilePath -> m ()
|
||||||
|
chmod_755 fp = do
|
||||||
|
let exe_mode =
|
||||||
|
nullFileMode
|
||||||
|
`unionFileModes` ownerExecuteMode
|
||||||
|
`unionFileModes` ownerReadMode
|
||||||
|
`unionFileModes` ownerWriteMode
|
||||||
|
`unionFileModes` groupExecuteMode
|
||||||
|
`unionFileModes` groupReadMode
|
||||||
|
`unionFileModes` otherExecuteMode
|
||||||
|
`unionFileModes` otherReadMode
|
||||||
|
$(logDebug) [i|chmod 755 #{fp}|]
|
||||||
|
liftIO $ setFileMode fp exe_mode
|
||||||
|
|
||||||
|
|
||||||
|
-- |Default permissions for a new file.
|
||||||
|
newFilePerms :: FileMode
|
||||||
|
newFilePerms =
|
||||||
|
ownerWriteMode
|
||||||
|
`unionFileModes` ownerReadMode
|
||||||
|
`unionFileModes` groupWriteMode
|
||||||
|
`unionFileModes` groupReadMode
|
||||||
|
`unionFileModes` otherWriteMode
|
||||||
|
`unionFileModes` otherReadMode
|
||||||
|
|
||||||
|
|
||||||
|
-- | Checks whether the binary is a broken link.
|
||||||
|
isBrokenSymlink :: FilePath -> IO Bool
|
||||||
|
isBrokenSymlink fp = do
|
||||||
|
try (pathIsSymbolicLink fp) >>= \case
|
||||||
|
Right True -> do
|
||||||
|
let symDir = takeDirectory fp
|
||||||
|
tfp <- getSymbolicLinkTarget fp
|
||||||
|
not <$> doesPathExist
|
||||||
|
-- this drops 'symDir' if 'tfp' is absolute
|
||||||
|
(symDir </> tfp)
|
||||||
|
Right b -> pure b
|
||||||
|
Left e | isDoesNotExistError e -> pure False
|
||||||
|
| otherwise -> throwIO e
|
||||||
247
lib/GHCup/Utils/File/Windows.hs
Normal file
247
lib/GHCup/Utils/File/Windows.hs
Normal file
@@ -0,0 +1,247 @@
|
|||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Utils.File.Windows
|
||||||
|
Description : File and windows APIs
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : Windows
|
||||||
|
|
||||||
|
This module handles file and executable handling.
|
||||||
|
Some of these functions use sophisticated logging.
|
||||||
|
-}
|
||||||
|
module GHCup.Utils.File.Windows where
|
||||||
|
|
||||||
|
import {-# SOURCE #-} GHCup.Utils ( getLinkTarget, pathIsLink )
|
||||||
|
import GHCup.Utils.Dirs
|
||||||
|
import GHCup.Utils.File.Common
|
||||||
|
import GHCup.Types
|
||||||
|
|
||||||
|
import Control.Concurrent
|
||||||
|
import Control.DeepSeq
|
||||||
|
import Control.Exception.Safe
|
||||||
|
import Control.Monad
|
||||||
|
import Control.Monad.Reader
|
||||||
|
import Data.List
|
||||||
|
import Foreign.C.Error
|
||||||
|
import GHC.IO.Exception
|
||||||
|
import GHC.IO.Handle
|
||||||
|
import System.Directory
|
||||||
|
import System.Environment
|
||||||
|
import System.FilePath
|
||||||
|
import System.IO
|
||||||
|
import System.Process
|
||||||
|
|
||||||
|
import qualified Control.Exception as EX
|
||||||
|
import qualified Data.ByteString as BS
|
||||||
|
import qualified Data.ByteString.Lazy as BL
|
||||||
|
import qualified Data.Map.Strict as Map
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
toProcessError :: FilePath
|
||||||
|
-> [FilePath]
|
||||||
|
-> ExitCode
|
||||||
|
-> Either ProcessError ()
|
||||||
|
toProcessError exe args exitcode = case exitcode of
|
||||||
|
(ExitFailure xi) -> Left $ NonZeroExit xi exe args
|
||||||
|
ExitSuccess -> Right ()
|
||||||
|
|
||||||
|
|
||||||
|
-- | @readCreateProcessWithExitCode@ works exactly like 'readProcessWithExitCode' except that it
|
||||||
|
-- lets you pass 'CreateProcess' giving better flexibility.
|
||||||
|
--
|
||||||
|
-- Note that @Handle@s provided for @std_in@, @std_out@, or @std_err@ via the CreateProcess
|
||||||
|
-- record will be ignored.
|
||||||
|
--
|
||||||
|
-- @since 1.2.3.0
|
||||||
|
readCreateProcessWithExitCodeBS
|
||||||
|
:: CreateProcess
|
||||||
|
-> BL.ByteString
|
||||||
|
-> IO (ExitCode, BL.ByteString, BL.ByteString) -- ^ exitcode, stdout, stderr
|
||||||
|
readCreateProcessWithExitCodeBS cp input = do
|
||||||
|
let cp_opts = cp {
|
||||||
|
std_in = CreatePipe,
|
||||||
|
std_out = CreatePipe,
|
||||||
|
std_err = CreatePipe
|
||||||
|
}
|
||||||
|
withCreateProcess_ "readCreateProcessWithExitCodeBS" cp_opts $
|
||||||
|
\mb_inh mb_outh mb_errh ph ->
|
||||||
|
case (mb_inh, mb_outh, mb_errh) of
|
||||||
|
(Just inh, Just outh, Just errh) -> do
|
||||||
|
|
||||||
|
out <- BS.hGetContents outh
|
||||||
|
err <- BS.hGetContents errh
|
||||||
|
|
||||||
|
-- fork off threads to start consuming stdout & stderr
|
||||||
|
withForkWait (EX.evaluate $ rnf out) $ \waitOut ->
|
||||||
|
withForkWait (EX.evaluate $ rnf err) $ \waitErr -> do
|
||||||
|
|
||||||
|
-- now write any input
|
||||||
|
unless (BL.null input) $
|
||||||
|
ignoreSigPipe $ BL.hPut inh input
|
||||||
|
-- hClose performs implicit hFlush, and thus may trigger a SIGPIPE
|
||||||
|
ignoreSigPipe $ hClose inh
|
||||||
|
|
||||||
|
-- wait on the output
|
||||||
|
waitOut
|
||||||
|
waitErr
|
||||||
|
|
||||||
|
hClose outh
|
||||||
|
hClose errh
|
||||||
|
|
||||||
|
-- wait on the process
|
||||||
|
ex <- waitForProcess ph
|
||||||
|
return (ex, BL.fromStrict out, BL.fromStrict err)
|
||||||
|
|
||||||
|
(Nothing,_,_) -> error "readCreateProcessWithExitCodeBS: Failed to get a stdin handle."
|
||||||
|
(_,Nothing,_) -> error "readCreateProcessWithExitCodeBS: Failed to get a stdout handle."
|
||||||
|
(_,_,Nothing) -> error "readCreateProcessWithExitCodeBS: Failed to get a stderr handle."
|
||||||
|
where
|
||||||
|
ignoreSigPipe :: IO () -> IO ()
|
||||||
|
ignoreSigPipe = EX.handle $ \e -> case e of
|
||||||
|
IOError { ioe_type = ResourceVanished
|
||||||
|
, ioe_errno = Just ioe }
|
||||||
|
| Errno ioe == ePIPE -> return ()
|
||||||
|
_ -> throwIO e
|
||||||
|
-- wrapper so we can get exceptions with the appropriate function name.
|
||||||
|
withCreateProcess_
|
||||||
|
:: String
|
||||||
|
-> CreateProcess
|
||||||
|
-> (Maybe Handle -> Maybe Handle -> Maybe Handle -> ProcessHandle -> IO a)
|
||||||
|
-> IO a
|
||||||
|
withCreateProcess_ fun c action =
|
||||||
|
EX.bracketOnError (createProcess_ fun c) cleanupProcess
|
||||||
|
(\(m_in, m_out, m_err, ph) -> action m_in m_out m_err ph)
|
||||||
|
|
||||||
|
-- | Fork a thread while doing something else, but kill it if there's an
|
||||||
|
-- exception.
|
||||||
|
--
|
||||||
|
-- This is important in the cases above because we want to kill the thread
|
||||||
|
-- that is holding the Handle lock, because when we clean up the process we
|
||||||
|
-- try to close that handle, which could otherwise deadlock.
|
||||||
|
--
|
||||||
|
withForkWait :: IO () -> (IO () -> IO a) -> IO a
|
||||||
|
withForkWait async' body = do
|
||||||
|
waitVar <- newEmptyMVar :: IO (MVar (Either SomeException ()))
|
||||||
|
mask $ \restore -> do
|
||||||
|
tid <- forkIO $ try (restore async') >>= putMVar waitVar
|
||||||
|
let wait' = takeMVar waitVar >>= either throwIO return
|
||||||
|
restore (body wait') `EX.onException` killThread tid
|
||||||
|
|
||||||
|
|
||||||
|
-- | Execute the given command and collect the stdout, stderr and the exit code.
|
||||||
|
-- The command is run in a subprocess.
|
||||||
|
executeOut :: MonadIO m
|
||||||
|
=> FilePath -- ^ command as filename, e.g. 'ls'
|
||||||
|
-> [String] -- ^ arguments to the command
|
||||||
|
-> Maybe FilePath -- ^ chdir to this path
|
||||||
|
-> m CapturedProcess
|
||||||
|
executeOut path args chdir = do
|
||||||
|
cp <- createProcessWithMingwPath ((proc path args){ cwd = chdir })
|
||||||
|
(exit, out, err) <- liftIO $ readCreateProcessWithExitCodeBS cp ""
|
||||||
|
pure $ CapturedProcess exit out err
|
||||||
|
|
||||||
|
|
||||||
|
execLogged :: (MonadReader AppState m, MonadIO m, MonadThrow m)
|
||||||
|
=> FilePath -- ^ thing to execute
|
||||||
|
-> [String] -- ^ args for the thing
|
||||||
|
-> Maybe FilePath -- ^ optionally chdir into this
|
||||||
|
-> FilePath -- ^ log filename (opened in append mode)
|
||||||
|
-> Maybe [(String, String)] -- ^ optional environment
|
||||||
|
-> m (Either ProcessError ())
|
||||||
|
execLogged exe args chdir lfile env = do
|
||||||
|
AppState { dirs = Dirs {..} } <- ask
|
||||||
|
let stdoutLogfile = logsDir </> lfile <> ".stdout.log"
|
||||||
|
stderrLogfile = logsDir </> lfile <> ".stderr.log"
|
||||||
|
cp <- createProcessWithMingwPath ((proc exe args)
|
||||||
|
{ cwd = chdir
|
||||||
|
, env = env
|
||||||
|
, std_in = CreatePipe
|
||||||
|
, std_out = CreatePipe
|
||||||
|
, std_err = CreatePipe
|
||||||
|
})
|
||||||
|
fmap (toProcessError exe args)
|
||||||
|
$ liftIO
|
||||||
|
$ withCreateProcess cp
|
||||||
|
$ \_ mout merr ph ->
|
||||||
|
case (mout, merr) of
|
||||||
|
(Just cStdout, Just cStderr) -> do
|
||||||
|
withForkWait (tee stdoutLogfile cStdout) $ \waitOut ->
|
||||||
|
withForkWait (tee stderrLogfile cStderr) $ \waitErr -> do
|
||||||
|
waitOut
|
||||||
|
waitErr
|
||||||
|
waitForProcess ph
|
||||||
|
_ -> fail "Could not acquire out/err handle"
|
||||||
|
|
||||||
|
where
|
||||||
|
tee :: FilePath -> Handle -> IO ()
|
||||||
|
tee logFile handle' = go
|
||||||
|
where
|
||||||
|
go = do
|
||||||
|
some <- BS.hGetSome handle' 512
|
||||||
|
if BS.null some
|
||||||
|
then pure ()
|
||||||
|
else do
|
||||||
|
void $ BS.appendFile logFile some
|
||||||
|
void $ BS.hPut stdout some
|
||||||
|
go
|
||||||
|
|
||||||
|
|
||||||
|
-- | Thin wrapper around `executeFile`.
|
||||||
|
exec :: MonadIO m
|
||||||
|
=> FilePath -- ^ thing to execute
|
||||||
|
-> [FilePath] -- ^ args for the thing
|
||||||
|
-> Maybe FilePath -- ^ optionally chdir into this
|
||||||
|
-> Maybe [(String, String)] -- ^ optional environment
|
||||||
|
-> m (Either ProcessError ())
|
||||||
|
exec exe args chdir env = do
|
||||||
|
cp <- createProcessWithMingwPath ((proc exe args) { cwd = chdir, env = env })
|
||||||
|
exit_code <- liftIO $ withCreateProcess cp $ \_ _ _ p -> waitForProcess p
|
||||||
|
pure $ toProcessError exe args exit_code
|
||||||
|
|
||||||
|
|
||||||
|
chmod_755 :: MonadIO m => FilePath -> m ()
|
||||||
|
chmod_755 fp =
|
||||||
|
let perm = setOwnerWritable True emptyPermissions
|
||||||
|
in liftIO $ setPermissions fp perm
|
||||||
|
|
||||||
|
|
||||||
|
createProcessWithMingwPath :: MonadIO m
|
||||||
|
=> CreateProcess
|
||||||
|
-> m CreateProcess
|
||||||
|
createProcessWithMingwPath cp = do
|
||||||
|
msys2Dir <- liftIO ghcupMsys2Dir
|
||||||
|
cEnv <- Map.fromList <$> maybe (liftIO getEnvironment) pure (env cp)
|
||||||
|
let mingWPaths = [msys2Dir </> "usr" </> "bin"
|
||||||
|
,msys2Dir </> "mingw64" </> "bin"]
|
||||||
|
paths = ["PATH", "Path"]
|
||||||
|
curPaths = (\x -> maybe [] splitSearchPath (Map.lookup x cEnv)) =<< paths
|
||||||
|
newPath = intercalate [searchPathSeparator] (mingWPaths ++ curPaths)
|
||||||
|
envWithoutPath = foldr (\x y -> Map.delete x y) cEnv paths
|
||||||
|
envWithNewPath = Map.insert "Path" newPath envWithoutPath
|
||||||
|
liftIO $ setEnv "Path" newPath
|
||||||
|
pure $ cp { env = Just $ Map.toList envWithNewPath }
|
||||||
|
|
||||||
|
ghcupMsys2Dir :: IO FilePath
|
||||||
|
ghcupMsys2Dir =
|
||||||
|
lookupEnv "GHCUP_MSYS2" >>= \case
|
||||||
|
Just fp -> pure fp
|
||||||
|
Nothing -> do
|
||||||
|
baseDir <- liftIO ghcupBaseDir
|
||||||
|
pure (baseDir </> "msys64")
|
||||||
|
|
||||||
|
-- | Checks whether the binary is a broken link.
|
||||||
|
isBrokenSymlink :: FilePath -> IO Bool
|
||||||
|
isBrokenSymlink fp = do
|
||||||
|
b <- pathIsLink fp
|
||||||
|
if b
|
||||||
|
then do
|
||||||
|
tfp <- getLinkTarget fp
|
||||||
|
not <$> doesPathExist
|
||||||
|
-- this drops 'symDir' if 'tfp' is absolute
|
||||||
|
(takeDirectory fp </> tfp)
|
||||||
|
else pure False
|
||||||
@@ -1,18 +1,33 @@
|
|||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Utils.Logger
|
||||||
|
Description : logger definition
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
|
||||||
|
Here we define our main logger.
|
||||||
|
-}
|
||||||
module GHCup.Utils.Logger where
|
module GHCup.Utils.Logger where
|
||||||
|
|
||||||
import GHCup.Utils
|
import GHCup.Utils.File
|
||||||
|
import GHCup.Utils.String.QQ
|
||||||
|
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
|
import Control.Monad.IO.Class
|
||||||
import Control.Monad.Logger
|
import Control.Monad.Logger
|
||||||
import HPath
|
|
||||||
import HPath.IO
|
|
||||||
import Prelude hiding ( appendFile )
|
import Prelude hiding ( appendFile )
|
||||||
import System.Console.Pretty
|
import System.Console.Pretty
|
||||||
|
import System.FilePath
|
||||||
import System.IO.Error
|
import System.IO.Error
|
||||||
|
import Text.Regex.Posix
|
||||||
|
|
||||||
import qualified Data.ByteString as B
|
import qualified Data.ByteString as B
|
||||||
|
import GHCup.Utils.Prelude
|
||||||
|
|
||||||
|
|
||||||
data LoggerConfig = LoggerConfig
|
data LoggerConfig = LoggerConfig
|
||||||
@@ -36,7 +51,7 @@ myLoggerT LoggerConfig {..} loggingt = runLoggingT loggingt mylogger
|
|||||||
LevelOther t -> toLogStr "[ " <> toLogStr t <> toLogStr " ]"
|
LevelOther t -> toLogStr "[ " <> toLogStr t <> toLogStr " ]"
|
||||||
let out = fromLogStr (l <> toLogStr " " <> str' <> toLogStr "\n")
|
let out = fromLogStr (l <> toLogStr " " <> str' <> toLogStr "\n")
|
||||||
|
|
||||||
when (lcPrintDebug || (lcPrintDebug == False && not (level == LevelDebug)))
|
when (lcPrintDebug || (not lcPrintDebug && (level /= LevelDebug)))
|
||||||
$ colorOutter out
|
$ colorOutter out
|
||||||
|
|
||||||
-- raw output
|
-- raw output
|
||||||
@@ -50,11 +65,17 @@ myLoggerT LoggerConfig {..} loggingt = runLoggingT loggingt mylogger
|
|||||||
rawOutter outr
|
rawOutter outr
|
||||||
|
|
||||||
|
|
||||||
initGHCupFileLogging :: Path Rel -> IO (Path Abs)
|
initGHCupFileLogging :: (MonadIO m) => FilePath -> m FilePath
|
||||||
initGHCupFileLogging context = do
|
initGHCupFileLogging logsDir = do
|
||||||
logs <- ghcupLogsDir
|
let logfile = logsDir </> "ghcup.log"
|
||||||
let logfile = logs </> context
|
liftIO $ do
|
||||||
createDirIfMissing newDirPerms logs
|
logFiles <- findFiles
|
||||||
hideError doesNotExistErrorType $ deleteFile logfile
|
logsDir
|
||||||
createRegularFile newFilePerms logfile
|
(makeRegexOpts compExtended
|
||||||
pure logfile
|
execBlank
|
||||||
|
([s|^.*\.log$|] :: B.ByteString)
|
||||||
|
)
|
||||||
|
forM_ logFiles $ hideError doesNotExistErrorType . rmFile . (logsDir </>)
|
||||||
|
|
||||||
|
writeFile logfile ""
|
||||||
|
pure logfile
|
||||||
|
|||||||
124
lib/GHCup/Utils/MegaParsec.hs
Normal file
124
lib/GHCup/Utils/MegaParsec.hs
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
{-# LANGUAGE CPP #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Utils.MegaParsec
|
||||||
|
Description : MegaParsec utilities
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
|
module GHCup.Utils.MegaParsec where
|
||||||
|
|
||||||
|
import GHCup.Types
|
||||||
|
|
||||||
|
import Control.Applicative
|
||||||
|
#if !MIN_VERSION_base(4,13,0)
|
||||||
|
import Control.Monad.Fail ( MonadFail )
|
||||||
|
#endif
|
||||||
|
import Data.Functor
|
||||||
|
import Data.Maybe
|
||||||
|
import Data.Text ( Text )
|
||||||
|
import Data.Versions
|
||||||
|
import Data.Void
|
||||||
|
import System.FilePath
|
||||||
|
|
||||||
|
import qualified Data.List.NonEmpty as NE
|
||||||
|
import qualified Data.Text as T
|
||||||
|
import qualified Text.Megaparsec as MP
|
||||||
|
|
||||||
|
|
||||||
|
choice' :: (MonadFail f, MP.MonadParsec e s f) => [f a] -> f a
|
||||||
|
choice' [] = fail "Empty list"
|
||||||
|
choice' [x ] = x
|
||||||
|
choice' (x : xs) = MP.try x <|> choice' xs
|
||||||
|
|
||||||
|
|
||||||
|
parseUntil :: MP.Parsec Void Text a -> MP.Parsec Void Text Text
|
||||||
|
parseUntil p = do
|
||||||
|
(MP.try (MP.lookAhead p) $> mempty)
|
||||||
|
<|> (do
|
||||||
|
c <- T.singleton <$> MP.anySingle
|
||||||
|
c2 <- parseUntil p
|
||||||
|
pure (c `mappend` c2)
|
||||||
|
)
|
||||||
|
|
||||||
|
parseUntil1 :: MP.Parsec Void Text a -> MP.Parsec Void Text Text
|
||||||
|
parseUntil1 p = do
|
||||||
|
i1 <- MP.getOffset
|
||||||
|
t <- parseUntil p
|
||||||
|
i2 <- MP.getOffset
|
||||||
|
if i1 == i2 then fail "empty parse" else pure t
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
-- | Parses e.g.
|
||||||
|
-- * armv7-unknown-linux-gnueabihf-ghc
|
||||||
|
-- * armv7-unknown-linux-gnueabihf-ghci
|
||||||
|
ghcTargetBinP :: Text -> MP.Parsec Void Text (Maybe Text, Text)
|
||||||
|
ghcTargetBinP t =
|
||||||
|
(,)
|
||||||
|
<$> ( MP.try
|
||||||
|
(Just <$> parseUntil1 (MP.chunk "-" *> MP.chunk t) <* MP.chunk "-"
|
||||||
|
)
|
||||||
|
<|> ((\ _ x -> x) Nothing <$> mempty)
|
||||||
|
)
|
||||||
|
<*> (MP.chunk t <* MP.eof)
|
||||||
|
|
||||||
|
|
||||||
|
-- | Extracts the version from @ProjectVersion="8.10.5"@.
|
||||||
|
ghcProjectVersion :: MP.Parsec Void Text Version
|
||||||
|
ghcProjectVersion = do
|
||||||
|
_ <- MP.chunk "ProjectVersion=\""
|
||||||
|
ver <- parseUntil1 $ MP.chunk "\""
|
||||||
|
MP.setInput ver
|
||||||
|
version'
|
||||||
|
|
||||||
|
|
||||||
|
-- | Extracts target triple and version from e.g.
|
||||||
|
-- * armv7-unknown-linux-gnueabihf-8.8.3
|
||||||
|
-- * armv7-unknown-linux-gnueabihf-8.8.3
|
||||||
|
ghcTargetVerP :: MP.Parsec Void Text GHCTargetVersion
|
||||||
|
ghcTargetVerP =
|
||||||
|
(\x y -> GHCTargetVersion x y)
|
||||||
|
<$> (MP.try (Just <$> parseUntil1 (MP.chunk "-" *> verP') <* MP.chunk "-")
|
||||||
|
<|> ((\ _ x -> x) Nothing <$> mempty)
|
||||||
|
)
|
||||||
|
<*> (version' <* MP.eof)
|
||||||
|
where
|
||||||
|
verP' :: MP.Parsec Void Text Text
|
||||||
|
verP' = do
|
||||||
|
v <- version'
|
||||||
|
let startsWithDigists =
|
||||||
|
and
|
||||||
|
. take 3
|
||||||
|
. concatMap
|
||||||
|
(map
|
||||||
|
(\case
|
||||||
|
(Digits _) -> True
|
||||||
|
(Str _) -> False
|
||||||
|
) . NE.toList)
|
||||||
|
. NE.toList
|
||||||
|
$ _vChunks v
|
||||||
|
if startsWithDigists && isNothing (_vEpoch v)
|
||||||
|
then pure $ prettyVer v
|
||||||
|
else fail "Oh"
|
||||||
|
|
||||||
|
|
||||||
|
verP :: MP.Parsec Void Text Text -> MP.Parsec Void Text Versioning
|
||||||
|
verP suffix = do
|
||||||
|
ver <- parseUntil suffix
|
||||||
|
if T.null ver
|
||||||
|
then fail "empty version"
|
||||||
|
else do
|
||||||
|
rest <- MP.getInput
|
||||||
|
MP.setInput ver
|
||||||
|
v <- versioning'
|
||||||
|
MP.setInput rest
|
||||||
|
pure v
|
||||||
|
|
||||||
|
|
||||||
|
pathSep :: MP.Parsec Void Text Char
|
||||||
|
pathSep = MP.oneOf pathSeparators
|
||||||
@@ -1,13 +1,22 @@
|
|||||||
|
{-# LANGUAGE CPP #-}
|
||||||
{-# LANGUAGE DataKinds #-}
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE DeriveLift #-}
|
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE FlexibleInstances #-}
|
{-# LANGUAGE FlexibleInstances #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
|
||||||
{-# LANGUAGE ScopedTypeVariables #-}
|
{-# LANGUAGE ScopedTypeVariables #-}
|
||||||
{-# LANGUAGE TypeApplications #-}
|
|
||||||
{-# LANGUAGE TypeFamilies #-}
|
{-# LANGUAGE TypeFamilies #-}
|
||||||
{-# LANGUAGE TypeOperators #-}
|
{-# LANGUAGE TypeOperators #-}
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Utils.Prelude
|
||||||
|
Description : MegaParsec utilities
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
|
||||||
|
GHCup specific prelude. Lots of Excepts functionality.
|
||||||
|
-}
|
||||||
module GHCup.Utils.Prelude where
|
module GHCup.Utils.Prelude where
|
||||||
|
|
||||||
import Control.Applicative
|
import Control.Applicative
|
||||||
@@ -17,18 +26,30 @@ import Control.Monad.IO.Class
|
|||||||
import Control.Monad.Trans.Class ( lift )
|
import Control.Monad.Trans.Class ( lift )
|
||||||
import Data.Bifunctor
|
import Data.Bifunctor
|
||||||
import Data.ByteString ( ByteString )
|
import Data.ByteString ( ByteString )
|
||||||
|
import Data.List ( nub )
|
||||||
|
import Data.Foldable
|
||||||
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.IO.Unsafe
|
||||||
|
import System.Directory
|
||||||
|
import System.FilePath
|
||||||
|
|
||||||
|
#if defined(IS_WINDOWS)
|
||||||
|
import Control.Retry
|
||||||
|
import GHC.IO.Exception
|
||||||
|
#endif
|
||||||
|
|
||||||
|
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
|
||||||
import qualified Data.Text.Encoding as E
|
import qualified Data.Text.Encoding as E
|
||||||
|
import qualified Data.Text.Encoding.Error as E
|
||||||
import qualified Data.Text.Lazy as TL
|
import qualified Data.Text.Lazy as TL
|
||||||
import qualified Data.Text.Lazy.Builder as B
|
import qualified Data.Text.Lazy.Builder as B
|
||||||
import qualified Data.Text.Lazy.Builder.Int as B
|
import qualified Data.Text.Lazy.Builder.Int as B
|
||||||
@@ -87,10 +108,6 @@ whileM_ ~action = void . whileM action
|
|||||||
guardM :: (Monad m, Alternative m) => m Bool -> m ()
|
guardM :: (Monad m, Alternative m) => m Bool -> m ()
|
||||||
guardM ~f = guard =<< f
|
guardM ~f = guard =<< f
|
||||||
|
|
||||||
lBS2sT :: L.ByteString -> Text
|
|
||||||
lBS2sT = TL.toStrict . TLE.decodeUtf8
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
handleIO' :: (MonadIO m, MonadCatch m)
|
handleIO' :: (MonadIO m, MonadCatch m)
|
||||||
=> IOErrorType
|
=> IOErrorType
|
||||||
@@ -121,7 +138,7 @@ lE' :: forall e' e es a m
|
|||||||
=> (e' -> e)
|
=> (e' -> e)
|
||||||
-> Either e' a
|
-> Either e' a
|
||||||
-> Excepts es m a
|
-> Excepts es m a
|
||||||
lE' f = liftE . veitherToExcepts . fromEither . bimap f id
|
lE' f = liftE . veitherToExcepts . fromEither . first f
|
||||||
|
|
||||||
lEM :: forall e es a m . (Monad m, e :< es) => m (Either e a) -> Excepts es m a
|
lEM :: forall e es a m . (Monad m, e :< es) => m (Either e a) -> Excepts es m a
|
||||||
lEM em = lift em >>= lE
|
lEM em = lift em >>= lE
|
||||||
@@ -131,7 +148,7 @@ lEM' :: forall e' e es a m
|
|||||||
=> (e' -> e)
|
=> (e' -> e)
|
||||||
-> m (Either e' a)
|
-> m (Either e' a)
|
||||||
-> Excepts es m a
|
-> Excepts es m a
|
||||||
lEM' f em = lift em >>= lE . bimap f id
|
lEM' f em = lift em >>= lE . first f
|
||||||
|
|
||||||
fromEither :: Either a b -> VEither '[a] b
|
fromEither :: Either a b -> VEither '[a] b
|
||||||
fromEither = either (VLeft . V) VRight
|
fromEither = either (VLeft . V) VRight
|
||||||
@@ -168,14 +185,19 @@ liftIOException errType ex =
|
|||||||
. lift
|
. lift
|
||||||
|
|
||||||
|
|
||||||
hideErrorDef :: IOErrorType -> a -> IO a -> IO a
|
-- | Uses safe-exceptions.
|
||||||
hideErrorDef err def =
|
hideError :: (MonadIO m, MonadCatch m) => IOErrorType -> m () -> m ()
|
||||||
handleIO (\e -> if err == ioeGetErrorType e then pure def else ioError e)
|
hideError err = handleIO (\e -> if err == ioeGetErrorType e then pure () else liftIO . ioError $ e)
|
||||||
|
|
||||||
|
|
||||||
hideErrorDefM :: IOErrorType -> IO a -> IO a -> IO a
|
hideErrorDef :: (MonadIO m, MonadCatch m) => [IOErrorType] -> a -> m a -> m a
|
||||||
hideErrorDefM err def =
|
hideErrorDef errs def =
|
||||||
handleIO (\e -> if err == ioeGetErrorType e then def else ioError e)
|
handleIO (\e -> if ioeGetErrorType e `elem` errs then pure def else liftIO $ ioError e)
|
||||||
|
|
||||||
|
|
||||||
|
hideErrorDefM :: (MonadIO m, MonadCatch m) => [IOErrorType] -> m a -> m a -> m a
|
||||||
|
hideErrorDefM errs def =
|
||||||
|
handleIO (\e -> if ioeGetErrorType e `elem` errs then def else liftIO $ ioError e)
|
||||||
|
|
||||||
|
|
||||||
-- TODO: does this work?
|
-- TODO: does this work?
|
||||||
@@ -185,8 +207,8 @@ hideExcept :: forall e es es' a m
|
|||||||
-> a
|
-> a
|
||||||
-> Excepts es m a
|
-> Excepts es m a
|
||||||
-> Excepts es' m a
|
-> Excepts es' m a
|
||||||
hideExcept _ a action =
|
hideExcept _ a =
|
||||||
catchLiftLeft ((\_ -> pure a) :: (e -> Excepts es' m a)) action
|
catchLiftLeft ((\_ -> pure a) :: (e -> Excepts es' m a))
|
||||||
|
|
||||||
|
|
||||||
hideExcept' :: forall e es es' m
|
hideExcept' :: forall e es es' m
|
||||||
@@ -194,8 +216,8 @@ hideExcept' :: forall e es es' m
|
|||||||
=> e
|
=> e
|
||||||
-> Excepts es m ()
|
-> Excepts es m ()
|
||||||
-> Excepts es' m ()
|
-> Excepts es' m ()
|
||||||
hideExcept' _ action =
|
hideExcept' _ =
|
||||||
catchLiftLeft ((\_ -> pure ()) :: (e -> Excepts es' m ())) action
|
catchLiftLeft ((\_ -> pure ()) :: (e -> Excepts es' m ()))
|
||||||
|
|
||||||
|
|
||||||
reThrowAll :: forall e es es' a m
|
reThrowAll :: forall e es es' a m
|
||||||
@@ -221,9 +243,17 @@ throwEither a = case a of
|
|||||||
Right r -> pure r
|
Right r -> pure r
|
||||||
|
|
||||||
|
|
||||||
|
throwEither' :: (Exception a, MonadThrow m) => a -> Either x b -> m b
|
||||||
|
throwEither' e eth = case eth of
|
||||||
|
Left _ -> throwM e
|
||||||
|
Right r -> pure r
|
||||||
|
|
||||||
|
|
||||||
verToBS :: Version -> ByteString
|
verToBS :: Version -> ByteString
|
||||||
verToBS = E.encodeUtf8 . prettyVer
|
verToBS = E.encodeUtf8 . prettyVer
|
||||||
|
|
||||||
|
verToS :: Version -> String
|
||||||
|
verToS = T.unpack . prettyVer
|
||||||
|
|
||||||
intToText :: Integral a => a -> T.Text
|
intToText :: Integral a => a -> T.Text
|
||||||
intToText = TL.toStrict . B.toLazyText . B.decimal
|
intToText = TL.toStrict . B.toLazyText . B.decimal
|
||||||
@@ -234,9 +264,164 @@ removeLensFieldLabel str' =
|
|||||||
maybe str' T.unpack . T.stripPrefix (T.pack "_") . T.pack $ str'
|
maybe str' T.unpack . T.stripPrefix (T.pack "_") . T.pack $ str'
|
||||||
|
|
||||||
|
|
||||||
addToCurrentEnv :: MonadIO m
|
pvpToVersion :: PVP -> Version
|
||||||
=> [(ByteString, ByteString)]
|
pvpToVersion =
|
||||||
-> m [(ByteString, ByteString)]
|
either (\_ -> error "Couldn't convert PVP to Version") id
|
||||||
addToCurrentEnv adds = do
|
. version
|
||||||
cEnv <- liftIO $ getEnvironment
|
. prettyPVP
|
||||||
pure (adds ++ cEnv)
|
|
||||||
|
|
||||||
|
-- | Safe 'decodeUtf8With'. Replaces an invalid input byte with
|
||||||
|
-- the Unicode replacement character U+FFFD.
|
||||||
|
decUTF8Safe :: ByteString -> Text
|
||||||
|
decUTF8Safe = E.decodeUtf8With E.lenientDecode
|
||||||
|
|
||||||
|
decUTF8Safe' :: L.ByteString -> Text
|
||||||
|
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
|
||||||
|
|
||||||
|
-- | More permissive version of 'createDirRecursive'. This doesn't
|
||||||
|
-- error when the destination is a symlink to a directory.
|
||||||
|
createDirRecursive' :: FilePath -> IO ()
|
||||||
|
createDirRecursive' p =
|
||||||
|
handleIO (\e -> if isAlreadyExistsError e then isSymlinkDir e else throwIO e)
|
||||||
|
. createDirectoryIfMissing True
|
||||||
|
$ p
|
||||||
|
|
||||||
|
where
|
||||||
|
isSymlinkDir e = do
|
||||||
|
ft <- pathIsSymbolicLink p
|
||||||
|
case ft of
|
||||||
|
True -> do
|
||||||
|
rp <- canonicalizePath p
|
||||||
|
rft <- doesDirectoryExist rp
|
||||||
|
case rft of
|
||||||
|
True -> pure ()
|
||||||
|
_ -> throwIO e
|
||||||
|
_ -> throwIO e
|
||||||
|
|
||||||
|
|
||||||
|
-- | Recursively copy the contents of one directory to another path.
|
||||||
|
--
|
||||||
|
-- This is a rip-off of Cabal library.
|
||||||
|
copyDirectoryRecursive :: FilePath -> FilePath -> IO ()
|
||||||
|
copyDirectoryRecursive srcDir destDir = do
|
||||||
|
srcFiles <- getDirectoryContentsRecursive srcDir
|
||||||
|
copyFilesWith copyFile destDir [ (srcDir, f)
|
||||||
|
| f <- srcFiles ]
|
||||||
|
where
|
||||||
|
-- | Common implementation of 'copyFiles', 'installOrdinaryFiles',
|
||||||
|
-- 'installExecutableFiles' and 'installMaybeExecutableFiles'.
|
||||||
|
copyFilesWith :: (FilePath -> FilePath -> IO ())
|
||||||
|
-> FilePath -> [(FilePath, FilePath)] -> IO ()
|
||||||
|
copyFilesWith doCopy targetDir srcFiles = do
|
||||||
|
|
||||||
|
-- Create parent directories for everything
|
||||||
|
let dirs = map (targetDir </>) . nub . map (takeDirectory . snd) $ srcFiles
|
||||||
|
traverse_ (createDirectoryIfMissing True) dirs
|
||||||
|
|
||||||
|
-- Copy all the files
|
||||||
|
sequence_ [ let src = srcBase </> srcFile
|
||||||
|
dest = targetDir </> srcFile
|
||||||
|
in doCopy src dest
|
||||||
|
| (srcBase, srcFile) <- srcFiles ]
|
||||||
|
|
||||||
|
|
||||||
|
-- | List all the files in a directory and all subdirectories.
|
||||||
|
--
|
||||||
|
-- The order places files in sub-directories after all the files in their
|
||||||
|
-- parent directories. The list is generated lazily so is not well defined if
|
||||||
|
-- the source directory structure changes before the list is used.
|
||||||
|
--
|
||||||
|
getDirectoryContentsRecursive :: FilePath -> IO [FilePath]
|
||||||
|
getDirectoryContentsRecursive topdir = recurseDirectories [""]
|
||||||
|
where
|
||||||
|
recurseDirectories :: [FilePath] -> IO [FilePath]
|
||||||
|
recurseDirectories [] = return []
|
||||||
|
recurseDirectories (dir:dirs) = unsafeInterleaveIO $ do
|
||||||
|
(files, dirs') <- collect [] [] =<< getDirectoryContents (topdir </> dir)
|
||||||
|
files' <- recurseDirectories (dirs' ++ dirs)
|
||||||
|
return (files ++ files')
|
||||||
|
|
||||||
|
where
|
||||||
|
collect files dirs' [] = return (reverse files
|
||||||
|
,reverse dirs')
|
||||||
|
collect files dirs' (entry:entries) | ignore entry
|
||||||
|
= collect files dirs' entries
|
||||||
|
collect files dirs' (entry:entries) = do
|
||||||
|
let dirEntry = dir </> entry
|
||||||
|
isDirectory <- doesDirectoryExist (topdir </> dirEntry)
|
||||||
|
if isDirectory
|
||||||
|
then collect files (dirEntry:dirs') entries
|
||||||
|
else collect (dirEntry:files) dirs' entries
|
||||||
|
|
||||||
|
ignore ['.'] = True
|
||||||
|
ignore ['.', '.'] = True
|
||||||
|
ignore _ = False
|
||||||
|
|
||||||
|
-- https://github.com/haskell/directory/issues/110
|
||||||
|
-- https://github.com/haskell/directory/issues/96
|
||||||
|
-- https://www.sqlite.org/src/info/89f1848d7f
|
||||||
|
rmPath :: (MonadIO m, MonadMask m)
|
||||||
|
=> FilePath
|
||||||
|
-> m ()
|
||||||
|
rmPath fp =
|
||||||
|
#if defined(IS_WINDOWS)
|
||||||
|
recovering (fullJitterBackoff 25000 <> limitRetries 10)
|
||||||
|
[\_ -> Handler (\e -> pure $ isPermissionError e)
|
||||||
|
,\_ -> Handler (\e -> pure (ioeGetErrorType e == UnsatisfiedConstraints))
|
||||||
|
,\_ -> Handler (\e -> pure (ioeGetErrorType e == InappropriateType))
|
||||||
|
]
|
||||||
|
(\_ -> liftIO $ removePathForcibly fp)
|
||||||
|
#else
|
||||||
|
liftIO $ removeDirectoryRecursive fp
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
-- https://www.sqlite.org/src/info/89f1848d7f
|
||||||
|
-- https://github.com/haskell/directory/issues/96
|
||||||
|
rmFile :: (MonadIO m, MonadMask m)
|
||||||
|
=> FilePath
|
||||||
|
-> m ()
|
||||||
|
rmFile fp =
|
||||||
|
#if defined(IS_WINDOWS)
|
||||||
|
recovering (fullJitterBackoff 25000 <> limitRetries 10)
|
||||||
|
[\_ -> Handler (\e -> pure $ isPermissionError e)
|
||||||
|
,\_ -> Handler (\e -> pure (ioeGetErrorType e == UnsatisfiedConstraints))
|
||||||
|
]
|
||||||
|
(\_ -> liftIO $ removeFile fp)
|
||||||
|
#else
|
||||||
|
liftIO $ removeFile fp
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
-- Gathering monoidal values
|
||||||
|
traverseFold :: (Foldable t, Applicative m, Monoid b) => (a -> m b) -> t a -> m b
|
||||||
|
traverseFold f = foldl (\mb a -> (<>) <$> mb <*> f a) (pure mempty)
|
||||||
|
|
||||||
|
-- | Gathering monoidal values
|
||||||
|
forFold :: (Foldable t, Applicative m, Monoid b) => t a -> (a -> m b) -> m b
|
||||||
|
forFold = \t -> (`traverseFold` t)
|
||||||
|
|
||||||
|
|
||||||
|
-- | Strip @\\r@ and @\\n@ from 'ByteString's
|
||||||
|
stripNewline :: String -> String
|
||||||
|
stripNewline s
|
||||||
|
| null s = []
|
||||||
|
| head s `elem` "\n\r" = stripNewline (tail s)
|
||||||
|
| otherwise = head s : stripNewline (tail s)
|
||||||
|
|
||||||
|
|
||||||
|
isNewLine :: Word8 -> Bool
|
||||||
|
isNewLine w
|
||||||
|
| w == _lf = True
|
||||||
|
| w == _cr = True
|
||||||
|
| otherwise = False
|
||||||
|
|||||||
@@ -1,25 +1,35 @@
|
|||||||
{-# LANGUAGE TemplateHaskell #-}
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
|
||||||
-- | QuasiQuoter for non-interpolated strings, texts and bytestrings.
|
{-|
|
||||||
--
|
Module : GHCup.Utils.String.QQ
|
||||||
-- The "s" quoter contains a multi-line string with no interpolation at all,
|
Description : String quasi quoters
|
||||||
-- except that the leading newline is trimmed and carriage returns stripped.
|
Copyright : (c) Audrey Tang <audreyt@audreyt.org> 2019, Julian Ospald <hasufell@posteo.de> 2020
|
||||||
--
|
License : LGPL-3.0
|
||||||
-- @
|
Maintainer : hasufell@hasufell.de
|
||||||
-- {-\# LANGUAGE QuasiQuotes #-}
|
Stability : experimental
|
||||||
-- import Data.Text (Text)
|
Portability : portable
|
||||||
-- import Data.String.QQ
|
|
||||||
-- foo :: Text -- "String", "ByteString" etc also works
|
QuasiQuoter for non-interpolated strings, texts and bytestrings.
|
||||||
-- foo = [s|
|
|
||||||
-- Well here is a
|
The "s" quoter contains a multi-line string with no interpolation at all,
|
||||||
-- multi-line string!
|
except that the leading newline is trimmed and carriage returns stripped.
|
||||||
-- |]
|
|
||||||
-- @
|
@
|
||||||
--
|
{-\# LANGUAGE QuasiQuotes #-}
|
||||||
-- Any instance of the IsString type is permitted.
|
import Data.Text (Text)
|
||||||
--
|
import Data.String.QQ
|
||||||
-- (For GHC versions 6, write "[$s||]" instead of "[s||]".)
|
foo :: Text -- "String", "ByteString" etc also works
|
||||||
--
|
foo = [s|
|
||||||
|
Well here is a
|
||||||
|
multi-line string!
|
||||||
|
|]
|
||||||
|
@
|
||||||
|
|
||||||
|
Any instance of the IsString type is permitted.
|
||||||
|
|
||||||
|
(For GHC versions 6, write "[$s||]" instead of "[s||]".)
|
||||||
|
|
||||||
|
-}
|
||||||
module GHCup.Utils.String.QQ
|
module GHCup.Utils.String.QQ
|
||||||
( s
|
( s
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
{-# OPTIONS_GHC -Wno-orphans #-}
|
{-# OPTIONS_GHC -Wno-orphans #-}
|
||||||
|
{-# LANGUAGE CPP #-}
|
||||||
{-# LANGUAGE DeriveDataTypeable #-}
|
{-# LANGUAGE DeriveDataTypeable #-}
|
||||||
{-# LANGUAGE DeriveLift #-}
|
{-# LANGUAGE DeriveLift #-}
|
||||||
{-# LANGUAGE FlexibleInstances #-}
|
{-# LANGUAGE FlexibleInstances #-}
|
||||||
@@ -6,11 +7,23 @@
|
|||||||
{-# LANGUAGE TemplateHaskell #-}
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Utils.Version.QQ
|
||||||
|
Description : Version quasi-quoters
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
module GHCup.Utils.Version.QQ where
|
module GHCup.Utils.Version.QQ where
|
||||||
|
|
||||||
import Data.Data
|
import Data.Data
|
||||||
import Data.Text ( Text )
|
import Data.Text ( Text )
|
||||||
import Data.Versions
|
import Data.Versions
|
||||||
|
#if !MIN_VERSION_base(4,13,0)
|
||||||
|
import GHC.Base
|
||||||
|
#endif
|
||||||
import Language.Haskell.TH
|
import Language.Haskell.TH
|
||||||
import Language.Haskell.TH.Quote ( QuasiQuoter(..) )
|
import Language.Haskell.TH.Quote ( QuasiQuoter(..) )
|
||||||
import Language.Haskell.TH.Syntax ( Lift
|
import Language.Haskell.TH.Syntax ( Lift
|
||||||
@@ -29,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
|
||||||
@@ -36,9 +51,13 @@ deriving instance Data VSep
|
|||||||
deriving instance Lift VUnit
|
deriving instance Lift VUnit
|
||||||
deriving instance Data VUnit
|
deriving instance Data VUnit
|
||||||
|
|
||||||
|
#if !MIN_VERSION_base(4,13,0)
|
||||||
|
deriving instance Lift (NonEmpty Word)
|
||||||
|
#endif
|
||||||
|
|
||||||
qq :: (Text -> Q Exp) -> QuasiQuoter
|
qq :: (Text -> Q Exp) -> QuasiQuoter
|
||||||
qq quoteExp' = QuasiQuoter
|
qq quoteExp' = QuasiQuoter
|
||||||
{ quoteExp = (\s -> quoteExp' . T.pack $ s)
|
{ quoteExp = \s -> quoteExp' . T.pack $ s
|
||||||
, quotePat = \_ ->
|
, quotePat = \_ ->
|
||||||
fail "illegal QuasiQuote (allowed as expression only, used as a pattern)"
|
fail "illegal QuasiQuote (allowed as expression only, used as a pattern)"
|
||||||
, quoteType = \_ ->
|
, quoteType = \_ ->
|
||||||
@@ -82,4 +101,4 @@ liftText :: T.Text -> Q Exp
|
|||||||
liftText txt = AppE (VarE 'T.pack) <$> TH.lift (T.unpack txt)
|
liftText txt = AppE (VarE 'T.pack) <$> TH.lift (T.unpack txt)
|
||||||
|
|
||||||
liftDataWithText :: Data a => a -> Q Exp
|
liftDataWithText :: Data a => a -> Q Exp
|
||||||
liftDataWithText = dataToExpQ (\a -> liftText <$> cast a)
|
liftDataWithText = dataToExpQ (fmap liftText . cast)
|
||||||
|
|||||||
@@ -1,16 +1,49 @@
|
|||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
|
||||||
|
|
||||||
|
{-|
|
||||||
|
Module : GHCup.Version
|
||||||
|
Description : Version information and version handling.
|
||||||
|
Copyright : (c) Julian Ospald, 2020
|
||||||
|
License : LGPL-3.0
|
||||||
|
Maintainer : hasufell@hasufell.de
|
||||||
|
Stability : experimental
|
||||||
|
Portability : portable
|
||||||
|
-}
|
||||||
module GHCup.Version where
|
module GHCup.Version where
|
||||||
|
|
||||||
import GHCup.Utils.Version.QQ
|
import GHCup.Types
|
||||||
|
import Paths_ghcup (version)
|
||||||
|
|
||||||
import Data.Versions
|
import Data.Version (Version(versionBranch))
|
||||||
|
import Data.Versions hiding (version)
|
||||||
import URI.ByteString
|
import URI.ByteString
|
||||||
import URI.ByteString.QQ
|
import URI.ByteString.QQ
|
||||||
|
|
||||||
ghcupURL :: URI
|
import qualified Data.List.NonEmpty as NE
|
||||||
ghcupURL = [uri|https://www.haskell.org/ghcup/data/ghcup-0.0.1.json|]
|
import qualified Data.Text as T
|
||||||
|
|
||||||
|
-- | This reflects the API version of the YAML.
|
||||||
|
ghcupURL :: URI
|
||||||
|
ghcupURL = [uri|https://www.haskell.org/ghcup/data/ghcup-0.0.5.yaml|]
|
||||||
|
|
||||||
|
-- | The current ghcup version.
|
||||||
ghcUpVer :: PVP
|
ghcUpVer :: PVP
|
||||||
ghcUpVer = [pver|0.0.1|]
|
ghcUpVer = PVP . NE.fromList . fmap fromIntegral $ versionBranch version
|
||||||
|
|
||||||
|
-- | ghcup version as numeric string.
|
||||||
|
numericVer :: String
|
||||||
|
numericVer = T.unpack . prettyPVP $ ghcUpVer
|
||||||
|
|
||||||
|
versionCmp :: Versioning -> VersionCmp -> Bool
|
||||||
|
versionCmp ver1 (VR_gt ver2) = ver1 > ver2
|
||||||
|
versionCmp ver1 (VR_gteq ver2) = ver1 >= ver2
|
||||||
|
versionCmp ver1 (VR_lt ver2) = ver1 < ver2
|
||||||
|
versionCmp ver1 (VR_lteq ver2) = ver1 <= ver2
|
||||||
|
versionCmp ver1 (VR_eq ver2) = ver1 == ver2
|
||||||
|
|
||||||
|
versionRange :: Versioning -> VersionRange -> Bool
|
||||||
|
versionRange ver' (SimpleRange cmps) = and $ fmap (versionCmp ver') cmps
|
||||||
|
versionRange ver' (OrRange cmps range) =
|
||||||
|
versionRange ver' (SimpleRange cmps) || versionRange ver' range
|
||||||
|
|
||||||
|
|||||||
14
shell-completions/bash
Normal file
14
shell-completions/bash
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
_ghcup()
|
||||||
|
{
|
||||||
|
local CMDLINE
|
||||||
|
local IFS=$'\n'
|
||||||
|
CMDLINE=(--bash-completion-index $COMP_CWORD)
|
||||||
|
|
||||||
|
for arg in ${COMP_WORDS[@]}; do
|
||||||
|
CMDLINE=(${CMDLINE[@]} --bash-completion-word $arg)
|
||||||
|
done
|
||||||
|
|
||||||
|
COMPREPLY=( $(ghcup "${CMDLINE[@]}") )
|
||||||
|
}
|
||||||
|
|
||||||
|
complete -o filenames -F _ghcup ghcup
|
||||||
19
shell-completions/fish
Normal file
19
shell-completions/fish
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
function _ghcup
|
||||||
|
set -l cl (commandline --tokenize --current-process)
|
||||||
|
# Hack around fish issue #3934
|
||||||
|
set -l cn (commandline --tokenize --cut-at-cursor --current-process)
|
||||||
|
set -l cn (count $cn)
|
||||||
|
set -l tmpline --bash-completion-enriched --bash-completion-index $cn
|
||||||
|
for arg in $cl
|
||||||
|
set tmpline $tmpline --bash-completion-word $arg
|
||||||
|
end
|
||||||
|
for opt in (ghcup $tmpline)
|
||||||
|
if test -d $opt
|
||||||
|
echo -E "$opt/"
|
||||||
|
else
|
||||||
|
echo -E "$opt"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
complete --no-files --command ghcup --arguments '(_ghcup)'
|
||||||
32
shell-completions/zsh
Normal file
32
shell-completions/zsh
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
#compdef ghcup
|
||||||
|
|
||||||
|
local request
|
||||||
|
local completions
|
||||||
|
local word
|
||||||
|
local index=$((CURRENT - 1))
|
||||||
|
|
||||||
|
request=(--bash-completion-enriched --bash-completion-index $index)
|
||||||
|
for arg in ${words[@]}; do
|
||||||
|
request=(${request[@]} --bash-completion-word $arg)
|
||||||
|
done
|
||||||
|
|
||||||
|
IFS=$'\n' completions=($( ghcup "${request[@]}" ))
|
||||||
|
|
||||||
|
for word in $completions; do
|
||||||
|
local -a parts
|
||||||
|
|
||||||
|
# Split the line at a tab if there is one.
|
||||||
|
IFS=$'\t' parts=($( echo $word ))
|
||||||
|
|
||||||
|
if [[ -n $parts[2] ]]; then
|
||||||
|
if [[ $word[1] == "-" ]]; then
|
||||||
|
local desc=("$parts[1] ($parts[2])")
|
||||||
|
compadd -d desc -- $parts[1]
|
||||||
|
else
|
||||||
|
local desc=($(print -f "%-019s -- %s" $parts[1] $parts[2]))
|
||||||
|
compadd -l -d desc -- $parts[1]
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
compadd -f -- $word
|
||||||
|
fi
|
||||||
|
done
|
||||||
90
stack.yaml
90
stack.yaml
@@ -1,39 +1,55 @@
|
|||||||
resolver: lts-14.27
|
resolver: lts-17.11
|
||||||
|
|
||||||
packages:
|
packages:
|
||||||
- .
|
- .
|
||||||
|
|
||||||
extra-deps:
|
extra-deps:
|
||||||
- IfElse-0.85@sha256:6939b94acc6a55f545f63a168a349dd2fbe4b9a7cca73bf60282db5cc6aa47d2
|
- git: https://github.com/hasufell/text-conversions.git
|
||||||
- ascii-string-1.0.1.3
|
commit: 9abf0e5e5664a3178367597c32db19880477a53c
|
||||||
- brotli-0.0.0.0@sha256:448061ceabdcaa752bbaf208f255bbb7e90bbcf8ea8a913d26ffa7887636823b
|
|
||||||
- brotli-streams-0.0.0.0@sha256:c75a1d5d33420cbc9399c315e9b50a1976a5370f4fa8a40c71e11d011c2fedd6
|
- git: https://github.com/Bodigrim/tar
|
||||||
- case-insensitive-1.2.1.0
|
commit: ac197ec7ea4838dc2b4e22b9b888b080cedf29cf
|
||||||
- data-default-instances-base-0.1.0.1@sha256:985a13d7103e45a65f06f277b735ef025636014f0d29dd6de998bc7628e09be9
|
|
||||||
- fusion-plugin-types-0.1.0@sha256:0f11bbc445ab8ae3dbbb3d5d2ea198bdb1ac020518b7f4f7579035dc89182438
|
- IfElse-0.85@sha256:6939b94acc6a55f545f63a168a349dd2fbe4b9a7cca73bf60282db5cc6aa47d2,445
|
||||||
- generics-sop-0.5.0.0
|
- ascii-string-1.0.1.4@sha256:fa34f1d9ba57e8e89c0d4c9cef5e01ba32cb2d4373d13f92dcc0b531a6c6749b,2582
|
||||||
- haskus-utils-data-1.2@sha256:48f62aa23d84b94edd0338379d3b3d74a34d3c2dbabf8c448a774a89ca70ea5d
|
- brotli-0.0.0.0@sha256:2bf383a4cd308745740986be0b18381c5a0784393fe69b91456aacb2d603de46,2964
|
||||||
- haskus-utils-types-1.5
|
- brotli-streams-0.0.0.0@sha256:1af1e22f67b8bfd6ad0d05e61825e7a178d738f689ebbb21c1aab5f1bbcae176,2331
|
||||||
- haskus-utils-variant-3.0
|
- chs-cabal-0.1.1.0@sha256:20ec6a9fb5ab6991f1a4adf157c537bd5d3b98d08d3c09c387c954c7c50bd011,1153
|
||||||
- hpath-0.11.0
|
- chs-deps-0.1.0.0@sha256:0cdada6d2c682c41b20331b8c63c2ecfc7e806928585195fd544c9d41f3074fd,2496
|
||||||
- hpath-directory-0.13.2
|
- composition-prelude-3.0.0.2@sha256:1ffed216bd28d810fce0b5be83a661e2a892696d73b3f8de5c0f5edb9b5f0090,1216
|
||||||
- hpath-filepath-0.10.4
|
- haskus-utils-data-1.3@sha256:f62c4e49021b463185d043f7b69c727b63af641a71d7edd582d9f4f98e80e500,1466
|
||||||
- hpath-io-0.13.1
|
- haskus-utils-types-1.5.1@sha256:991c472f4e751e2f0d7aab6ad4220ef151d6160876dcf0511bbf876bbd432020,1298
|
||||||
- hpath-posix-0.13.1
|
- haskus-utils-variant-3.0@sha256:8d51e45d3b664e61ccc25a58b37c0ccc4ee7537138b9fee21cd15c356906dd34,2159
|
||||||
- http-io-streams-0.1.2.0
|
- http-io-streams-0.1.6.0@sha256:53f5bab177efb52cd65ec396fd04ed59b93e5f919fb3700cd7dacd6cfce6f06d,3582
|
||||||
- indexed-profunctors-0.1@sha256:ddf618d0d4c58319c1e735e746bc69a1021f13b6f475dc9614b80af03432e6d4
|
- hpath-filepath-0.10.4@sha256:e9e44fb5fdbade7f30b5b5451257dbee15b6ef1aae4060034d73008bb3b5d878,1269
|
||||||
- language-bash-0.9.0
|
- hpath-posix-0.13.3@sha256:abe472cf16bccd3a8b8814865ed3551a728fde0f3a2baea2acc03023bec6c565,1615
|
||||||
- optics-0.2
|
- hspec-2.7.10@sha256:c9e82c90086acebac576552a06f3cabd249bba048edd1667c7fae0b1313d5bce,1712
|
||||||
- optics-core-0.2@sha256:cfdf39871553769b59fcc54863a3521d262ea25d8d05d0f41ab87296c560cfa6
|
- hspec-core-2.7.10@sha256:2aba6ea126442b29e8183ab27f1c811706b19b1d83b02f193a896f6fc1589d13,4621
|
||||||
- optics-extra-0.2@sha256:211ce1dfd1b3ffd95c1158d8c8beb53cbd17c4d477169e226b1831607f6789eb
|
- hspec-discover-2.7.10@sha256:d08bf5dd785629f589571477d9beb7cd91529471bd89f39517c1cb4b9b38160f,2184
|
||||||
- optics-th-0.2@sha256:b4f6c5a3f134c697837190ed6da969dd284aefe79e43c3ef793093b607bb83b9
|
- hspec-golden-aeson-0.9.0.0@sha256:aa17274114026661ba4dfc9c60c230673c8f408bd86482fd611d2d5cb6aff996,2179
|
||||||
- optics-vl-0.2
|
- libarchive-3.0.2.1@sha256:40ebf2a278e585802427bc58826867208bb33822f63d56107a1fcc3ca04d691d,10990
|
||||||
- optparse-applicative-0.15.1.0
|
- lzma-static-5.2.5.3@sha256:2758ee58c35992fcf7db78e98684c357a16a82fa2a4e7c352a6c210c08c555d8,7308
|
||||||
- pretty-terminal-0.1.0.0
|
- os-release-1.0.1@sha256:1281c62081f438fc3f0874d3bae6a4887d5964ac25261ba06e29d368ab173467,2716
|
||||||
- sop-core-0.5.0.0@sha256:8734ab38b8c84837094eec657da0b58942e481e20166131f34cf6c7fe9787b07
|
- primitive-0.7.0.1@sha256:a381571c36edc7dca28b77fe8159b43c14c640087ec5946adacf949feec64231,3433
|
||||||
- streamly-0.7.1
|
- regex-posix-clib-2.7
|
||||||
- streamly-bytestring-0.1.2
|
- streamly-0.7.3@sha256:ad2a488fe802692ed47cab9fd0416c2904aac9e51cf2d8aafd1c3a40064c42f5,27421
|
||||||
- streamly-posix-0.1.0.0
|
- streamly-bytestring-0.1.2@sha256:cc828f41d1c714c711d38fb213b4ed186febabba598ab080e13255f69c20b13c,2469
|
||||||
- strict-base-0.4.0.0
|
- streamly-posix-0.1.0.1@sha256:5d89b806281035d34020387ed99dde1ddab282c7ed66df3b7cd010b38fd3517b,2138
|
||||||
- string-interpolate-0.2.0.0
|
- strict-base-0.4.0.0@sha256:2ff4e43cb95eedf2995558d7fc34d19362846413dd39e6aa6a5b3ea8228fef9f,1248
|
||||||
- table-layout-0.8.0.5
|
- xor-0.0.1.0@sha256:f8362b4a68562b9afbcd727ff64c1a303970df3a032e0033d2f4c094c3501df3,2243
|
||||||
- tar-bytestring-0.6.3.0
|
- zip-1.7.1@sha256:0ce03d0fbffba47c1ab6fbb9166f8ba5373d828d78587df21b7e9d7bb150f929,3918
|
||||||
- time-1.9.3
|
|
||||||
|
flags:
|
||||||
|
http-io-streams:
|
||||||
|
brotli: false
|
||||||
|
|
||||||
|
libarchive:
|
||||||
|
system-libarchive: false
|
||||||
|
|
||||||
|
regex-posix:
|
||||||
|
_regex-posix-clib: true
|
||||||
|
|
||||||
|
ghc-options:
|
||||||
|
"$locals": -O2
|
||||||
|
streamly: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
|
ghcup: -O2 -fspec-constr-recursive=16 -fmax-worker-args=16
|
||||||
|
|||||||
196
test/GHCup/ArbitraryTypes.hs
Normal file
196
test/GHCup/ArbitraryTypes.hs
Normal file
@@ -0,0 +1,196 @@
|
|||||||
|
{-# OPTIONS_GHC -Wno-orphans #-}
|
||||||
|
|
||||||
|
{-# LANGUAGE FlexibleInstances #-}
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
module GHCup.ArbitraryTypes where
|
||||||
|
|
||||||
|
|
||||||
|
import GHCup.Types
|
||||||
|
|
||||||
|
import Data.ByteString ( ByteString )
|
||||||
|
import Data.Versions
|
||||||
|
import Data.List.NonEmpty
|
||||||
|
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 [ pure (Scheme "http"), pure (Scheme "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 <*> pure (Query []) <*> 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 VersionRange where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary (NonEmpty VersionCmp) where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary VersionCmp where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary TarDir where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary Tool where
|
||||||
|
arbitrary = genericArbitrary
|
||||||
|
shrink = genericShrink
|
||||||
|
|
||||||
|
instance Arbitrary GlobalTool 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
|
||||||
|
|
||||||
16
test/GHCup/Types/JSONSpec.hs
Normal file
16
test/GHCup/Types/JSONSpec.hs
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
{-# 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)
|
||||||
10
test/Main.hs
Normal file
10
test/Main.hs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
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 #-}
|
||||||
@@ -1,66 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
set -eu
|
|
||||||
|
|
||||||
status_message() {
|
|
||||||
printf "\\033[0;32m%s\\033[0m\\n" "$1"
|
|
||||||
}
|
|
||||||
|
|
||||||
error_message() {
|
|
||||||
printf "\\033[0;31m%s\\033[0m\\n" "$1"
|
|
||||||
}
|
|
||||||
|
|
||||||
SCRIPTPATH="$( cd "$(dirname "$0")" ; pwd -P )"
|
|
||||||
CACHE_LOCATION="${HOME}/.cabal/packages/hackage.haskell.org/01-index.cache"
|
|
||||||
|
|
||||||
if [ ! -f "${CACHE_LOCATION}" ] ; then
|
|
||||||
error_message "${CACHE_LOCATION} does not exist, did you run 'cabal update'?"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "${SCRIPTPATH}/cabal.project" ] ; then
|
|
||||||
error_message "Could not find ${SCRIPTPATH}/cabal.project, skipping index state update."
|
|
||||||
exit 3
|
|
||||||
fi
|
|
||||||
|
|
||||||
cabal v2-update
|
|
||||||
|
|
||||||
arch=$(getconf LONG_BIT)
|
|
||||||
|
|
||||||
case "${arch}" in
|
|
||||||
32)
|
|
||||||
byte_size=4
|
|
||||||
magic_word="CABA1002"
|
|
||||||
;;
|
|
||||||
64)
|
|
||||||
byte_size=8
|
|
||||||
magic_word="00000000CABA1002"
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
error_message "Unknown architecture (long bit): ${arch}"
|
|
||||||
exit 2
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
# This is the logic to parse the binary format of 01-index.cache.
|
|
||||||
# The first word is a magic 'caba1002', the second one is the timestamp in unix epoch.
|
|
||||||
# Better than copying the cabal-install source code.
|
|
||||||
if [ "$(xxd -u -p -l${byte_size} -s 0 "${CACHE_LOCATION}")" != "${magic_word}" ] ; then
|
|
||||||
error_message "Magic word does not match!"
|
|
||||||
exit 4
|
|
||||||
fi
|
|
||||||
cache_timestamp=$(echo "ibase=16;obase=A;$(xxd -u -p -l${byte_size} -s ${byte_size} "${CACHE_LOCATION}")" | bc)
|
|
||||||
|
|
||||||
# If we got junk from the binary file, this should fail.
|
|
||||||
cache_date=$(date --utc --date "@${cache_timestamp}" "+%FT%TZ")
|
|
||||||
|
|
||||||
|
|
||||||
status_message "Updating index state in ${SCRIPTPATH}/cabal.project"
|
|
||||||
|
|
||||||
if grep -q "^index-state: .*" "${SCRIPTPATH}/cabal.project" ; then
|
|
||||||
awk '/index-state:/ {gsub(/.*/, "index-state: '${cache_date}'")}; { print }' "${SCRIPTPATH}/cabal.project" > "${SCRIPTPATH}/cabal.project.tmp"
|
|
||||||
mv "${SCRIPTPATH}/cabal.project.tmp" "${SCRIPTPATH}/cabal.project"
|
|
||||||
else
|
|
||||||
printf "index-state: %s\n" "${cache_date}" >> "${SCRIPTPATH}/cabal.project"
|
|
||||||
fi
|
|
||||||
|
|
||||||
208
www/LICENSE
Normal file
208
www/LICENSE
Normal file
@@ -0,0 +1,208 @@
|
|||||||
|
The ghcup website, excluding ghcup, fonts, bootstrap-haskell script are subject
|
||||||
|
to the license below. Design, javascript and css are used from the rustup
|
||||||
|
project: https://github.com/rust-lang/rustup.rs/tree/master/www
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
===============================================================================
|
||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work.
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following
|
||||||
|
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||||
|
replaced with your own identifying information. (Don't include
|
||||||
|
the brackets!) The text should be enclosed in the appropriate
|
||||||
|
comment syntax for the file format. We also recommend that a
|
||||||
|
file or class name and description of purpose be included on the
|
||||||
|
same "printed page" as the copyright notice for easier
|
||||||
|
identification within third-party archives.
|
||||||
|
|
||||||
|
Copyright [yyyy] [name of copyright owner]
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
1
www/copy.svg
Normal file
1
www/copy.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" focusable="false" width="1.66em" height="2em" style="-ms-transform: rotate(360deg); -webkit-transform: rotate(360deg); transform: rotate(360deg);" preserveAspectRatio="xMidYMid meet" viewBox="0 0 14 16"><path fill-rule="evenodd" d="M2 13h4v1H2v-1zm5-6H2v1h5V7zm2 3V8l-3 3l3 3v-2h5v-2H9zM4.5 9H2v1h2.5V9zM2 12h2.5v-1H2v1zm9 1h1v2c-.02.28-.11.52-.3.7c-.19.18-.42.28-.7.3H1c-.55 0-1-.45-1-1V4c0-.55.45-1 1-1h3c0-1.11.89-2 2-2c1.11 0 2 .89 2 2h3c.55 0 1 .45 1 1v5h-1V6H1v9h10v-2zM2 5h8c0-.55-.45-1-1-1H8c-.55 0-1-.45-1-1s-.45-1-1-1s-1 .45-1 1s-.45 1-1 1H3c-.55 0-1 .45-1 1z" fill="#626262"/></svg>
|
||||||
|
After Width: | Height: | Size: 696 B |
BIN
www/fonts/FiraSans-Light.woff
Normal file
BIN
www/fonts/FiraSans-Light.woff
Normal file
Binary file not shown.
BIN
www/fonts/FiraSans-Medium.woff
Normal file
BIN
www/fonts/FiraSans-Medium.woff
Normal file
Binary file not shown.
BIN
www/fonts/FiraSans-Regular.woff
Normal file
BIN
www/fonts/FiraSans-Regular.woff
Normal file
Binary file not shown.
BIN
www/fonts/Inconsolata-Regular.ttf
Normal file
BIN
www/fonts/Inconsolata-Regular.ttf
Normal file
Binary file not shown.
92
www/fonts/OFL.txt
Normal file
92
www/fonts/OFL.txt
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
Copyright (c) 2011, Raph Levien (firstname.lastname@gmail.com), Copyright (c) 2012, Cyreal (cyreal.org)
|
||||||
|
This Font Software is licensed under the SIL Open Font License, Version 1.1.
|
||||||
|
This license is copied below, and is also available with a FAQ at:
|
||||||
|
http://scripts.sil.org/OFL
|
||||||
|
|
||||||
|
|
||||||
|
-----------------------------------------------------------
|
||||||
|
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
|
||||||
|
-----------------------------------------------------------
|
||||||
|
|
||||||
|
PREAMBLE
|
||||||
|
The goals of the Open Font License (OFL) are to stimulate worldwide
|
||||||
|
development of collaborative font projects, to support the font creation
|
||||||
|
efforts of academic and linguistic communities, and to provide a free and
|
||||||
|
open framework in which fonts may be shared and improved in partnership
|
||||||
|
with others.
|
||||||
|
|
||||||
|
The OFL allows the licensed fonts to be used, studied, modified and
|
||||||
|
redistributed freely as long as they are not sold by themselves. The
|
||||||
|
fonts, including any derivative works, can be bundled, embedded,
|
||||||
|
redistributed and/or sold with any software provided that any reserved
|
||||||
|
names are not used by derivative works. The fonts and derivatives,
|
||||||
|
however, cannot be released under any other type of license. The
|
||||||
|
requirement for fonts to remain under this license does not apply
|
||||||
|
to any document created using the fonts or their derivatives.
|
||||||
|
|
||||||
|
DEFINITIONS
|
||||||
|
"Font Software" refers to the set of files released by the Copyright
|
||||||
|
Holder(s) under this license and clearly marked as such. This may
|
||||||
|
include source files, build scripts and documentation.
|
||||||
|
|
||||||
|
"Reserved Font Name" refers to any names specified as such after the
|
||||||
|
copyright statement(s).
|
||||||
|
|
||||||
|
"Original Version" refers to the collection of Font Software components as
|
||||||
|
distributed by the Copyright Holder(s).
|
||||||
|
|
||||||
|
"Modified Version" refers to any derivative made by adding to, deleting,
|
||||||
|
or substituting -- in part or in whole -- any of the components of the
|
||||||
|
Original Version, by changing formats or by porting the Font Software to a
|
||||||
|
new environment.
|
||||||
|
|
||||||
|
"Author" refers to any designer, engineer, programmer, technical
|
||||||
|
writer or other person who contributed to the Font Software.
|
||||||
|
|
||||||
|
PERMISSION & CONDITIONS
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of the Font Software, to use, study, copy, merge, embed, modify,
|
||||||
|
redistribute, and sell modified and unmodified copies of the Font
|
||||||
|
Software, subject to the following conditions:
|
||||||
|
|
||||||
|
1) Neither the Font Software nor any of its individual components,
|
||||||
|
in Original or Modified Versions, may be sold by itself.
|
||||||
|
|
||||||
|
2) Original or Modified Versions of the Font Software may be bundled,
|
||||||
|
redistributed and/or sold with any software, provided that each copy
|
||||||
|
contains the above copyright notice and this license. These can be
|
||||||
|
included either as stand-alone text files, human-readable headers or
|
||||||
|
in the appropriate machine-readable metadata fields within text or
|
||||||
|
binary files as long as those fields can be easily viewed by the user.
|
||||||
|
|
||||||
|
3) No Modified Version of the Font Software may use the Reserved Font
|
||||||
|
Name(s) unless explicit written permission is granted by the corresponding
|
||||||
|
Copyright Holder. This restriction only applies to the primary font name as
|
||||||
|
presented to the users.
|
||||||
|
|
||||||
|
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
|
||||||
|
Software shall not be used to promote, endorse or advertise any
|
||||||
|
Modified Version, except to acknowledge the contribution(s) of the
|
||||||
|
Copyright Holder(s) and the Author(s) or with their explicit written
|
||||||
|
permission.
|
||||||
|
|
||||||
|
5) The Font Software, modified or unmodified, in part or in whole,
|
||||||
|
must be distributed entirely under this license, and must not be
|
||||||
|
distributed under any other license. The requirement for fonts to
|
||||||
|
remain under this license does not apply to any document created
|
||||||
|
using the Font Software.
|
||||||
|
|
||||||
|
TERMINATION
|
||||||
|
This license becomes null and void if any of the above conditions are
|
||||||
|
not met.
|
||||||
|
|
||||||
|
DISCLAIMER
|
||||||
|
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
|
||||||
|
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
|
||||||
|
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||||
|
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
|
||||||
|
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||||
|
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
|
||||||
|
OTHER DEALINGS IN THE FONT SOFTWARE.
|
||||||
BIN
www/fonts/WorkSans-Medium.ttf
Normal file
BIN
www/fonts/WorkSans-Medium.ttf
Normal file
Binary file not shown.
246
www/ghcup.css
Normal file
246
www/ghcup.css
Normal file
@@ -0,0 +1,246 @@
|
|||||||
|
@font-face {
|
||||||
|
font-family: 'Fira Sans';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 300;
|
||||||
|
src: local('Fira Sans Light'), url("fonts/FiraSans-Light.woff") format('woff');
|
||||||
|
}
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Fira Sans';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 400;
|
||||||
|
src: local('Fira Sans'), url("fonts/FiraSans-Regular.woff") format('woff');
|
||||||
|
}
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Fira Sans';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 500;
|
||||||
|
src: local('Fira Sans Medium'), url("fonts/FiraSans-Medium.woff") format('woff');
|
||||||
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Work Sans';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 500;
|
||||||
|
src: local('Work Sans Medium'), url("fonts/WorkSans-Medium.ttf") format('ttf');
|
||||||
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Inconsolata';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 400;
|
||||||
|
src: local('Inconsolata Regular'), url("fonts/Inconsolata-Regular.ttf") format('ttf');
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
margin-top: 2em;
|
||||||
|
background-color: white;
|
||||||
|
color: #515151;
|
||||||
|
font-family: "Fira Sans","Helvetica Neue",Helvetica,Arial,sans-serif;
|
||||||
|
font-weight: 300;
|
||||||
|
font-size: 25px;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre {
|
||||||
|
font-family: Inconsolata,Menlo,Monaco,Consolas,"Courier New",monospace;
|
||||||
|
font-weight: 400;
|
||||||
|
}
|
||||||
|
|
||||||
|
body#idx #pitch > a {
|
||||||
|
font-weight: 500;
|
||||||
|
line-height: 2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: #428bca;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover {
|
||||||
|
color: rgb(42, 100, 150);
|
||||||
|
}
|
||||||
|
|
||||||
|
body#idx > * {
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
text-align: center;
|
||||||
|
width: 37em;
|
||||||
|
}
|
||||||
|
|
||||||
|
body#idx > #pitch {
|
||||||
|
width: 30rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
#pitch em {
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 400;
|
||||||
|
}
|
||||||
|
|
||||||
|
body#idx p {
|
||||||
|
margin-top: 2em;
|
||||||
|
margin-bottom: 2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
body#idx p.other-help {
|
||||||
|
font-size: 0.6em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.instructions {
|
||||||
|
background-color: rgb(250, 250, 250);
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
border-radius: 3px;
|
||||||
|
border: 1px solid rgb(204, 204, 204);
|
||||||
|
box-shadow: 0px 1px 4px 0px rgb(204, 204, 204);
|
||||||
|
}
|
||||||
|
|
||||||
|
.instructions > * {
|
||||||
|
width: 55rem;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.instructions div.command-button {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.instructions div.command-button button {
|
||||||
|
color: white;
|
||||||
|
/* border: none; */
|
||||||
|
background-color: rgb(242, 242, 242);
|
||||||
|
border-width: 2px;
|
||||||
|
border-style: solid;
|
||||||
|
border-radius: 3px;
|
||||||
|
|
||||||
|
margin-left: 0.5rem;
|
||||||
|
margin-right: auto;
|
||||||
|
margin-top: 25px;
|
||||||
|
margin-bottom: 25px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.instructions div.command-button button:hover {
|
||||||
|
background: rgb(232, 232, 232);
|
||||||
|
}
|
||||||
|
|
||||||
|
.instructions div.command-button button:focus {
|
||||||
|
background: rgb(222, 222, 222);
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
margin-top: 2em;
|
||||||
|
margin-bottom: 2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.code {
|
||||||
|
font-family: 'Lucida Console', monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
#platform-instructions-linux div > pre,
|
||||||
|
#platform-instructions-mac div > pre,
|
||||||
|
#platform-instructions-freebsd div > pre,
|
||||||
|
#platform-instructions-win32 div > pre,
|
||||||
|
#platform-instructions-win64 div > pre,
|
||||||
|
#platform-instructions-default div > div > pre,
|
||||||
|
#platform-instructions-unknown div > div > pre {
|
||||||
|
background-color: #515151;
|
||||||
|
color: white;
|
||||||
|
margin-left: auto;
|
||||||
|
padding-top: 1rem;
|
||||||
|
padding-bottom: 1rem;
|
||||||
|
padding-right: 1rem;
|
||||||
|
text-align: center;
|
||||||
|
border-radius: 3px;
|
||||||
|
box-shadow: inset 0px 0px 20px 0px #333333;
|
||||||
|
font-size: 0.6em;
|
||||||
|
width: 40rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
#platform-instructions-win32 a.windows-download,
|
||||||
|
#platform-instructions-win64 a.windows-download,
|
||||||
|
#platform-instructions-default a.windows-download,
|
||||||
|
#platform-instructions-unknown a.windows-download {
|
||||||
|
display: block;
|
||||||
|
padding-top: 0.4rem;
|
||||||
|
padding-bottom: 0.6rem;
|
||||||
|
font-family: "Work Sans", "Fira Sans","Helvetica Neue",Helvetica,Arial,sans-serif;
|
||||||
|
font-weight: 500;
|
||||||
|
letter-spacing: 0.1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is the box that prints navigator.platform, navigator.appVersion values */
|
||||||
|
#platform-instructions-unknown > div:first-of-type {
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
#about {
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
#about > img {
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
transform: translateY(11px);
|
||||||
|
}
|
||||||
|
|
||||||
|
#platform-button {
|
||||||
|
background-color: #515151;
|
||||||
|
color: white;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
padding: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ghcup-command:before {
|
||||||
|
color: #999;
|
||||||
|
content: " $ ";
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tooltip container */
|
||||||
|
.tooltip {
|
||||||
|
position: relative;
|
||||||
|
display: inline-block;
|
||||||
|
/* border-bottom: 1px dotted black; [> If you want dots under the hoverable text <] */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tooltip text */
|
||||||
|
.tooltip .tooltiptext {
|
||||||
|
visibility: hidden;
|
||||||
|
width: 120px;
|
||||||
|
background-color: #555;
|
||||||
|
color: #fff;
|
||||||
|
text-align: center;
|
||||||
|
padding: 5px 0;
|
||||||
|
border-radius: 6px;
|
||||||
|
|
||||||
|
/* Position the tooltip text */
|
||||||
|
position: absolute;
|
||||||
|
z-index: 1;
|
||||||
|
bottom: 125%;
|
||||||
|
left: 50%;
|
||||||
|
margin-left: -60px;
|
||||||
|
|
||||||
|
/* Fade in tooltip */
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tooltip arrow */
|
||||||
|
.tooltip .tooltiptext::after {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
top: 100%;
|
||||||
|
left: 50%;
|
||||||
|
margin-left: -5px;
|
||||||
|
border-width: 5px;
|
||||||
|
border-style: solid;
|
||||||
|
border-color: #555 transparent transparent transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Show the tooltip text when you mouse over the tooltip container */
|
||||||
|
.tooltip:hover .tooltiptext {
|
||||||
|
visibility: visible;
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
176
www/ghcup.js
Normal file
176
www/ghcup.js
Normal file
@@ -0,0 +1,176 @@
|
|||||||
|
var platforms = ["default", "unknown", "win32", "win64", "linux", "freebsd", "mac"];
|
||||||
|
var platform_override = null;
|
||||||
|
|
||||||
|
function detect_platform() {
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
if (platform_override !== null) {
|
||||||
|
return platforms[platform_override];
|
||||||
|
}
|
||||||
|
|
||||||
|
var os = "unknown";
|
||||||
|
|
||||||
|
if (navigator.platform == "Linux x86_64") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux i686") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux i686 on x86_64") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux aarch64") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux armv6l") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux armv7l") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux armv8l") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux ppc64") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux mips") {os = "linux";}
|
||||||
|
if (navigator.platform == "Linux mips64") {os = "linux";}
|
||||||
|
if (navigator.platform == "Mac") {os = "mac";}
|
||||||
|
if (navigator.platform == "Win32") {os = "win32";}
|
||||||
|
if (navigator.platform == "Win64" ||
|
||||||
|
navigator.userAgent.indexOf("WOW64") != -1 ||
|
||||||
|
navigator.userAgent.indexOf("Win64") != -1) { os = "win64"; }
|
||||||
|
if (navigator.platform == "FreeBSD x86_64") {os = "freebsd";}
|
||||||
|
if (navigator.platform == "FreeBSD amd64") {os = "freebsd";}
|
||||||
|
// if (navigator.platform == "NetBSD x86_64") {os = "unix";}
|
||||||
|
// if (navigator.platform == "NetBSD amd64") {os = "unix";}
|
||||||
|
|
||||||
|
// I wish I knew by now, but I don't. Try harder.
|
||||||
|
if (os == "unknown") {
|
||||||
|
if (navigator.appVersion.indexOf("Win")!=-1) {os = "win32";}
|
||||||
|
if (navigator.appVersion.indexOf("Mac")!=-1) {os = "mac";}
|
||||||
|
if (navigator.appVersion.indexOf("FreeBSD")!=-1) {os = "freebsd";}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Firefox Quantum likes to hide platform and appVersion but oscpu works
|
||||||
|
if (navigator.oscpu) {
|
||||||
|
if (navigator.oscpu.indexOf("Win32")!=-1) {os = "win32";}
|
||||||
|
if (navigator.oscpu.indexOf("Win64")!=-1) {os = "win64";}
|
||||||
|
if (navigator.oscpu.indexOf("Mac")!=-1) {os = "mac";}
|
||||||
|
if (navigator.oscpu.indexOf("Linux")!=-1) {os = "linux";}
|
||||||
|
if (navigator.oscpu.indexOf("FreeBSD")!=-1) {os = "freebsd";}
|
||||||
|
// if (navigator.oscpu.indexOf("NetBSD")!=-1) {os = "unix";}
|
||||||
|
}
|
||||||
|
|
||||||
|
return os;
|
||||||
|
}
|
||||||
|
|
||||||
|
function adjust_for_platform() {
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var platform = detect_platform();
|
||||||
|
|
||||||
|
platforms.forEach(function (platform_elem) {
|
||||||
|
var platform_div = document.getElementById("platform-instructions-" + platform_elem);
|
||||||
|
platform_div.style.display = "none";
|
||||||
|
if (platform == platform_elem) {
|
||||||
|
platform_div.style.display = "block";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
adjust_platform_specific_instrs(platform);
|
||||||
|
}
|
||||||
|
|
||||||
|
function adjust_platform_specific_instrs(platform) {
|
||||||
|
var platform_specific = document.getElementsByClassName("platform-specific");
|
||||||
|
for (var el of platform_specific) {
|
||||||
|
var el_is_not_win = el.className.indexOf("not-win") !== -1;
|
||||||
|
var el_is_inline = el.tagName.toLowerCase() == "span";
|
||||||
|
var el_visible_style = "block";
|
||||||
|
if (el_is_inline) {
|
||||||
|
el_visible_style = "inline";
|
||||||
|
}
|
||||||
|
if (platform == "win64" || platform == "win32") {
|
||||||
|
if (el_is_not_win) {
|
||||||
|
el.style.display = "none";
|
||||||
|
} else {
|
||||||
|
el.style.display = el_visible_style;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (el_is_not_win) {
|
||||||
|
el.style.display = el_visible_style;
|
||||||
|
} else {
|
||||||
|
el.style.display = "none";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function cycle_platform() {
|
||||||
|
if (platform_override == null) {
|
||||||
|
platform_override = 0;
|
||||||
|
} else {
|
||||||
|
platform_override = (platform_override + 1) % platforms.length;
|
||||||
|
}
|
||||||
|
adjust_for_platform();
|
||||||
|
}
|
||||||
|
|
||||||
|
function set_up_cycle_button() {
|
||||||
|
var cycle_button = document.getElementById("platform-button");
|
||||||
|
cycle_button.onclick = cycle_platform;
|
||||||
|
|
||||||
|
var key="test";
|
||||||
|
var idx=0;
|
||||||
|
var unlocked=false;
|
||||||
|
|
||||||
|
document.onkeypress = function(event) {
|
||||||
|
if (event.key == "n" && unlocked) {
|
||||||
|
cycle_platform();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key == key[idx]) {
|
||||||
|
idx += 1;
|
||||||
|
|
||||||
|
if (idx == key.length) {
|
||||||
|
cycle_button.style.display = "block";
|
||||||
|
unlocked = true;
|
||||||
|
cycle_platform();
|
||||||
|
}
|
||||||
|
} else if (event.key == key[0]) {
|
||||||
|
idx = 1;
|
||||||
|
} else {
|
||||||
|
idx = 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function go_to_default_platform() {
|
||||||
|
platform_override = 0;
|
||||||
|
adjust_for_platform();
|
||||||
|
}
|
||||||
|
|
||||||
|
function set_up_default_platform_buttons() {
|
||||||
|
var defaults_buttons = document.getElementsByClassName('default-platform-button');
|
||||||
|
for (var i = 0; i < defaults_buttons.length; i++) {
|
||||||
|
defaults_buttons[i].onclick = go_to_default_platform;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function fill_in_bug_report_values() {
|
||||||
|
var nav_plat = document.getElementById("nav-plat");
|
||||||
|
var nav_app = document.getElementById("nav-app");
|
||||||
|
nav_plat.textContent = navigator.platform;
|
||||||
|
nav_app.textContent = navigator.appVersion;
|
||||||
|
}
|
||||||
|
|
||||||
|
function copyToClipboard() {
|
||||||
|
const text = document.getElementById("ghcup-command-normal").innerText;
|
||||||
|
const el = document.createElement('textarea');
|
||||||
|
el.value = text;
|
||||||
|
document.body.appendChild(el);
|
||||||
|
el.select();
|
||||||
|
document.execCommand('copy');
|
||||||
|
document.body.removeChild(el);
|
||||||
|
}
|
||||||
|
|
||||||
|
function copyToClipboardPowershell() {
|
||||||
|
const text = document.getElementById("ghcup-command-powershell").innerText;
|
||||||
|
const el = document.createElement('textarea');
|
||||||
|
el.value = text;
|
||||||
|
document.body.appendChild(el);
|
||||||
|
el.select();
|
||||||
|
document.execCommand('copy');
|
||||||
|
document.body.removeChild(el);
|
||||||
|
}
|
||||||
|
|
||||||
|
(function () {
|
||||||
|
adjust_for_platform();
|
||||||
|
set_up_cycle_button();
|
||||||
|
set_up_default_platform_buttons();
|
||||||
|
fill_in_bug_report_values();
|
||||||
|
}());
|
||||||
66
www/haskell-logo.svg
Normal file
66
www/haskell-logo.svg
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<svg
|
||||||
|
xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
xmlns:cc="http://creativecommons.org/ns#"
|
||||||
|
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||||
|
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||||
|
width="109"
|
||||||
|
height="80"
|
||||||
|
viewBox="0 0 109.00001 80"
|
||||||
|
version="1.1"
|
||||||
|
id="svg14"
|
||||||
|
sodipodi:docname="haskell-logo.svg"
|
||||||
|
inkscape:version="0.92.3 (2405546, 2018-03-11)">
|
||||||
|
<metadata
|
||||||
|
id="metadata20">
|
||||||
|
<rdf:RDF>
|
||||||
|
<cc:Work
|
||||||
|
rdf:about="">
|
||||||
|
<dc:format>image/svg+xml</dc:format>
|
||||||
|
<dc:type
|
||||||
|
rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
|
||||||
|
<dc:title></dc:title>
|
||||||
|
</cc:Work>
|
||||||
|
</rdf:RDF>
|
||||||
|
</metadata>
|
||||||
|
<defs
|
||||||
|
id="defs18" />
|
||||||
|
<sodipodi:namedview
|
||||||
|
pagecolor="#ffffff"
|
||||||
|
bordercolor="#666666"
|
||||||
|
borderopacity="1"
|
||||||
|
objecttolerance="10"
|
||||||
|
gridtolerance="10"
|
||||||
|
guidetolerance="10"
|
||||||
|
inkscape:pageopacity="0"
|
||||||
|
inkscape:pageshadow="2"
|
||||||
|
inkscape:window-width="1916"
|
||||||
|
inkscape:window-height="1033"
|
||||||
|
id="namedview16"
|
||||||
|
showgrid="false"
|
||||||
|
inkscape:zoom="2.0159478"
|
||||||
|
inkscape:cx="298.15447"
|
||||||
|
inkscape:cy="-2.7202801"
|
||||||
|
inkscape:window-x="1366"
|
||||||
|
inkscape:window-y="22"
|
||||||
|
inkscape:window-maximized="0"
|
||||||
|
inkscape:current-layer="svg14" />
|
||||||
|
<path
|
||||||
|
d="M 1.842,77.722 26.586,40.63 1.842,3.537 H 20.4 L 45.144,40.63 20.4,77.722 Z m 0,0"
|
||||||
|
id="path8"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
style="fill:#453a62" />
|
||||||
|
<path
|
||||||
|
d="M 26.586,77.722 51.33,40.63 26.586,3.537 H 45.144 L 94.63,77.722 H 76.074 L 60.61,54.54 45.143,77.722 Z m 0,0"
|
||||||
|
id="path10"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
style="fill:#5e5086" />
|
||||||
|
<path
|
||||||
|
d="M 86.384,56.085 78.136,43.72 h 28.868 V 56.086 H 86.384 Z M 74.012,37.54 65.764,25.175 h 41.24 V 37.54 Z m 0,0"
|
||||||
|
id="path12"
|
||||||
|
inkscape:connector-curvature="0"
|
||||||
|
style="fill:#8f4e8b" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.1 KiB |
154
www/index.html
Normal file
154
www/index.html
Normal file
@@ -0,0 +1,154 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>ghcup - The Haskell (GHC) toolchain installer</title>
|
||||||
|
<meta name="keywords" content="Haskell, Haskell programming language, ghc, ghcup">
|
||||||
|
<meta name="description" content="The Haskell (GHC) toolchain installer">
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="normalize.css">
|
||||||
|
<link rel="stylesheet" href="ghcup.css">
|
||||||
|
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body id="idx">
|
||||||
|
|
||||||
|
<a id="platform-button" style="display: none;" href="#">
|
||||||
|
click or press "n" to cycle platforms
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<p id="pitch">
|
||||||
|
<em>ghcup</em> is an installer for<br/>
|
||||||
|
the general purpose language <a href="https://www.haskell.org/">Haskell</a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div id="platform-instructions-linux" class="instructions" style="display: none;">
|
||||||
|
<p>Run the following in your terminal (as a user other than root), then follow the onscreen instructions.</p>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>curl --proto '=https' --tlsv1.2 -sSf https://get-ghcup.haskell.org | sh</span></pre><button class="tooltip" onclick="copyToClipboard()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button></div>
|
||||||
|
<p class="other-help">If you don't like curl | sh, see <a href="https://gitlab.haskell.org/haskell/ghcup-hs#manual-install">other installation methods</a>.<br/>You appear to be running Linux. If not, <a class="default-platform-button" href="#">display all supported installers</a>.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="platform-instructions-mac" class="instructions" style="display: none;">
|
||||||
|
<p>Run the following in your terminal (as a user other than root), then follow the onscreen instructions.</p>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command' id="ghcup-command-normal">curl --proto '=https' --tlsv1.2 -sSf https://get-ghcup.haskell.org | sh</span></pre><button class="tooltip" onclick="copyToClipboard()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button></div>
|
||||||
|
<p class="other-help">If you don't like curl | sh, see <a href="https://gitlab.haskell.org/haskell/ghcup-hs#manual-install">other installation methods</a>.<br/>You appear to be running macOS. If not, <a class="default-platform-button" href="#">display all supported installers</a>.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="platform-instructions-freebsd" class="instructions" style="display: none;">
|
||||||
|
<p>Run the following in your terminal (as a user other than root), then follow the onscreen instructions.</p>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>curl --proto '=https' --tlsv1.2 -sSf https://get-ghcup.haskell.org | sh</span></pre><button class="tooltip" onclick="copyToClipboard()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button></div>
|
||||||
|
<p class="other-help">If you don't like curl | sh, see <a href="https://gitlab.haskell.org/haskell/ghcup-hs#manual-install">other installation methods</a>.<br/>You appear to be running FreeBSD. If not, <a class="default-platform-button" href="#">display all supported installers</a>.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="platform-instructions-win32" class="instructions">
|
||||||
|
<p>
|
||||||
|
To install Haskell,<br/>run the following in a powershell session (as a non-admin user).
|
||||||
|
<div>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command' id="ghcup-command-powershell">Set-ExecutionPolicy Bypass -Scope Process -Force;[System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072;Invoke-Command -ScriptBlock ([ScriptBlock]::Create((Invoke-WebRequest https://www.haskell.org/ghcup/sh/bootstrap-haskell.ps1 -UseBasicParsing))) -ArgumentList $false</span></span></pre><button class="tooltip" onclick="copyToClipboardPowershell()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button>
|
||||||
|
</div>
|
||||||
|
<p class="other-help">If you want to run an interactive installation, change <span class='code'>$false</span> to <span class='code'>$true</span> at the end of the script.</p>
|
||||||
|
</div>
|
||||||
|
<p>If you're a Windows Subsystem 2 for Linux user run the following in your terminal, then follow the onscreen instructions to install Haskell.
|
||||||
|
</p>
|
||||||
|
<div>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>curl --proto '=https' --tlsv1.2 -sSf https://get-ghcup.haskell.org | sh</span></pre><button class="tooltip" onclick="copyToClipboard()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button>
|
||||||
|
</div>
|
||||||
|
<p class="other-help">WSL1 does not work with ghcup, follow <a href="https://docs.microsoft.com/en-us/windows/wsl/install-win10">the instructions here</a> to upgrade to WSL2 if needed.</p>
|
||||||
|
</div>
|
||||||
|
</p>
|
||||||
|
<hr/>
|
||||||
|
<p class="other-help">You appear to be running Windows 32-bit. If not, <a class="default-platform-button" href="#">display all supported installers</a>.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="platform-instructions-win64" class="instructions" style="display: none;">
|
||||||
|
<p>
|
||||||
|
To install Haskell,<br/>run the following in a powershell session (as a non-admin user).
|
||||||
|
<div>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>Set-ExecutionPolicy Bypass -Scope Process -Force;[System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072;Invoke-Command -ScriptBlock ([ScriptBlock]::Create((Invoke-WebRequest https://www.haskell.org/ghcup/sh/bootstrap-haskell.ps1 -UseBasicParsing))) -ArgumentList $false</span></span></pre><button class="tooltip" onclick="copyToClipboardPowershell()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button>
|
||||||
|
</div>
|
||||||
|
<p class="other-help">If you want to run an interactive installation, change <span class='code'>$false</span> to <span class='code'>$true</span> at the end of the script.</p>
|
||||||
|
</div>
|
||||||
|
</p>
|
||||||
|
<p>If you're a Windows Subsystem 2 for Linux user run the following in your terminal, then follow the onscreen instructions to install Haskell.
|
||||||
|
</p>
|
||||||
|
<div>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>curl --proto '=https' --tlsv1.2 -sSf https://get-ghcup.haskell.org | sh</span></pre><button class="tooltip" onclick="copyToClipboard()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button>
|
||||||
|
</div>
|
||||||
|
<p class="other-help">WSL1 does not work with ghcup, follow <a href="https://docs.microsoft.com/en-us/windows/wsl/install-win10">the instructions here</a> to upgrade to WSL2 if needed.</p>
|
||||||
|
</div>
|
||||||
|
<p class="other-help">You appear to be running Windows 64-bit. If not, <a class="default-platform-button" href="#">display all supported installers</a>.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="platform-instructions-unknown" class="instructions" style="display: none;">
|
||||||
|
<!-- unrecognized platform: ask for help -->
|
||||||
|
<p>I don't recognize your platform.</p>
|
||||||
|
<p>
|
||||||
|
ghcup runs on Linux, macOS and FreeBSD. If
|
||||||
|
you are on one of these platforms and are seeing this then please
|
||||||
|
<a href="https://gitlab.haskell.org/haskell/ghcup-hs/issues">report an issue</a>,
|
||||||
|
along with the following values:
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<div>navigator.platform:</div>
|
||||||
|
<div id="nav-plat"></div>
|
||||||
|
<div>navigator.appVersion:</div>
|
||||||
|
<div id="nav-app"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- duplicate the default cross-platform instructions -->
|
||||||
|
<div>
|
||||||
|
<p>If you are running Linux, macOS, FreeBSD or Windows Subsystem 2 for Linux, run the following in your terminal (as a user other than root), then follow the onscreen instructions.</p>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>curl --proto '=https' --tlsv1.2 -sSf https://get-ghcup.haskell.org | sh</span></pre><button class="tooltip" onclick="copyToClipboard()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button></div>
|
||||||
|
<p class="other-help">If you don't like curl | sh, see <a href="https://gitlab.haskell.org/haskell/ghcup-hs#manual-install">other installation methods</a>.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<p>
|
||||||
|
If you are running Windows,<br/>run the following in a powershell session (as a non-admin user).
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>Set-ExecutionPolicy Bypass -Scope Process -Force;[System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072;Invoke-Command -ScriptBlock ([ScriptBlock]::Create((Invoke-WebRequest https://www.haskell.org/ghcup/sh/bootstrap-haskell.ps1 -UseBasicParsing))) -ArgumentList $false</span></span></pre><button class="tooltip" onclick="copyToClipboardPowershell()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button></div>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="platform-instructions-default" class="instructions">
|
||||||
|
<div>
|
||||||
|
<p>To install Haskell, if you are running Linux, macOS, FreeBSD or Windows Subsystem 2 for Linux, run the following
|
||||||
|
in your terminal (as a user other than root), then follow the onscreen instructions.</p>
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>curl --proto '=https' --tlsv1.2 -sSf https://get-ghcup.haskell.org | sh</span></pre><button class="tooltip" onclick="copyToClipboard()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button></div>
|
||||||
|
<p class="other-help">If you don't like curl | sh, see <a href="https://gitlab.haskell.org/haskell/ghcup-hs#manual-install">other installation methods</a>.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr/>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<p>
|
||||||
|
If you are running Windows,<br/>run the following in a powershell session (as a non-admin user).
|
||||||
|
<div class="command-button"><pre><span class='ghcup-command'>Set-ExecutionPolicy Bypass -Scope Process -Force;[System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072;Invoke-Command -ScriptBlock ([ScriptBlock]::Create((Invoke-WebRequest https://www.haskell.org/ghcup/sh/bootstrap-haskell.ps1 -UseBasicParsing))) -ArgumentList $false</span></span></pre><button class="tooltip" onclick="copyToClipboardPowershell()"><img src="copy.svg" alt="" /><span class="tooltiptext">Copy to clipboard</span></button></div>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Need help? Ask on <a href="https://kiwiirc.com/nextclient/irc.libera.chat/#haskell-ghcup">#haskell-ghcup</a>, <a href="https://kiwiirc.com/nextclient/irc.libera.chat/#haskell">#haskell</a> or <a href="https://gitlab.haskell.org/haskell/ghcup-hs/issues">report a bug</a>.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p id="about">
|
||||||
|
<img src="haskell-logo.svg" alt="" />
|
||||||
|
ghcup is a haskell.org hosted project.
|
||||||
|
<br/>
|
||||||
|
<a href="https://www.haskell.org/downloads/">other installation options</a>
|
||||||
|
·
|
||||||
|
<a href="https://gitlab.haskell.org/haskell/ghcup-hs">about ghcup</a>
|
||||||
|
·
|
||||||
|
<a href="https://github.com/rust-lang/rustup.rs/tree/master/www">web design from rustup</a>
|
||||||
|
</p>
|
||||||
|
<script type="text/javascript" src="ghcup.js"></script>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
427
www/normalize.css
vendored
Normal file
427
www/normalize.css
vendored
Normal file
@@ -0,0 +1,427 @@
|
|||||||
|
/*! normalize.css v3.0.2 | MIT License | git.io/normalize */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 1. Set default font family to sans-serif.
|
||||||
|
* 2. Prevent iOS text size adjust after orientation change, without disabling
|
||||||
|
* user zoom.
|
||||||
|
*/
|
||||||
|
|
||||||
|
html {
|
||||||
|
font-family: sans-serif; /* 1 */
|
||||||
|
-ms-text-size-adjust: 100%; /* 2 */
|
||||||
|
-webkit-text-size-adjust: 100%; /* 2 */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove default margin.
|
||||||
|
*/
|
||||||
|
|
||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* HTML5 display definitions
|
||||||
|
========================================================================== */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Correct `block` display not defined for any HTML5 element in IE 8/9.
|
||||||
|
* Correct `block` display not defined for `details` or `summary` in IE 10/11
|
||||||
|
* and Firefox.
|
||||||
|
* Correct `block` display not defined for `main` in IE 11.
|
||||||
|
*/
|
||||||
|
|
||||||
|
article,
|
||||||
|
aside,
|
||||||
|
details,
|
||||||
|
figcaption,
|
||||||
|
figure,
|
||||||
|
footer,
|
||||||
|
header,
|
||||||
|
hgroup,
|
||||||
|
main,
|
||||||
|
menu,
|
||||||
|
nav,
|
||||||
|
section,
|
||||||
|
summary {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 1. Correct `inline-block` display not defined in IE 8/9.
|
||||||
|
* 2. Normalize vertical alignment of `progress` in Chrome, Firefox, and Opera.
|
||||||
|
*/
|
||||||
|
|
||||||
|
audio,
|
||||||
|
canvas,
|
||||||
|
progress,
|
||||||
|
video {
|
||||||
|
display: inline-block; /* 1 */
|
||||||
|
vertical-align: baseline; /* 2 */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prevent modern browsers from displaying `audio` without controls.
|
||||||
|
* Remove excess height in iOS 5 devices.
|
||||||
|
*/
|
||||||
|
|
||||||
|
audio:not([controls]) {
|
||||||
|
display: none;
|
||||||
|
height: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address `[hidden]` styling not present in IE 8/9/10.
|
||||||
|
* Hide the `template` element in IE 8/9/11, Safari, and Firefox < 22.
|
||||||
|
*/
|
||||||
|
|
||||||
|
[hidden],
|
||||||
|
template {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Links
|
||||||
|
========================================================================== */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove the gray background color from active links in IE 10.
|
||||||
|
*/
|
||||||
|
|
||||||
|
a {
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Improve readability when focused and also mouse hovered in all browsers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
a:active,
|
||||||
|
a:hover {
|
||||||
|
outline: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Text-level semantics
|
||||||
|
========================================================================== */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address styling not present in IE 8/9/10/11, Safari, and Chrome.
|
||||||
|
*/
|
||||||
|
|
||||||
|
abbr[title] {
|
||||||
|
border-bottom: 1px dotted;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address style set to `bolder` in Firefox 4+, Safari, and Chrome.
|
||||||
|
*/
|
||||||
|
|
||||||
|
b,
|
||||||
|
strong {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address styling not present in Safari and Chrome.
|
||||||
|
*/
|
||||||
|
|
||||||
|
dfn {
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address variable `h1` font-size and margin within `section` and `article`
|
||||||
|
* contexts in Firefox 4+, Safari, and Chrome.
|
||||||
|
*/
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size: 2em;
|
||||||
|
margin: 0.67em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address styling not present in IE 8/9.
|
||||||
|
*/
|
||||||
|
|
||||||
|
mark {
|
||||||
|
background: #ff0;
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address inconsistent and variable font size in all browsers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
small {
|
||||||
|
font-size: 80%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prevent `sub` and `sup` affecting `line-height` in all browsers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
sub,
|
||||||
|
sup {
|
||||||
|
font-size: 75%;
|
||||||
|
line-height: 0;
|
||||||
|
position: relative;
|
||||||
|
vertical-align: baseline;
|
||||||
|
}
|
||||||
|
|
||||||
|
sup {
|
||||||
|
top: -0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub {
|
||||||
|
bottom: -0.25em;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Embedded content
|
||||||
|
========================================================================== */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove border when inside `a` element in IE 8/9/10.
|
||||||
|
*/
|
||||||
|
|
||||||
|
img {
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Correct overflow not hidden in IE 9/10/11.
|
||||||
|
*/
|
||||||
|
|
||||||
|
svg:not(:root) {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Grouping content
|
||||||
|
========================================================================== */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address margin not present in IE 8/9 and Safari.
|
||||||
|
*/
|
||||||
|
|
||||||
|
figure {
|
||||||
|
margin: 1em 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address differences between Firefox and other browsers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
hr {
|
||||||
|
-moz-box-sizing: content-box;
|
||||||
|
box-sizing: content-box;
|
||||||
|
height: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Contain overflow in all browsers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
pre {
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address odd `em`-unit font size rendering in all browsers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
code,
|
||||||
|
kbd,
|
||||||
|
pre,
|
||||||
|
samp {
|
||||||
|
font-family: monospace, monospace;
|
||||||
|
font-size: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Forms
|
||||||
|
========================================================================== */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Known limitation: by default, Chrome and Safari on OS X allow very limited
|
||||||
|
* styling of `select`, unless a `border` property is set.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 1. Correct color not being inherited.
|
||||||
|
* Known issue: affects color of disabled elements.
|
||||||
|
* 2. Correct font properties not being inherited.
|
||||||
|
* 3. Address margins set differently in Firefox 4+, Safari, and Chrome.
|
||||||
|
*/
|
||||||
|
|
||||||
|
button,
|
||||||
|
input,
|
||||||
|
optgroup,
|
||||||
|
select,
|
||||||
|
textarea {
|
||||||
|
color: inherit; /* 1 */
|
||||||
|
font: inherit; /* 2 */
|
||||||
|
margin: 0; /* 3 */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address `overflow` set to `hidden` in IE 8/9/10/11.
|
||||||
|
*/
|
||||||
|
|
||||||
|
button {
|
||||||
|
overflow: visible;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address inconsistent `text-transform` inheritance for `button` and `select`.
|
||||||
|
* All other form control elements do not inherit `text-transform` values.
|
||||||
|
* Correct `button` style inheritance in Firefox, IE 8/9/10/11, and Opera.
|
||||||
|
* Correct `select` style inheritance in Firefox.
|
||||||
|
*/
|
||||||
|
|
||||||
|
button,
|
||||||
|
select {
|
||||||
|
text-transform: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 1. Avoid the WebKit bug in Android 4.0.* where (2) destroys native `audio`
|
||||||
|
* and `video` controls.
|
||||||
|
* 2. Correct inability to style clickable `input` types in iOS.
|
||||||
|
* 3. Improve usability and consistency of cursor style between image-type
|
||||||
|
* `input` and others.
|
||||||
|
*/
|
||||||
|
|
||||||
|
button,
|
||||||
|
html input[type="button"], /* 1 */
|
||||||
|
input[type="reset"],
|
||||||
|
input[type="submit"] {
|
||||||
|
-webkit-appearance: button; /* 2 */
|
||||||
|
cursor: pointer; /* 3 */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Re-set default cursor for disabled elements.
|
||||||
|
*/
|
||||||
|
|
||||||
|
button[disabled],
|
||||||
|
html input[disabled] {
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove inner padding and border in Firefox 4+.
|
||||||
|
*/
|
||||||
|
|
||||||
|
button::-moz-focus-inner,
|
||||||
|
input::-moz-focus-inner {
|
||||||
|
border: 0;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Address Firefox 4+ setting `line-height` on `input` using `!important` in
|
||||||
|
* the UA stylesheet.
|
||||||
|
*/
|
||||||
|
|
||||||
|
input {
|
||||||
|
line-height: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* It's recommended that you don't attempt to style these elements.
|
||||||
|
* Firefox's implementation doesn't respect box-sizing, padding, or width.
|
||||||
|
*
|
||||||
|
* 1. Address box sizing set to `content-box` in IE 8/9/10.
|
||||||
|
* 2. Remove excess padding in IE 8/9/10.
|
||||||
|
*/
|
||||||
|
|
||||||
|
input[type="checkbox"],
|
||||||
|
input[type="radio"] {
|
||||||
|
box-sizing: border-box; /* 1 */
|
||||||
|
padding: 0; /* 2 */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fix the cursor style for Chrome's increment/decrement buttons. For certain
|
||||||
|
* `font-size` values of the `input`, it causes the cursor style of the
|
||||||
|
* decrement button to change from `default` to `text`.
|
||||||
|
*/
|
||||||
|
|
||||||
|
input[type="number"]::-webkit-inner-spin-button,
|
||||||
|
input[type="number"]::-webkit-outer-spin-button {
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 1. Address `appearance` set to `searchfield` in Safari and Chrome.
|
||||||
|
* 2. Address `box-sizing` set to `border-box` in Safari and Chrome
|
||||||
|
* (include `-moz` to future-proof).
|
||||||
|
*/
|
||||||
|
|
||||||
|
input[type="search"] {
|
||||||
|
-webkit-appearance: textfield; /* 1 */
|
||||||
|
-moz-box-sizing: content-box;
|
||||||
|
-webkit-box-sizing: content-box; /* 2 */
|
||||||
|
box-sizing: content-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove inner padding and search cancel button in Safari and Chrome on OS X.
|
||||||
|
* Safari (but not Chrome) clips the cancel button when the search input has
|
||||||
|
* padding (and `textfield` appearance).
|
||||||
|
*/
|
||||||
|
|
||||||
|
input[type="search"]::-webkit-search-cancel-button,
|
||||||
|
input[type="search"]::-webkit-search-decoration {
|
||||||
|
-webkit-appearance: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Define consistent border, margin, and padding.
|
||||||
|
*/
|
||||||
|
|
||||||
|
fieldset {
|
||||||
|
border: 1px solid #c0c0c0;
|
||||||
|
margin: 0 2px;
|
||||||
|
padding: 0.35em 0.625em 0.75em;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 1. Correct `color` not being inherited in IE 8/9/10/11.
|
||||||
|
* 2. Remove padding so people aren't caught out if they zero out fieldsets.
|
||||||
|
*/
|
||||||
|
|
||||||
|
legend {
|
||||||
|
border: 0; /* 1 */
|
||||||
|
padding: 0; /* 2 */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove default vertical scrollbar in IE 8/9/10/11.
|
||||||
|
*/
|
||||||
|
|
||||||
|
textarea {
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Don't inherit the `font-weight` (applied by a rule above).
|
||||||
|
* NOTE: the default cannot safely be changed in Chrome and Safari on OS X.
|
||||||
|
*/
|
||||||
|
|
||||||
|
optgroup {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tables
|
||||||
|
========================================================================== */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove most spacing between table cells.
|
||||||
|
*/
|
||||||
|
|
||||||
|
table {
|
||||||
|
border-collapse: collapse;
|
||||||
|
border-spacing: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
td,
|
||||||
|
th {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user