forked from toolshed/abra
Compare commits
2303 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| d8f283c7db | |||
| 6488d13e5b | |||
|
1e80d111e6
|
|||
| 7a079b78de | |||
| 7524a785ce | |||
| 1013f669bb | |||
|
aae20f07cc
|
|||
|
6ef8e1ff52
|
|||
|
7fb9675b1e
|
|||
|
d88b478503
|
|||
|
7a735043cd
|
|||
|
e610f32c35
|
|||
| e04a1e15c4 | |||
|
9d401202b4
|
|||
|
6504be6403
|
|||
| d4944dbf35 | |||
|
8d8d4f799d
|
|||
| 0633f24d1b | |||
|
2e062899c7
|
|||
|
fbd7275f03
|
|||
| cedf185e97 | |||
|
06a57ded02
|
|||
| 6f92ba0deb | |||
| dcd830e3f8 | |||
| 8056703d59 | |||
|
566bdf2bd8
|
|||
|
24288c81d3
|
|||
|
2ef2a7ed2c
|
|||
|
cf8cd7423d
|
|||
| a18f57488f | |||
|
b2e691265a
|
|||
|
bff23f0ae6
|
|||
| 403c7a3e5b | |||
| 66b932a553 | |||
| f64e4b62cf | |||
| e80ecbc332 | |||
| abcbdf57f1 | |||
|
78899f173c
|
|||
| 90142cb783 | |||
|
8dbde3d158
|
|||
| 8f42e36302 | |||
| c2552ec2f6 | |||
|
8c64a8049d
|
|||
|
b073072489
|
|||
|
9daa4fee48
|
|||
| ea48917e6c | |||
| 728f873a3e | |||
|
ddb90dd44d
|
|||
|
7a8485492e
|
|||
| 32bb05abba | |||
|
3d2006a696
|
|||
| 521f5c1647 | |||
| 5eb41bc803 | |||
| fc39721501 | |||
| 44bacc582b | |||
|
53e8b52717
|
|||
|
0aba922dda
|
|||
| 4e0eb739b4 | |||
| 6b661dd7a7 | |||
|
39102752c0
|
|||
| 5cfc1c076c | |||
| 10f7ed74b0 | |||
|
227d37dc26
|
|||
|
0ccf3d2b12
|
|||
|
f87ce74027
|
|||
| 4349ee82bc | |||
|
f9ea7506d0
|
|||
|
1fe2d0421b
|
|||
| 59c0d1f4c5 | |||
|
7c3364f87a
|
|||
| 2c5a273fa7 | |||
|
c54fe3ef85
|
|||
| bda0d23d39 | |||
| b9dc7b8437 | |||
| 8f7dbfedbc | |||
| 4b863f1e15 | |||
| 064c9f5d65 | |||
| 98e48c95c7 | |||
| c85d8ee6d1 | |||
| 23268a0e92 | |||
| d60d426752 | |||
| 0e273de8f6 | |||
| ab32118bfe | |||
| 683396d75a | |||
| 4db6755f0d | |||
| 4c132e30f6 | |||
| f5aeae30c7 | |||
| 1d24107956 | |||
| f835b87255 | |||
|
dba21d6a29
|
|||
| 182fc41c58 | |||
| 304ac87cec | |||
| 5b3929d885 | |||
| c41df874d1 | |||
|
b721adbf9c
|
|||
| 42f9e6d458 | |||
| 9e7bc31d4d | |||
| b79c4f33b6 | |||
| cc87d5b3da | |||
| 8b5e3f3c78 | |||
|
db7c4042d0
|
|||
|
ed1a66dc5f
|
|||
|
bb93e4266a
|
|||
|
a2cc70b2f5
|
|||
|
ce1aa3d870
|
|||
|
d75700c8a9
|
|||
|
0ccc4aae72
|
|||
|
ec22d5d51d
|
|||
|
ab42584d05
|
|||
|
40eb6e9a18
|
|||
| 35eb9d4a89 | |||
|
08cc63d523
|
|||
| 797b8d899b | |||
|
fb786306b5
|
|||
| c3a2048eba | |||
| 1bdc11ba62 | |||
| cc8703310c | |||
|
fcd5bd863d
|
|||
|
e6af2da9dd
|
|||
| 4b688825e0 | |||
| b0cf2a1f8e | |||
| 6b7020d457 | |||
| efdac610bd | |||
| cd6021f116 | |||
| ee8de8ef5c | |||
|
e5a653c002
|
|||
|
2cca04de90
|
|||
| f2f79e2df8 | |||
|
dd83741a9f
|
|||
|
dc2cd85d91
|
|||
|
96e59cf196
|
|||
|
11656c009d
|
|||
|
e4e1b58501
|
|||
|
3b8f12643c
|
|||
|
e5f5154197
|
|||
|
6c1c0a8a8a
|
|||
| 662f45008c | |||
|
708c5f5223
|
|||
| d58552b748 | |||
|
51fe809851
|
|||
| 3f6a22747f | |||
|
4e75b96914
|
|||
| fd4ee75ab7 | |||
|
964ed834ee
|
|||
| fcb3167394 | |||
|
3845b40aa3
|
|||
| 0dc5c307af | |||
| fc5855ff28 | |||
|
5b504a1550
|
|||
| fc16a21f1c | |||
|
7b4d2d7230
|
|||
|
d0ccb805c6
|
|||
|
2460dd9438
|
|||
| 9c648a2566 | |||
|
22ecfb9c4c
|
|||
|
9f3cf718be
|
|||
|
b737ce2107
|
|||
|
a3d0ece7cb
|
|||
|
d63a1c28ea
|
|||
| 1c10e64c58 | |||
|
21826ec555
|
|||
|
4b4c56d406
|
|||
|
4314195dd7
|
|||
| df4447b038 | |||
|
3fa660e579
|
|||
|
a430b1e4fd
|
|||
| 896c434f38 | |||
|
847b7238c5
|
|||
|
89d5fc91b0
|
|||
| 5af3c5f56e | |||
|
beb3864b2d
|
|||
|
581e6ef538
|
|||
|
fd642ddb84
|
|||
|
1ad8c127d9
|
|||
| 40aab6a6c1 | |||
|
4d33a24a07
|
|||
|
ee59eb350b
|
|||
|
5da13ff15a
|
|||
|
491c594ad3
|
|||
|
c794d533be
|
|||
|
a6daf7030e
|
|||
|
fe3b7ffa9c
|
|||
|
4c066a92d8
|
|||
|
7899b57781
|
|||
|
6e0a901887
|
|||
|
713fdebc90
|
|||
|
6944d138c6
|
|||
|
fbb1f16470
|
|||
|
2473cafdf5
|
|||
| 0ccfbd253e | |||
|
6c4bee0ac7
|
|||
| 4fa9f536eb | |||
|
033c9bfc13
|
|||
| 0db1ee87fc | |||
|
d180bb924f
|
|||
|
d50d68d95a
|
|||
|
f468bc7443
|
|||
|
dee2d9d104
|
|||
|
5c892b1d6a
|
|||
|
81b96fc7b1
|
|||
| c92a0d0703 | |||
|
1c4abcf12f
|
|||
|
f590870672
|
|||
|
a31a25cfa1
|
|||
| 870dcfb342 | |||
| f53ba48efa | |||
| 26c920e570 | |||
| c67fc57902 | |||
| 07cafd371c | |||
| 5bb6241172 | |||
| 66e6a2c47e | |||
| d866527138 | |||
| 39d1997edf | |||
| d5f5d96944 | |||
| 076d522b1a | |||
| 34934cf62d | |||
| 241dffb8cd | |||
| e42b42e882 | |||
| 0a45424658 | |||
| e73b0cc2fc | |||
| 33aca42181 | |||
| 5c659bae5f | |||
| d9f1f82923 | |||
| 117f64a9d6 | |||
| 90e9e9b5aa | |||
| 7e217f8892 | |||
| bf68ec56a3 | |||
| 40b5c5cd63 | |||
| 14d3f1f669 | |||
| 8e8f7715a2 | |||
| 745651e962 | |||
| c2848cb3ec | |||
| f3edfea744 | |||
| 719722a25b | |||
| 7f9f8f9d6a | |||
| 155df518dd | |||
| 984bdd8792 | |||
| 0b7c38c213 | |||
| 1df0de2e65 | |||
| 6d634ea4e2 | |||
|
dc207a0138
|
|||
|
02add8c3ef
|
|||
|
560d609013
|
|||
|
b4c9fbfe6d
|
|||
| 7f456a3f24 | |||
| 709a9ad659 | |||
| a468245413 | |||
| e895b852bc | |||
| bef92d53a8 | |||
| a4b47b431b | |||
| bddf8039af | |||
| d74e760940 | |||
| 7f75d25d56 | |||
| 0bb6d9609c | |||
| e858dcdd14 | |||
| 3606349a4a | |||
|
4547cf2579
|
|||
|
e1f029d2db
|
|||
|
cf2952dc65
|
|||
|
2291712661
|
|||
|
f0e2b012c6
|
|||
|
9c37b9b748
|
|||
|
824f314472
|
|||
|
61849a358c
|
|||
|
8c7b06a7bb
|
|||
|
4c9abbf925
|
|||
|
09176801e1
|
|||
|
36d4648114
|
|||
|
83ca2a63d1
|
|||
|
e25ce5d1a0
|
|||
|
4cb5091d50
|
|||
|
4bfbc53b94
|
|||
|
52f02ad9b9
|
|||
| c0acc3663b | |||
|
d5c66020ad
|
|||
|
86ba006e17
|
|||
|
cb4355e61e
|
|||
|
069f8fec54
|
|||
| c2819b9366 | |||
|
850264d085
|
|||
|
e019142c9e
|
|||
| e23c6197b5 | |||
| 6539b1be7e | |||
| 02b520200e | |||
| acb6170768 | |||
|
e04af4e582
|
|||
|
8bf0d7addc
|
|||
| 20909695e0 | |||
| baf7631105 | |||
| 57fbc4c061 | |||
| c43e68ea6a | |||
| d3d3358a79 | |||
| cb310c56b5 | |||
| 67c0a64f60 | |||
| db5da1656a | |||
| 5b6254a243 | |||
| 2a0857c388 | |||
| 92a0294f2f | |||
| f79775d726 | |||
| e095bbe9d6 | |||
| eb12127578 | |||
| 82779b233b | |||
| ac4ac1d40f | |||
| 7c31e4dc45 | |||
| b86cd2e85f | |||
| 7dd7f763f4 | |||
| 7b7477062f | |||
| 238647a987 | |||
|
f39eab8f1e
|
|||
|
6a52575ae0
|
|||
|
44a7d288af
|
|||
|
97377dea39
|
|||
|
a2940a84b3
|
|||
|
0c708f6592
|
|||
|
4cb660c348
|
|||
|
42dde0930d
|
|||
|
7ccbbe8916
|
|||
|
e421e00631
|
|||
|
a5104336a2
|
|||
|
4e205cf13e
|
|||
|
5cf6048ecb
|
|||
|
3e2797c433
|
|||
| df89e8143a | |||
| b4ddd3e77c | |||
|
81c28e3006
|
|||
|
34d2e3b092
|
|||
|
1894c2f5fc
|
|||
|
e0bd03bec3
|
|||
|
77ff146991
|
|||
|
6fad1a1dcc
|
|||
|
a90e239547
|
|||
|
9ee094fcd7
|
|||
|
1aa7016789
|
|||
| 60b3af1fa4 | |||
| 5f4b5e0fad | |||
| feadfca0d6 | |||
| 73d4ee1c98 | |||
|
f46c18c8d7
|
|||
| f5a843bd90 | |||
|
fac372dc73
|
|||
|
8a3be01c3e
|
|||
|
4193d63d23
|
|||
|
38f308910a
|
|||
|
4aaa7400b8
|
|||
| 091611b984 | |||
| 2cfc40dc28 | |||
| 6849e3554d | |||
|
452de7fdc2
|
|||
|
952d768ab0
|
|||
| 2c91d2040e | |||
| eff4435971 | |||
| 032fe99086 | |||
| 7add56df00 | |||
| 0ab05cece2 | |||
| c63f6db61e | |||
| 56a68dfa91 | |||
| 157d131b37 | |||
| 3fae036db2 | |||
| ce9d0934b6 | |||
|
a32e30374f
|
|||
|
cf46569f04
|
|||
|
022606c13c
|
|||
|
8cfda5229f
|
|||
|
855a4c37c4
|
|||
| 7c3b740e14 | |||
|
2fbef41a3a
|
|||
| 6fb41e5300 | |||
| 1432f480c7 | |||
|
83af39771b
|
|||
|
4d1333202e
|
|||
|
55c24f070c
|
|||
| 229e8eb9da | |||
|
b3ab95750e
|
|||
| de009921a2 | |||
|
d081bbaefa
|
|||
|
515b5466ca
|
|||
|
6965799bdc
|
|||
|
f75c9a6259
|
|||
| a43a092ba7 | |||
| fa084a61d2 | |||
|
895a7fe7d6
|
|||
|
742a726778
|
|||
|
2b9a185aff
|
|||
|
b7c1e87c0b
|
|||
|
cdfb8a08bb
|
|||
|
8943cea13f
|
|||
|
6d64e0edd3
|
|||
| 47045ca8f1 | |||
| d0f982456e | |||
| 80ad6c6681 | |||
|
cb63cfe9c2
|
|||
|
d1e49d17ce
|
|||
|
1574aa0631
|
|||
|
1723025fbf
|
|||
|
a2b678caf6
|
|||
| 0a371ec360 | |||
| e58a716fe1 | |||
| d09a19a385 | |||
| cee808ff06 | |||
| 4326d1d259 | |||
| b976872f77 | |||
| 7b6ea76437 | |||
| 9069758969 | |||
| 15d6b1a2a5 | |||
|
8a7fe4ca07
|
|||
| 64ad60663f | |||
| cb3f46b46e | |||
|
41e514ae9a
|
|||
|
086b4828ff
|
|||
|
ed263854d4
|
|||
|
eb6fe4ba6e
|
|||
| 993172d31b | |||
| c70b6e72a7 | |||
| 22e4dd7fca | |||
|
b6009057a8
|
|||
|
b978f04910
|
|||
|
3ac29d54d9
|
|||
|
877c17fab5
|
|||
|
f01fd26ce3
|
|||
|
273c165a41
|
|||
|
c88fc66c99
|
|||
|
9b271a6963
|
|||
|
8af87aa382
|
|||
|
ac0b9cd052
|
|||
|
4923984e84
|
|||
|
2bc77de751
|
|||
|
b3a2402cec
|
|||
|
a773fd4256
|
|||
|
b1a0d54bd3
|
|||
|
3869d6bce9
|
|||
|
0ff07ab224
|
|||
|
936c1b0626
|
|||
|
b576cba227
|
|||
|
d087f3debf
|
|||
|
e57a6d87a3
|
|||
|
74b64099de
|
|||
|
354712ca46
|
|||
|
81cdc843ec
|
|||
|
d2931e3af0
|
|||
|
b9f2d1f568
|
|||
| a379b31a19 | |||
|
17e15dba77
|
|||
|
1194f3b228
|
|||
|
2dc8034c16
|
|||
|
c5ddeb2d8a
|
|||
|
0a63f9ce27
|
|||
|
3a71dc47f8
|
|||
|
f07c64f7b8
|
|||
|
dd03c40e10
|
|||
|
48198d55bd
|
|||
|
c0931b96d8
|
|||
|
64ea0f9684
|
|||
|
b0cd8ccbb9
|
|||
|
5975be6870
|
|||
|
bfed51a69c
|
|||
|
5d0faf5e13
|
|||
|
cd6af9708c
|
|||
|
ef95bce1e4
|
|||
|
a159583874
|
|||
|
e3b0500875
|
|||
|
994310a4ff
|
|||
| 74108b0dd9 | |||
|
3727c7fa78
|
|||
|
9a4414fd13
|
|||
|
9f189680f3
|
|||
|
356e527f1f
|
|||
|
7ec61c6d03
|
|||
|
fab93a559a
|
|||
|
8ac31330be
|
|||
|
03000c25e0
|
|||
|
3f32dbb1a3
|
|||
|
27f68b1dc5
|
|||
| a0da5299fe | |||
|
866c5c4536
|
|||
|
dc4c6784cb
|
|||
| 97959ef5da | |||
| b6573720ec | |||
|
4e8995cc0e
|
|||
|
efb3fd8759
|
|||
|
008582c3d9
|
|||
|
8fa20e2c7f
|
|||
|
aa1dc795ef
|
|||
|
18df498295
|
|||
|
671e1ca276
|
|||
|
0df2b15c33
|
|||
|
3f29084664
|
|||
|
0bb25a00ec
|
|||
| 28c7676413 | |||
| 730fef09a3 | |||
| 8d076a308a | |||
| 9510c04aeb | |||
|
d9e60afd71
|
|||
|
31fa9b1a7a
|
|||
| f664599836 | |||
| bba1640913 | |||
| 7b54c2b5b9 | |||
| 8ee1947fe9 | |||
|
b313b0a145
|
|||
|
1f9b863be0
|
|||
|
3b3ce85ef9
|
|||
|
1f8662cd95
|
|||
|
375e17a4a0
|
|||
|
04aec8232f
|
|||
|
2a5985e44e
|
|||
|
c65be64e7d
|
|||
|
fd8652e26d
|
|||
|
518c5795f4
|
|||
|
827edcb0da
|
|||
|
05489a129c
|
|||
|
c02e11eb0a
|
|||
|
8b8e158664
|
|||
|
e5a6dea10c
|
|||
|
1132b09b5b
|
|||
|
b2436174b0
|
|||
|
ea10019068
|
|||
|
9b0b3c2e4c
|
|||
|
8084bff104
|
|||
|
d22e2c38ce
|
|||
|
e945087f79
|
|||
|
7734dd555d
|
|||
|
aedf5e5ff7
|
|||
|
95c598d030
|
|||
|
56068362e8
|
|||
|
cf14731b46
|
|||
|
486cfa68d8
|
|||
|
1718903834
|
|||
|
eb9894e5bb
|
|||
|
a2116774e8
|
|||
|
d2efdf8bf5
|
|||
|
b15c05929c
|
|||
|
f167a91868
|
|||
|
8cded8752a
|
|||
|
d1876e2fae
|
|||
|
e42a1bca29
|
|||
|
b5493ba059
|
|||
|
a41a36b8fd
|
|||
|
de006782b6
|
|||
|
f28cffe6d8
|
|||
|
d3ede0f0f6
|
|||
|
ae4653f5e3
|
|||
|
7f0a74d3c3
|
|||
|
e99114e695
|
|||
|
b1208f9db5
|
|||
|
b8e1a3b75f
|
|||
|
ff90b43929
|
|||
| c5724d56f8 | |||
|
ce7dda1eae
|
|||
|
d38f3ab7f5
|
|||
|
4be8c8daed
|
|||
|
3c9405a4ed
|
|||
| f6b7510da6 | |||
| 7596982282 | |||
| 4085eb6654 | |||
| 790dbca362 | |||
| d7a870b887 | |||
|
1a3ec7a107
|
|||
|
7f910b4e5b
|
|||
|
b82ac3bd63
|
|||
|
00d60f7114
|
|||
|
71d93cbbea
|
|||
|
2fb5493ab5
|
|||
|
0ff8e49cfd
|
|||
|
addbda9145
|
|||
|
c33ca1c6bc
|
|||
|
4580df72cb
|
|||
|
f003430a8d
|
|||
|
5426464092
|
|||
|
72c021c727
|
|||
|
f2e076b35f
|
|||
|
4ccb4198d6
|
|||
|
a9f7579ca9
|
|||
| 9cd1fe658b | |||
| 41c16db670 | |||
| 87ecc05962 | |||
| f14d49cc64 | |||
| f638b6a16b | |||
| 5617a9ba07 | |||
| c1b03bcbd7 | |||
| 99da8d4e57 | |||
| ca1db33e97 | |||
| eb62e0ecc3 | |||
| 6f90fc3025 | |||
| c861c09cce | |||
| 2f41b6d8b4 | |||
| 73e9b818b4 | |||
| f268e5893b | |||
| 47013c63d6 | |||
| 4cf6155fb8 | |||
| 01f3f4be17 | |||
| eee2ecda06 | |||
| 950f85e2b4 | |||
|
9ef64778f5
|
|||
|
735f521bc0
|
|||
|
96a25425a4
|
|||
|
1a8dca9804
|
|||
|
465827d5ee
|
|||
|
cde06f4f00
|
|||
|
050a479df7
|
|||
|
ef108d63e1
|
|||
|
cf8ff410cc
|
|||
|
6ec678208f
|
|||
|
a001be3021
|
|||
|
6712bd446f
|
|||
|
1097daa69f
|
|||
|
beaa233421
|
|||
|
f871f9beee
|
|||
|
0f8f0f908f
|
|||
|
c5211fbd7e
|
|||
| 0076b31253 | |||
| 37aff723c0 | |||
| f18c642226 | |||
| ac695ae28e | |||
|
ac87898005
|
|||
|
32ae2499b6
|
|||
|
1136ec5dcd
|
|||
|
6a2db1abaa
|
|||
|
9554ad40c8
|
|||
|
2014cd6622
|
|||
|
a9ce2106c6
|
|||
|
34de38928a
|
|||
|
f58522d822
|
|||
|
712ebfb701
|
|||
|
1fe601cd16
|
|||
|
7b7e1bfa97
|
|||
|
1a12bef53e
|
|||
|
d787f71215
|
|||
|
9bf44c15ed
|
|||
|
349cacc1f2
|
|||
| 938534f5ac | |||
| 6cd331ebd6 | |||
|
40517171f7
|
|||
| b2485cc122 | |||
| 9ec99c7712 | |||
|
aa3910f8df
|
|||
|
43990b6fae
|
|||
|
91ea2c01a5
|
|||
|
316fdd3643
|
|||
|
e07ae8cccd
|
|||
|
300a4ead01
|
|||
|
f209b6f564
|
|||
|
791183adfe
|
|||
| e6b35e8524 | |||
| 8a0274cac0 | |||
| e609924af0 | |||
| 70e2943301 | |||
| 0590c1824d | |||
| 459abecfa5 | |||
| 183ad8f576 | |||
|
03f94da2d8
|
|||
|
766f69b0fd
|
|||
|
004cd70aed
|
|||
|
a4de446f58
|
|||
| d21c35965d | |||
| 63ea58ffaa | |||
|
2ecace3e90
|
|||
| d5ac3958a4 | |||
| 72c20e0039 | |||
|
575f9905f1
|
|||
|
e3a0af5840
|
|||
|
9a3a39a185
|
|||
|
cea56dddde
|
|||
|
2c515ce70a
|
|||
| 40c0fb4bac | |||
| 0643df6d73 | |||
| e9b99fe921 | |||
| 4920dfedb3 | |||
| 0a3624c15b | |||
|
c5687dfbd7
|
|||
| ca91abbed9 | |||
| d4727db8f9 | |||
| af8cd1f67a | |||
|
cdd7516e54
|
|||
| 99e3ed416f | |||
| 02b726db02 | |||
| 2de6934322 | |||
|
cb49cf06d1
|
|||
|
9affda8a70
|
|||
| 3957b7c965 | |||
| 0d83339d80 | |||
|
6e54ec7213
|
|||
|
66b40a9189
|
|||
|
049f02f063
|
|||
|
15857e6453
|
|||
|
31e0ed75b0
|
|||
| b1d3fcbb0b | |||
| 7b6134f35e | |||
|
316b59b465
|
|||
|
92b073d5b6
|
|||
| 9b0dd933b5 | |||
| f255fa1555 | |||
| 74200318ab | |||
| 609656b4e1 | |||
|
856c9f2f7d
|
|||
| bd5cdd3443 | |||
| 79d274e074 | |||
| 51e3df17f1 | |||
| ccf0215495 | |||
|
254df7f2be
|
|||
|
6a673ef101
|
|||
|
7f7f7224c6
|
|||
|
f96bf9a8ac
|
|||
|
dcecf32999
|
|||
|
bc88dac150
|
|||
|
704c0e9c74
|
|||
|
c9bb7e15c2
|
|||
|
d90c9b88f1
|
|||
|
69ce07f81f
|
|||
| 85b90ef80c | |||
|
3e511446aa
|
|||
|
7566b4262b
|
|||
|
c249c6ae9c
|
|||
|
be693e9df0
|
|||
|
a43125701c
|
|||
|
b57edb440a
|
|||
|
6fc4573a71
|
|||
| cbe6676881 | |||
|
b4fd39828f
|
|||
|
14f2d72aba
|
|||
|
57692ec3c9
|
|||
|
47d3b77003
|
|||
|
8078e91e52
|
|||
|
dc5d3a8dd6
|
|||
|
ab6107610c
|
|||
|
e837835e00
|
|||
|
c646263e9e
|
|||
| 422c642949 | |||
|
379915587c
|
|||
|
970ae0fc4e
|
|||
|
d11ad61efb
|
|||
|
54dc696c69
|
|||
|
7e3ce9c42a
|
|||
| 7751423c7d | |||
|
f18f0b6f82
|
|||
|
892f6c0730
|
|||
|
b53fd2689c
|
|||
|
906bf65d47
|
|||
|
1e6a6e6174
|
|||
|
1e4f1b4ade
|
|||
|
306fe02d1c
|
|||
|
e4610f8ad5
|
|||
|
e1f900de14
|
|||
|
d5b18d74ef
|
|||
|
776a83d8d1
|
|||
|
810cea8269
|
|||
|
c0f3e6f2a4
|
|||
|
7b240059b0
|
|||
|
c456d13881
|
|||
|
c7c553164d
|
|||
|
7616528f4e
|
|||
|
6cd85f7239
|
|||
|
b1774cc44b
|
|||
|
e438fc6e8e
|
|||
|
c065ceb1f0
|
|||
|
ce4b775428
|
|||
|
d02f659bf8
|
|||
|
f3ded88ed8
|
|||
|
bf648eeb5d
|
|||
|
533edbf172
|
|||
|
78b8cf9725
|
|||
|
f0560ca975
|
|||
|
ce7b4733d7
|
|||
|
575bfbb0fb
|
|||
|
510ce66570
|
|||
|
82631d9ab1
|
|||
|
358490e939
|
|||
|
79b9cc9be7
|
|||
|
9b6eb613aa
|
|||
|
8f1231e409
|
|||
|
aa37c936eb
|
|||
|
3d1158a425
|
|||
|
8788558cf1
|
|||
|
76035e003e
|
|||
|
b708382d26
|
|||
|
557b670fc5
|
|||
|
e116148c49
|
|||
|
d5593b69e0
|
|||
|
0be532692d
|
|||
|
7a9224b2b2
|
|||
| e73d1a8359 | |||
|
f8c49c82c8
|
|||
|
ab7edd2a62
|
|||
|
b1888dcf0f
|
|||
| e5e122296f | |||
|
83bf148304
|
|||
| d80b882b83 | |||
| c345c6f5f1 | |||
|
f8c4fd72a3
|
|||
|
10f612f998
|
|||
| 58e78e4d7c | |||
| 25258d3d64 | |||
| b3bd058962 | |||
| b4fd7fd77c | |||
| 64cfdae6b7 | |||
| 0a765794f2 | |||
| 18dc6e9434 | |||
| 4ba4107288 | |||
|
d9b4f4ef3b
|
|||
| c365dcf96d | |||
| 0c6a7cc0b8 | |||
|
6640cfab64
|
|||
| 71addcd1b2 | |||
| 60c0e55e3d | |||
| e42139fd83 | |||
| 2d826e47d0 | |||
| 2db172ea5a | |||
| 2077658f6a | |||
| 502e26b534 | |||
| e22b692ada | |||
|
5ae73f700e
|
|||
|
63d419caae
|
|||
|
179b66d65c
|
|||
|
c9144d90f3
|
|||
|
ebf5d82c56
|
|||
|
8bb98ed0ed
|
|||
|
23f5745cb8
|
|||
|
2cd453ae8d
|
|||
|
e42cc0f91d
|
|||
| 1de45a6508 | |||
|
55c7aca3c0
|
|||
|
8fa9419c99
|
|||
|
73ad0a802e
|
|||
|
798fd2336c
|
|||
| 70e65d6667 | |||
|
efc9602808
|
|||
|
1e110f1375
|
|||
|
473cae0146
|
|||
|
2da859896a
|
|||
|
ab00578ee1
|
|||
|
3dc5662821
|
|||
|
ab64eb2e8d
|
|||
|
4f22228aab
|
|||
|
a7f1af7476
|
|||
| 949510d4c3 | |||
|
9f478dac1d
|
|||
|
69f38ea445
|
|||
|
0582147874
|
|||
|
bdeeb75973
|
|||
|
2518e65e3e
|
|||
|
8354c92654
|
|||
| 173e81b885 | |||
| d91731518b | |||
| 2bfee5058d | |||
| a7ce71d6cf | |||
| 10f60fee1d | |||
| 6025ab443f | |||
| 43ecf35449 | |||
| 4d2a1065d2 | |||
| 0b67500cab | |||
|
e0c3a06182
|
|||
| a86ba4e97b | |||
| b5b3395138 | |||
|
502b78ef5c
|
|||
| 3e2b4dae6a | |||
| 573fe403b3 | |||
|
76862e9d66
|
|||
| e8e337a608 | |||
| 500389c5f5 | |||
|
dea665652c
|
|||
| e8cf84b523 | |||
| fab25a6124 | |||
| e71377539c | |||
|
497ecf476a
|
|||
|
ff1c043ec5
|
|||
|
c4d2e297f8
|
|||
| e98b8e3666 | |||
| f5835fe404 | |||
| 07bbe9394f | |||
| 6974681af5 | |||
| 73250fb899 | |||
| 4ce377cffe | |||
| c7dd029689 | |||
| 51319d2ae2 | |||
| d1c2343a54 | |||
| 135ffde0e5 | |||
|
6e4dd51b27
|
|||
| 81b652718b | |||
|
442f46e17f
|
|||
|
574794d4e8
|
|||
| 88184125c4 | |||
| 8a4baa66ee | |||
| 16ecbd0291 | |||
| f65b262c11 | |||
| c5d9d88359 | |||
| 87e5909363 | |||
| 152c5d4563 | |||
| 34b274bc52 | |||
| 62f8103fc2 | |||
| 2dcbfa1d65 | |||
| 049da94629 | |||
| b2739dcdf2 | |||
|
343b2bfb91
|
|||
|
17aeed6dbd
|
|||
| 27cac81830 | |||
| 31ec322c55 | |||
| 18615eaaef | |||
| 5e508538f3 | |||
| 9e05000476 | |||
|
f088a0d327
|
|||
| 3832403c97 | |||
| 47058c897c | |||
| 5d4c7f8ef0 | |||
| ee4315adb3 | |||
| 9ade250f01 | |||
|
81b032be85
|
|||
| 5409990a68 | |||
| b1595c0ec9 | |||
|
6c99a2980b
|
|||
|
a9405a36c6
|
|||
| 15a417d9bd | |||
| 0ce8b3a5c2 | |||
| edff63b446 | |||
| d5979436c1 | |||
| cb33edaac3 | |||
| e9879e2226 | |||
| 5428ebf43b | |||
| d120299929 | |||
| 3753357ef8 | |||
| 611430aab2 | |||
| f56b02b951 | |||
|
f29278f80a
|
|||
| a9a294cbb7 | |||
|
73004789a4
|
|||
| 440aba77d5 | |||
| e4a89bcc4f | |||
|
eb07617e73
|
|||
|
9fca4e56fb
|
|||
|
f17523010a
|
|||
|
3058178d84
|
|||
|
d62c4e3400
|
|||
|
5739758c3a
|
|||
|
a6b5566fa6
|
|||
|
4dbe1362a8
|
|||
|
98fc36c830
|
|||
|
b8abc8705c
|
|||
|
636261934f
|
|||
|
6381b73a6a
|
|||
|
1a72e27045
|
|||
|
9754c1b2d1
|
|||
| b14ec0cda4 | |||
| c7730ba604 | |||
|
47c61df444
|
|||
|
312b93e794
|
|||
|
992e675921
|
|||
|
d4f3a7be31
|
|||
| d619f399e7 | |||
|
96a8cb7aff
|
|||
| 9b51d22c20 | |||
| d789830ce4 | |||
|
e4b4084dfd
|
|||
|
ff58646cfc
|
|||
|
eec6469ba1
|
|||
|
e94f947d20
|
|||
|
cccbe4a2ec
|
|||
|
f53cfb6c36
|
|||
|
f55f01a25c
|
|||
|
ce5c1a9ebb
|
|||
|
5e3b039f93
|
|||
|
0e9d218bbc
|
|||
|
e1c635af86
|
|||
|
f6b139dfea
|
|||
|
3d2b8fa446
|
|||
|
2eebac6fc0
|
|||
|
f5e2710138
|
|||
|
986470784d
|
|||
| e76ed771df | |||
|
f28af5e42f
|
|||
|
fdf4854b0c
|
|||
|
6b9512d09c
|
|||
|
21a86731d0
|
|||
|
91102e6607
|
|||
|
fadafda0b8
|
|||
|
c03cf76702
|
|||
|
ebb748b7e7
|
|||
|
2b3dbee24c
|
|||
|
a448cfdd0d
|
|||
| 5ee6eb53b2 | |||
| 7b2880d425 | |||
| 928d6f5d7f | |||
|
29fa607190
|
|||
|
7c541ffdfa
|
|||
|
7ccc4b4c08
|
|||
|
ef4df35995
|
|||
|
71a9155042
|
|||
|
2a88491d7c
|
|||
|
bf79552204
|
|||
|
0a7fa54759
|
|||
| 7c1a97be72 | |||
| f20fbbc913 | |||
| 76717531bd | |||
| 6774893412 | |||
| ebb86391af | |||
| 50db39424c | |||
| ca1ea32c46 | |||
| 32851d4d99 | |||
|
c47aa49373
|
|||
|
cdee6b00c4
|
|||
|
a3e9383a4a
|
|||
|
b4cce7dcf4
|
|||
|
b089109c94
|
|||
|
27e0708ac7
|
|||
|
a93786c6be
|
|||
|
521570224b
|
|||
|
c72462e0b6
|
|||
| 54646650c7 | |||
|
903aac9d7a
|
|||
| 49865c6a97 | |||
|
a694c8c20e
|
|||
|
d4a42d8378
|
|||
|
e16ca45fa7
|
|||
|
32de2ee5de
|
|||
|
834d41ef50
|
|||
|
6fe5aed408
|
|||
|
03041b88d0
|
|||
|
9338afb492
|
|||
| 313ae0dbe2 | |||
| 0dc7ec8570 | |||
|
8a1a3aeb12
|
|||
| 910469cfa0 | |||
| 4f055096e9 | |||
| 6c93f980dc | |||
| 57f52bbf33 | |||
| 9f5620d881 | |||
| 44c4555aae | |||
| 025d1e0a8c | |||
| f484021148 | |||
| 1403eac72c | |||
| a6e23938eb | |||
| cae0d9ef79 | |||
|
89fcb5b216
|
|||
| 56b3e9bb19 | |||
|
9aa4a98b0b
|
|||
|
5fbba0c934
|
|||
|
d772f4b2c6
|
|||
| 7513fbd57d | |||
| 9082761f86 | |||
| a3bd6e14d0 | |||
|
49dd702d98
|
|||
|
e4cd5e3efe
|
|||
| 1db4602020 | |||
| b50718050b | |||
| 9e39e1dc88 | |||
|
1a3a53cfc2
|
|||
|
5f53d591f8
|
|||
| d7013518cc | |||
| b204b289d1 | |||
|
3a0d9f7ed7
|
|||
|
e794c17fb4
|
|||
| e788ac21f6 | |||
|
4e78b060e0
|
|||
|
4fada9c1b7
|
|||
|
08d26e1a39
|
|||
|
581b28a2b1
|
|||
| e4d58849ce | |||
| 5e8b9d9bf7 | |||
| 11dd665794 | |||
| ba163e9bf3 | |||
| 09048ee223 | |||
| 19a055b59b | |||
| 1b28a07e17 | |||
| 82866cd213 | |||
|
47f3d2638b
|
|||
|
a3b894320a
|
|||
| 9424a58c52 | |||
| 1751ba534e | |||
| a21d431541 | |||
| 8fad34e430 | |||
|
a036de3c26
|
|||
| 4c2109e8ce | |||
|
7f745ff19f
|
|||
| 521d3d1259 | |||
|
14187449a5
|
|||
|
2037f4cc19
|
|||
| 05d492d30b | |||
| 9591e91ed6 | |||
|
f6f587e506
|
|||
| 4f28dbee87 | |||
| ad1cc038e3 | |||
| 15dbd85d25 | |||
| 2a97955586 | |||
|
9e44d1dfba
|
|||
| 87ad8e2761 | |||
| cfe703b15d | |||
| 96503fa9e9 | |||
|
07d49d8566
|
|||
|
5a7c25375a
|
|||
|
652143e76c
|
|||
|
8afce6eebf
|
|||
|
d3e6c9dc94
|
|||
| 4fd0ca3dd1 | |||
| dc0b6c2c8c | |||
| 54f242baf7 | |||
| 07620c7d89 | |||
| 1cae4cce4e | |||
|
9347ade82c
|
|||
|
3fa18a8050
|
|||
|
4ac67662a2
|
|||
|
d1be4077c5
|
|||
|
5a88c34a7c
|
|||
|
2e452e3213
|
|||
|
9d16a8e10c
|
|||
| 8755a6c3b4 | |||
| 8cee8ae33a | |||
| 15b138e026 | |||
| 4a8ed36dea | |||
| 7d0c3cc496 | |||
| 3cf479ffd5 | |||
| d402050a40 | |||
|
664edce09d
|
|||
|
e41caa891d
|
|||
|
42a6818ff4
|
|||
|
8f709c05bf
|
|||
|
a4ebf7befc
|
|||
|
8458e61d17
|
|||
|
b42d5bf113
|
|||
|
f684c6d6e4
|
|||
| 6593baf9f4 | |||
|
50123f3810
|
|||
|
d132e87f14
|
|||
| 37a1c3fb85 | |||
| c8183aa6d1 | |||
|
4711de29ae
|
|||
|
b719aaba41
|
|||
| 074c51b672 | |||
| 1aa6be704a | |||
|
e8e3cb8598
|
|||
|
85fec6b107
|
|||
| 12dbb061a9 | |||
| 351bd7d4ba | |||
|
cdc7037c25
|
|||
| 682237c98e | |||
|
08d97be43a
|
|||
|
786dfde27e
|
|||
| 6e012b910e | |||
| c153c5da2e | |||
|
0540e42168
|
|||
|
4bc95a5b52
|
|||
|
febc6e2874
|
|||
|
b2c990bf12
|
|||
|
3b8893502a
|
|||
| e0a0378f73 | |||
| 0837045d44 | |||
|
cd8137a7d8
|
|||
|
ece4537a2d
|
|||
|
16fe1b68c6
|
|||
| e37f235fd4 | |||
|
0423ce7e84
|
|||
|
d46ac22bd7
|
|||
|
cef5cd8611
|
|||
| 8b38dac9ab | |||
| 89fc875088 | |||
|
026a9ba2d7
|
|||
| 99f2b9c6dc | |||
|
578e91eeec
|
|||
|
49f79dbd45
|
|||
|
574d556bb9
|
|||
|
801aad64df
|
|||
|
b0a0829712
|
|||
| 6aae06c3ec | |||
| d0c6fa5b45 | |||
| c947354ee3 | |||
|
9b7e5752fb
|
|||
| 9bc51629d4 | |||
|
4ba15df9b7
|
|||
|
5721b357a2
|
|||
| 6140abbcac | |||
| 996255188b | |||
|
11d78234b2
|
|||
|
c214937e4a
|
|||
|
3a3f41988b
|
|||
|
f6690a80bd
|
|||
|
2337c4648b
|
|||
|
a1190f1352
|
|||
|
e421922f5b
|
|||
|
10d5705d1a
|
|||
|
a4f1634b24
|
|||
|
cbd924060f
|
|||
|
3c4bb6a55e
|
|||
|
a0d7a76f9d
|
|||
|
c71efb46ba
|
|||
|
ce69967ec5
|
|||
| 1a04439b1f | |||
|
979f417a63
|
|||
|
b27acb2f61
|
|||
|
622ecc4885
|
|||
|
ed5bbda811
|
|||
|
7b627ea518
|
|||
|
1ac66da83f
|
|||
| 061de96b62 | |||
|
6998298d32
|
|||
|
323f4467c8
|
|||
|
e8e41850b5
|
|||
|
0e23ec53d7
|
|||
|
b943a8b9b1
|
|||
|
acc665f054
|
|||
| 860f1d6376 | |||
| 2122f0e67c | |||
| 6aa23a76a1 | |||
|
338360096c
|
|||
|
7a8c7cd50f
|
|||
|
bafc8a8e34
|
|||
|
3d44d8c9fd
|
|||
|
b8b4616498
|
|||
| da97117929 | |||
| 978297c464 | |||
| 11da4808fc | |||
|
4023e6a066
|
|||
|
f432bfdd23
|
|||
| 848e17578d | |||
|
1615130929
|
|||
|
7f315315f0
|
|||
|
6a50981120
|
|||
|
c67471e6ca
|
|||
|
f0fc1027e5
|
|||
|
c66695d55e
|
|||
|
262009701e
|
|||
|
b31cb6b866
|
|||
|
f39e186b66
|
|||
|
a8f35bdf2f
|
|||
|
6e1e02ac28
|
|||
|
16fc5ee54b
|
|||
|
37a1fcc4af
|
|||
|
a9b522719f
|
|||
|
ce70932a1c
|
|||
|
d61e104536
|
|||
|
d5f30a3ae4
|
|||
|
2555096510
|
|||
|
3797292b20
|
|||
|
6333815b71
|
|||
|
793a850fd5
|
|||
|
42c1450384
|
|||
|
a2377882f6
|
|||
|
e78b395662
|
|||
|
cdec834ca9
|
|||
|
b4b0b464bd
|
|||
|
d8a1b0ccc1
|
|||
|
3fbd381f55
|
|||
|
d3e127e5c8
|
|||
|
e9cfb076c6
|
|||
|
8ccf856110
|
|||
|
d0945aa09d
|
|||
|
123619219e
|
|||
|
a27410952e
|
|||
| 13e0392af6 | |||
| 99a6135f72 | |||
|
a6b52c1354
|
|||
| fa51459191 | |||
|
c529988427
|
|||
|
231cc3c718
|
|||
|
3381b8936d
|
|||
|
823f869f1d
|
|||
|
ecbeacf10f
|
|||
|
3f838038d5
|
|||
| 91b4e021d0 | |||
| 598e87dca2 | |||
| 001511876d | |||
| b295958c17 | |||
|
2fbdcfb958
|
|||
|
09ac74d205
|
|||
|
5da4afa0ec
|
|||
|
9d5e805748
|
|||
| 770ae5ed9b | |||
|
e056d8dc44
|
|||
|
c3442354e7
|
|||
|
6b2a0011af
|
|||
|
46fca7cfa7
|
|||
|
82d560a946
|
|||
|
fc5107865b
|
|||
|
53ed1fc545
|
|||
|
cc9e3d4e60
|
|||
| 0557284461 | |||
| b5f23d3791 | |||
|
2b2dcc01b4
|
|||
|
0a208d049e
|
|||
|
141711ecd0
|
|||
| cd46d71ce4 | |||
| 6fa090352d | |||
|
227c02cd09
|
|||
|
bfeda40e34
|
|||
|
5237c7ed50
|
|||
|
4e09f3b9a8
|
|||
|
dfb32cbb68
|
|||
|
bdd9b0a1aa
|
|||
|
b2d17a1829
|
|||
|
c905376472
|
|||
|
d316de218c
|
|||
|
123475bd36
|
|||
|
58e98f490d
|
|||
|
224b8865bf
|
|||
|
8fb9f42f13
|
|||
|
dc5e2a5b24
|
|||
|
40b4ef5ab2
|
|||
|
4a912ae3bc
|
|||
|
1150fcc595
|
|||
|
45224d1349
|
|||
|
7a40e2d616
|
|||
|
2277e4ef72
|
|||
|
c0c3d9fe76
|
|||
|
2493921ade
|
|||
|
22f9cf2be4
|
|||
| a23124aede | |||
| e670844b56 | |||
|
bc1729c5ca
|
|||
|
fa8611b115
|
|||
|
415df981ff
|
|||
|
57728e58e8
|
|||
|
c7062e0494
|
|||
|
cff7534bf9
|
|||
|
13e582349c
|
|||
|
b1b9612e01
|
|||
|
afeee1270e
|
|||
|
cb210d0c81
|
|||
|
9f2bb3f74f
|
|||
|
a33767f848
|
|||
|
a1abe5c6be
|
|||
|
672b44f965
|
|||
|
6d9573ec7e
|
|||
|
53cd3b8b71
|
|||
|
b9ec41647b
|
|||
|
f4b563528f
|
|||
|
f9a2c1d58f
|
|||
|
7a66a90ecb
|
|||
|
0e688f1407
|
|||
|
c6db9ee355
|
|||
|
7733637767
|
|||
|
88f9796aaf
|
|||
|
6cdba0f9de
|
|||
|
199aa5f4e3
|
|||
|
9b26c24a5f
|
|||
|
ca75654769
|
|||
|
fc2d83d203
|
|||
|
2f4f288a46
|
|||
|
e98f00d354
|
|||
| b4c2773b87 | |||
|
3aec5d1d7e
|
|||
|
e0fa1b6995
|
|||
|
b69ab0df65
|
|||
|
69a7d37fb7
|
|||
|
87649cbbd0
|
|||
|
4b7ec6384c
|
|||
|
b22b63c2ba
|
|||
|
d9f3a11265
|
|||
|
d7cf11b876
|
|||
|
d7e1b2947a
|
|||
|
1b37d2d5f5
|
|||
|
74dfb12fd6
|
|||
|
49ccf2d204
|
|||
|
76adc45431
|
|||
|
e38a0078f3
|
|||
|
25b44dc54e
|
|||
|
0c2f6fb676
|
|||
|
10e4a8b97f
|
|||
|
eed2756784
|
|||
|
b61b8f0d2a
|
|||
|
763e7b5bff
|
|||
|
d5ab9aedbf
|
|||
|
2ebb00c9d4
|
|||
|
6d76b3646a
|
|||
|
636dc82258
|
|||
|
66d5453248
|
|||
|
ba9abcb0d7
|
|||
|
a1cbf21f61
|
|||
|
bd1da39374
|
|||
|
8b90519bc9
|
|||
|
65feda7f1d
|
|||
|
64e223a810
|
|||
|
379e01d855
|
|||
|
a421c0dca5
|
|||
|
abf56f9054
|
|||
|
4dec3c4646
|
|||
|
c900cebc30
|
|||
|
30209de3e2
|
|||
|
625747d048
|
|||
|
a71b070921
|
|||
|
33ff04c686
|
|||
|
c69a3c23c5
|
|||
|
0b46909961
|
|||
|
832e8e5a96
|
|||
|
abf83aa641
|
|||
|
1df69aa259
|
|||
|
7596a67ad5
|
|||
|
93c7612efc
|
|||
|
2c78ac22e0
|
|||
|
13661c72ce
|
|||
|
454092644a
|
|||
|
224c0c38db
|
|||
|
560e0eab86
|
|||
|
b92fdbbd52
|
|||
|
0a550363b8
|
|||
|
3119220c21
|
|||
|
49f565e5db
|
|||
|
94522178b1
|
|||
|
810bc27967
|
|||
|
35d95fb9fb
|
|||
|
d26fabe8ef
|
|||
|
84bf3ffa50
|
|||
|
575485ec7a
|
|||
|
0b17292219
|
|||
|
fffd8b2647
|
|||
|
c07128b308
|
|||
|
929ff88013
|
|||
|
0353427c71
|
|||
|
7a0d18ceb6
|
|||
|
8992050409
|
|||
|
abd094387f
|
|||
|
a556ca625b
|
|||
|
1b7836009f
|
|||
|
eb3509ab3f
|
|||
|
87851d26f7
|
|||
|
c4f344b50a
|
|||
|
60e4dfd9cb
|
|||
|
d957adb675
|
|||
|
5254af0fe4
|
|||
|
ce96269be0
|
|||
|
299276c383
|
|||
|
866cdd1f29
|
|||
|
95d385c420
|
|||
|
605e2553b8
|
|||
|
1245827dff
|
|||
|
9bdb07463c
|
|||
|
be26f80f03
|
|||
|
930ff68bb2
|
|||
|
62441acf03
|
|||
|
7460668ef4
|
|||
|
047d0e6fbc
|
|||
|
8785f66391
|
|||
|
24882e95b4
|
|||
|
1fd0941239
|
|||
|
26a11533b4
|
|||
|
b4f48c3c59
|
|||
|
43e68a99b0
|
|||
|
bac6fb0fa8
|
|||
|
dc9c9715ce
|
|||
|
1f91b3bb03
|
|||
|
a700aca23d
|
|||
|
5cacd09a04
|
|||
|
6a98024a2b
|
|||
|
e85117be22
|
|||
|
fb24357d38
|
|||
|
f5d2d3adf6
|
|||
|
07119b0575
|
|||
|
d2a6e35986
|
|||
|
0aa37fcee8
|
|||
|
eb1b6be4c5
|
|||
|
b98397144a
|
|||
|
4c186678b8
|
|||
|
b1d9d9d858
|
|||
|
a06043375d
|
|||
|
3eef1e8587
|
|||
|
37e48f262b
|
|||
|
06cc5d1cc3
|
|||
|
c13f438580
|
|||
|
5cd4317580
|
|||
|
2ba1ec3df0
|
|||
|
34cdb9c9d8
|
|||
|
9c281d8608
|
|||
|
321ba1e0ec
|
|||
|
c5a74e9f6b
|
|||
|
f8191ac248
|
|||
|
027c8a1420
|
|||
|
cdc08ae95a
|
|||
|
3f35510507
|
|||
|
9f70a69bbf
|
|||
|
b0834925a3
|
|||
|
86d87253c5
|
|||
|
17340a79da
|
|||
|
779c810521
|
|||
|
9cc2554846
|
|||
|
9a1cf258a5
|
|||
|
ba8138079f
|
|||
|
8735a8f0ea
|
|||
|
a84a5bc320
|
|||
|
ae0e7b8e4c
|
|||
|
c0caf14d74
|
|||
|
d66c558b5c
|
|||
|
c8541e1b9d
|
|||
|
653b6c6d49
|
|||
|
e2c3bc35c3
|
|||
|
6937bfbb0d
|
|||
|
decfe095fe
|
|||
|
4283f130a2
|
|||
|
3b5354b2a5
|
|||
|
14400d4ed8
|
|||
|
dddf84d92b
|
|||
|
fefb042716
|
|||
|
ab8db8df64
|
|||
|
20f7a18caa
|
|||
|
58a24a50e1
|
|||
|
e839f100df
|
|||
|
41a757b7ed
|
|||
|
4b4298caf1
|
|||
|
8e8c241fdf
|
|||
|
9b8ff1ddcd
|
|||
|
a85cfe40d0
|
|||
|
fc29ca6fce
|
|||
|
cfb02f45ed
|
|||
|
696172ad48
|
|||
|
4089949a3f
|
|||
|
a75b01e78a
|
|||
|
014d32112e
|
|||
|
a7894cbda9
|
|||
|
e03761f251
|
|||
|
190c1033e6
|
|||
|
15d1e9dee0
|
|||
|
0362928840
|
|||
|
844961d016
|
|||
|
d0cc51b829
|
|||
|
606b5ac3e4
|
|||
| 6f1bf258b3 | |||
|
7a5aa1b005
|
|||
|
db453f0ab1
|
|||
|
a07e71f7df
|
|||
|
4c6d52c426
|
|||
|
327c5adef2
|
|||
|
0dc8425a27
|
|||
|
48c965bb21
|
|||
|
5513754c22
|
|||
|
3a27d9d9fb
|
|||
|
04b58230ea
|
|||
|
1b9097f9f3
|
|||
|
3d100093dc
|
|||
|
ef4383209e
|
|||
|
74f688350b
|
|||
|
737a22aacc
|
|||
|
56a1e7f8c4
|
|||
|
6be2f36334
|
|||
|
a18d0e290d
|
|||
|
7e0feec311
|
|||
|
29a4d05944
|
|||
|
b72bad955a
|
|||
|
e9b4541c91
|
|||
|
5b1b16d64a
|
|||
|
ec7223146b
|
|||
|
fa45264ea0
|
|||
|
f57222d6aa
|
|||
|
28d10928a4
|
|||
|
0f4da38f98
|
|||
| 11c2d1efe6 | |||
|
2b1cc9f6dd
|
|||
|
6100a636a6
|
|||
|
ddbf923338
|
|||
|
c1a00520dc
|
|||
|
0dc4b2beef
|
|||
|
f75284364d
|
|||
|
fbc3b48d39
|
|||
|
6f0d8b190d
|
|||
|
fc3742212c
|
|||
|
fccbd7c7d7
|
|||
|
2457b5fe95
|
|||
|
72df640d99
|
|||
|
ae9e66c319
|
|||
|
3589a7d56e
|
|||
|
8d499c0810
|
|||
|
cb2bb3f532
|
|||
|
0a903f041f
|
|||
|
053a06ccba
|
|||
|
398deec272
|
|||
|
bf82bc9c7f
|
|||
|
217d4bc2cc
|
|||
|
9c8e6b63a6
|
|||
|
5113db1612
|
|||
|
66666e30b7
|
|||
|
88d4984248
|
|||
|
bc34be4357
|
|||
|
3d1aa55587
|
|||
|
e7469acf5b
|
|||
|
a293179e89
|
|||
|
b912e73c5e
|
|||
|
4c66e44b3a
|
|||
|
033bad3d10
|
|||
|
a750344653
|
|||
|
f5caf5587a
|
|||
|
fdc9e8b5fd
|
|||
|
75edcabb23
|
|||
|
fa0a63c11d
|
|||
|
3d3eefb2fe
|
|||
|
6998a87eef
|
|||
|
b71a379788
|
|||
|
ba217dccbd
|
|||
|
45259b3266
|
|||
|
59b80d5def
|
|||
|
8f6e1de1a1
|
|||
|
cd0d3b8892
|
|||
|
0d1f65daac
|
|||
|
cf1b46fa61
|
|||
|
0fe0ffbafa
|
|||
|
af3def7267
|
|||
|
c7de9c0719
|
|||
|
cf5ee4e682
|
|||
|
9ddf69b988
|
|||
|
a925da8dee
|
|||
|
06f8078866
|
|||
|
467947edf2
|
|||
|
512cd9d85b
|
|||
|
b8e2d1de67
|
|||
|
3b7a8e6498
|
|||
|
5bae262a79
|
|||
|
6ad253b866
|
|||
| b603069514 | |||
| d999cedd97 | |||
|
8215bb455b
|
|||
|
37ab9a9c08
|
|||
|
48dd9cdeed
|
|||
|
d02e1f247f
|
|||
|
d087a60e09
|
|||
|
48e16c414c
|
|||
|
f3e55e5023
|
|||
|
ae6adace50
|
|||
|
32dcddb631
|
|||
|
3dbd343600
|
|||
|
8393f4b134
|
|||
|
8e56607cc9
|
|||
|
85a543afac
|
|||
|
665396b679
|
|||
|
870c561fee
|
|||
|
3fb43ffa2c
|
|||
|
2bc2f8630b
|
|||
|
6094dfaf92
|
|||
|
3789e56404
|
|||
|
2db5378418
|
|||
|
7d8f3f1fab
|
|||
|
9be78bc5fa
|
|||
|
6c87d501e6
|
|||
| 930c29f4a2 | |||
| 1d6c3e98e4 | |||
| a90f3b7463 | |||
| 962f566228 | |||
| 9896c57399 | |||
| 748d607ddc | |||
| 3901258a96 | |||
| 4347083f98 | |||
| 4641a942d8 | |||
| 759a00eeb3 | |||
| d1526fad21 | |||
|
6ef15e0a26
|
|||
| dd0f328a65 | |||
| aea5cc69c3 | |||
| b02475eca5 | |||
| d0a30f6b7b | |||
| 8635922b9f | |||
| 9d62fff074 | |||
| 711c4e5ee8 | |||
| cb32e88cde | |||
| a18729bf98 | |||
| dbf84b7640 | |||
| 75db249053 | |||
| fdf4fc6737 | |||
| ef6a9abba9 | |||
| ce57d5ed54 | |||
| 3b01b1bb2e | |||
| fbdb792795 | |||
| 900f40f07a | |||
| ecd2a63f0a | |||
| 304b70639f | |||
| d821975aa2 | |||
| 1b836dbab6 | |||
| fc51cf7775 | |||
| a7ebcd8950 | |||
| e589709cb0 | |||
| 56c3e070f5 | |||
| cc37615d83 | |||
| 0b37f63248 | |||
| 9c3a06a7d9 | |||
| cdef8b5ea5 | |||
| cba261b18c | |||
| 1f6e4fa4a3 | |||
| 4a245c3e02 | |||
|
299faa1adf
|
|||
| 704e773a16 | |||
| 7143d09fd4 | |||
| 4e76d49c80 | |||
| c9dff0c3bd | |||
| e77e72a9e6 | |||
| af6f759c92 | |||
| 034295332c | |||
| dac2489e6d | |||
| 7bdc1946a2 | |||
| 2439643895 | |||
| 0876f677d1 | |||
| 31dafb3ae4 | |||
| 915083b426 | |||
| 486a1717e7 | |||
| 9122c0a9b8 | |||
| 85ff04202f | |||
| ecba4e01f1 | |||
| 751b187df6 | |||
| f74261dbe6 | |||
| 2600a8137c | |||
| b6a6163eff | |||
|
c25b2b17df
|
|||
| 713308e0b8 | |||
| fcbf41ee95 | |||
|
5add4ccc1b
|
|||
|
9220a8c09b
|
|||
| f78a04109c | |||
| b67ad02f87 | |||
| 215431696e | |||
| cd361237e7 | |||
| db10c7b849 | |||
| d38f82ebe7 | |||
| 59031595ea | |||
| 6f26b51f3e | |||
|
17a5f1529a
|
|||
| 2ba6445daa | |||
| edb427a7ae | |||
| 3dc186e231 | |||
| 1467ae5007 | |||
| 2b9395be1a | |||
| a539033b55 | |||
| 63d9703d9d | |||
| f9726b6643 | |||
| 4a0761926c | |||
| de7054fd74 | |||
| 0e0e2db755 | |||
| 04e24022f5 | |||
| c227972c12 | |||
| 911f22233f | |||
| 7d8e2d9dd1 | |||
| f041083604 | |||
| f57ae1e904 | |||
| 49a87cae2e | |||
| f0de18a7f0 | |||
| 1caef09cd2 | |||
| e4e606efb0 | |||
| 08aca28d9d | |||
| f02ea7ca0d | |||
| 3d3c4b3aae | |||
| e37b49201f | |||
| ede5a59562 | |||
| fc2deda1f6 | |||
| c76601c9ce | |||
| 7f176d8e2f | |||
| 9b704b002b | |||
| ab02c5f0dd | |||
| f2b02e39a7 | |||
| 31f6bd06a5 | |||
| bd92c52eed | |||
| 0486091768 | |||
| 3b77607f36 | |||
| f833ccb864 | |||
| 7022f42711 | |||
| c76bd25c1d | |||
| a6b5ac3410 | |||
|
71225d2099
|
|||
|
5d59d12d75
|
|||
| d56400eea8 | |||
| b3496ad286 | |||
| 066b2b9373 | |||
| aec11bda28 | |||
| 9a513a0700 | |||
| 9f3ab0de9e | |||
| e26afb97af | |||
| 960e47437c | |||
| 8e3f90a7f3 | |||
| 1d7cb0d9b6 | |||
| 4d2a2d42fb | |||
| bdae61ed51 | |||
| 766e3008f6 | |||
|
383f857f4a
|
|||
| 3d46ce6db2 | |||
| 9e0d77d5c6 | |||
| f9e2d24550 | |||
| 8772217f41 | |||
| a7970132c2 | |||
| 2d091a6b00 | |||
| 147687d7ce | |||
| 9a0e12258a | |||
| 1396f15c78 | |||
| 2e2560dea7 | |||
| c789a70653 | |||
| 8f55330210 | |||
| d54a45bef7 | |||
| fdc0246f1d | |||
| a394618965 | |||
| 8cd9f2700f | |||
| b72fa28ddb | |||
| 313e3beb1e | |||
| 94c7f59113 | |||
| 5ae06bbd42 | |||
| 9f9248b987 | |||
| 2bb4a9c063 | |||
| 0c8dba0681 | |||
| a491332c1c | |||
| 6a75ffc051 | |||
| 5261d1a033 | |||
| a458a5d9f7 | |||
| 5ce2419354 | |||
| 963f8dcc73 | |||
| dc04cf5ff7 | |||
| 80921c9f55 | |||
| 8b15f2de5b | |||
| cdb76e7276 | |||
| a170e26e27 | |||
| 03b1882b81 | |||
| 2fcdaca75f | |||
| c5f44cf340 | |||
| 7a5ad65178 | |||
| 6d4ee3de0d | |||
| 63318fb6ff | |||
| 07ffa08a07 | |||
| 0e5e7490b3 | |||
| 640032b8fe | |||
| 39babea963 | |||
| 07613f5163 | |||
| 7f1d9eeaec | |||
| 02d24104e1 | |||
|
da8d72620a
|
|||
|
96ccadc70f
|
|||
| 8703370785 | |||
| 7d8c53299d | |||
| 0110aceb1f | |||
| aec1e4520d | |||
| 74bcb99c70 | |||
| dd4f2b48ec | |||
| 7f3f41ede4 | |||
| 597b4b586e | |||
| 7ea3df45d4 | |||
| 5941ed9728 | |||
| d1e42752e2 | |||
| 9dfbd21c61 | |||
| 9526d1fde6 | |||
| 62cc7ef92d | |||
| c5a7a831d2 | |||
| 4aae186f5f | |||
| 2f9b11f389 | |||
| 6d42e72f16 | |||
| 5be190e110 | |||
| c1390f232e | |||
| 95e19f03c4 | |||
|
dc040a0b38
|
|||
|
e6e2e5214f
|
|||
|
61452b5f32
|
|||
|
78460ac0ba
|
|||
| 0615c3f745 | |||
| e820e0219d | |||
| 75fb9a2774 | |||
| 0d500b636d | |||
| 5dd97cace0 | |||
| ae32b1eed2 | |||
| 113bdf9e86 | |||
| d4d4da19b7 | |||
| 454ee696d6 | |||
| ca16c002ba | |||
| 91cc8b00b3 | |||
| d0828c4d8d | |||
| b69aed3bcf | |||
| 875255fd8c | |||
| 2dca602c0b | |||
| 1dca8a1067 | |||
| 37022bf0c8 | |||
|
eb5b35d47f
|
|||
|
ece1130797
|
|||
|
c266316f7e
|
|||
| d804276cf2 | |||
| 4235e06943 | |||
| a9af0b3627 | |||
| a0b4886eba | |||
| 84489495dc | |||
| a8683dc38a | |||
| e2128ea5b6 | |||
| ca3c5fef0f | |||
| 4a01e411be | |||
| 777d49ac1d | |||
| deb7d21158 | |||
| 6db1fdcfba | |||
| 44dc0edf7b | |||
|
36ff50312c
|
|||
| ff4b978876 | |||
| b68547b2c2 | |||
| 0140f96ca1 | |||
| 1cb45113db | |||
| c764243f3a | |||
| dde8afcd43 | |||
| 98ffc210e1 | |||
| 7c0d883135 | |||
| e78ced41fb | |||
| e9113500d8 | |||
| 7368cabc49 | |||
| f75e264811 | |||
| 8bfd76fd04 | |||
|
1cb5e3509d
|
|||
| 3cd2399cca | |||
| 11c4651a3b | |||
| 49f90674f2 | |||
| 74a70edb03 | |||
| 6fc5c31347 | |||
| c616907b71 | |||
| a58cea3e0a | |||
| 700f89425a | |||
| 8cc0a350e6 | |||
| 46e67fa420 | |||
| cacbb5a0f1 | |||
| e7046a15aa | |||
| c1fd97c427 | |||
| 2f218bd99f | |||
| 48290aa316 | |||
| db5cbfa992 | |||
| 4c11e813e8 | |||
|
6ae75e013a
|
|||
| 09f49cdc76 | |||
| 22118b88e4 | |||
| e6db064149 | |||
| 3688ea9d69 | |||
| 7c4cdc530c | |||
| 49781c7e3f | |||
| 10b15d65b4 | |||
| 1c5d6d6357 | |||
| 75bdd59585 | |||
|
96bb145981
|
|||
|
c4c76f4848
|
|||
| 2076c566bb | |||
| 62f6327b66 | |||
| 6f9120b59c | |||
| 8c617a9f12 | |||
|
857d12d23c
|
|||
|
22c4d0d864
|
|||
|
e700e44363
|
|||
|
9faefd2592
|
|||
|
cd179175f5
|
|||
|
c0f92ca13d
|
|||
|
48d28c8dd1
|
|||
| e840328e44 | |||
| 6f43778691 | |||
| 9783563fa6 | |||
| 1392afc015 | |||
| 886009975d | |||
| b1147cd136 | |||
| 95a9013658 | |||
| bd1bf3b0d6 | |||
| 7b349732ac | |||
| a8ce64a9db | |||
| 96aa74a977 | |||
| 700f022790 | |||
| d188327b17 | |||
| fdd46a4d98 | |||
| e00920643e | |||
| 754fe81e01 | |||
| bece2e8351 | |||
|
e47d7029d7
|
|||
|
31edbbd32e
|
|||
|
0a1c73bf00
|
|||
| a74a8bc21b | |||
| 357cc0593a | |||
| 8e111dc32f | |||
| 20ecdb8061 | |||
| f87aad4688 | |||
| 6794236b77 | |||
| 6c9bb89a10 | |||
| 66aeeee768 | |||
| 6c115926e3 | |||
| b6fe86f2ad | |||
| d290a4ec0b | |||
| f93563588a | |||
| 59c55c0a2f | |||
| 9fcdc45851 | |||
| 27d665c3be | |||
| bc5fc0b0cb | |||
| 99160967a8 | |||
| 683ef0c3de | |||
| 3c3d8dc0e7 | |||
| 855e9ea26d | |||
| 50d663ff6e | |||
| 39ad6e8aa8 | |||
| f39c8cbe21 | |||
| e114b2a939 | |||
|
511619722f
|
|||
|
cf2653fef8
|
|||
| 5ba40ad883 | |||
| 2e0c16d198 | |||
|
4c216fdf40
|
|||
| 5f50c7960c | |||
| 719e24eb80 | |||
| c441a1ab52 | |||
| b0460bd923 | |||
| f1659b3bda | |||
| eb4a2b3339 | |||
| 265bfe92fd | |||
|
1757fabb89
|
|||
| abf0ebf41d | |||
| 45f1692c99 | |||
| 48bc03db51 | |||
| f0e966afc3 | |||
| a1d1166308 | |||
| 1438fdf3c2 | |||
| ddda02a6fc | |||
| 4712131f36 | |||
|
50c321aecc
|
|||
| 5bfd233f2a | |||
| d19c56d75b | |||
| 20fa0da1bb | |||
| 1de4f95267 | |||
| 874550ff7f | |||
| e38917339d | |||
| dcf1a90c31 | |||
| a06870f5cb | |||
| b477bf8ece | |||
| 87f0985ebb | |||
| 2cb0fb8d66 | |||
| 76372bb8cb | |||
| 160ccf9745 | |||
| 96cfd8da2b | |||
| ff8d9a554a | |||
| cf94c5acd0 | |||
|
bb8124030e
|
|||
| 448dadd292 | |||
| 8aaedee39e | |||
| f4d8b45859 | |||
| 7ed37547a5 | |||
| 9862cf17a9 | |||
| d1527741ba | |||
| 9d6739a711 | |||
| 356c8f8c4e | |||
| 6a1ecd0f85 | |||
| b5d8fb1270 | |||
| e1a10723ce | |||
| a0625bf133 | |||
| 691a2c7a50 | |||
| c03d187256 | |||
| 5e05bcd8b0 | |||
| d4333c2dc0 | |||
| 48bcc9cb36 | |||
| ec40d88134 | |||
| cc249e8187 | |||
| 273db078b0 | |||
| d82f854ebd | |||
| b7742d5e18 | |||
| f59380a35e | |||
| c99f0fc908 | |||
| 317be4cc01 | |||
| a3a66ef972 | |||
| 7155a33d31 | |||
| d5f49594a9 | |||
| 5287f097e7 | |||
| 1961cdcfee | |||
| 0727223009 | |||
| 07a43cb314 | |||
| dac679db48 | |||
| 254a4d6d43 | |||
| fb75567729 | |||
| cb637ca89e | |||
| ff21237a21 | |||
| 5e4114036b | |||
| 4e92057f61 | |||
| fadbbabe09 | |||
| ba7b18f703 | |||
| 9bf11961d5 | |||
| a3e02540f6 | |||
| e68c7fc71c | |||
| a8f30426ea | |||
| dc616fd3a0 | |||
| 56796cf768 | |||
| da049ad69a | |||
| f65090bd2f | |||
| b1d4f12e7d | |||
|
836420e369
|
|||
| 5c56e4521d | |||
| 612fc5a531 | |||
| 92c8e9aab9 | |||
| 97188b57d9 | |||
| 7ab44cea57 | |||
|
c150856a66
|
|||
| 063fa66af9 | |||
| 09873b42ce | |||
| ac86912ead | |||
|
45c6be02b1
|
|||
|
7835c1f91d
|
|||
|
542e9eea5c
|
|||
|
32b2bf245b
|
|||
|
3b93f893fd
|
|||
|
7dce352366
|
|||
|
8fdac00a38
|
|||
| dc193734df | |||
| 57e641689a | |||
| d68f2f5686 | |||
| f9ae9c9a56 | |||
| 15651822f1 | |||
| 89b5f12fb1 | |||
| 66a8630101 | |||
| e1630dc2b3 | |||
| 9310a85df8 | |||
| 440911f983 | |||
| 8cc691ab52 | |||
| bef2c862bf | |||
| db4908c3ae | |||
| ddbe9ffcb5 | |||
| df236b6c25 | |||
| 8651e22441 | |||
| 45e2442e83 | |||
| 547f785da5 | |||
| 16297e8651 | |||
| acca710a5a | |||
| 0825321b26 | |||
| cc45e722e8 | |||
| 8ad51c1fd5 | |||
| 23737ed3a7 | |||
| 462599a791 | |||
| 3a8296a8fe | |||
| 73ebf998c7 | |||
| ed551763de | |||
| ff0b0b5ad8 | |||
| ce5a03d579 | |||
| de5169ea24 | |||
| 4f1cb86b6b | |||
| 62ceca798c | |||
| b34acefa21 | |||
| c5bb680fed | |||
| ed11634abf | |||
| 3211994b2e | |||
| d2d0ce3d05 | |||
|
c3088a5158
|
|||
|
5663659d23
|
|||
|
ca4001a805
|
|||
|
e2f4ed11ec
|
|||
|
cc9c690cd0
|
|||
|
451c3d772d
|
|||
|
98ec23761f
|
|||
|
d5b893d9de
|
|||
|
b143b544b6
|
|||
|
6df08df509
|
|||
|
8f9ffa0667
|
|||
| 6f0eff5919 | |||
|
cefad74e22
|
|||
| c2499e35d4 | |||
| edd0b1e098 | |||
| 33fc4844ba | |||
| ba5e87f754 | |||
| cbe74b24c4 | |||
| 83671f42a2 | |||
| c6ea18311e | |||
| 1c217b127b | |||
| 2028b9d7c7 | |||
| fa5f5f650d | |||
| a12b53abab | |||
| e39c6a05be | |||
| 210baf1905 | |||
| 1b03836210 | |||
| 334e417abf | |||
| 7b1a6dd4d7 | |||
| a18a9493f2 | |||
| 16e844643a | |||
| 7ad812ad98 | |||
| 260edad142 | |||
| 5ac4604f8a | |||
| 3d7961282a | |||
| 828417c92b | |||
| 11ef64ead3 | |||
| c75c2254e4 | |||
| 36af302d5f | |||
| 6732edf8db | |||
| 8554e68418 | |||
| 202f7ce561 | |||
| 9378db1979 | |||
| efb9d6f6a5 | |||
| 327e2afcd0 | |||
| e22d22056d | |||
| 532bb8a336 | |||
| 3a42288a59 | |||
| 471c982f63 | |||
| 43238d379c | |||
|
239c925d66
|
|||
|
b351760f6e
|
|||
| 102f4e22b5 | |||
| 444ac52476 | |||
| 5294e84d5e | |||
| 3e91174ce0 | |||
|
fa16ce20eb
|
|||
|
38d8b51bd5
|
|||
|
9070806f8d
|
|||
|
bb1eb372ef
|
|||
|
d777eb2af1
|
|||
|
a3f574a8fa
|
|||
|
30d11f48a7
|
|||
|
c2f53e493e
|
|||
|
dc4e490497
|
|||
|
ffd1b3a771
|
|||
|
8267d4202b
|
|||
| d74b7636a1 | |||
|
9d621404fd
|
|||
| 4ae5e6123d | |||
| 19d435c5e5 | |||
| 6be54c670a | |||
| a1bce4661b | |||
| 8a5ee68b7b | |||
| 1846f965ec | |||
|
805defec09
|
|||
|
f958b888b6
|
|||
|
1768809872
|
|||
| 8abc47d2e0 | |||
| bf7de84c66 | |||
| 760ac495b3 | |||
| 4d12a75494 | |||
| 1442c71911 | |||
| e4c864a60c | |||
| 42968fb8e1 | |||
| 932803453e | |||
| 5771f6c158 | |||
| e728bcd7ac | |||
| 769c5b899b | |||
| f56ddef6c8 | |||
|
ac6b8ab147
|
|||
|
a581049cf1
|
|||
| 58bdb456df | |||
| d97da9f45c | |||
| 064a0f271f | |||
| 6c36e77722 | |||
| d422902e09 | |||
| e4ed2aeebf | |||
| f7b085dfa2 | |||
| 1187d6bfd5 | |||
| bf0212c520 | |||
| de3ea8188e | |||
| bf7d437571 | |||
| 1ee572363a | |||
| 2c1b8ee7e2 | |||
| 622e0127ea | |||
| d581d3313a | |||
| 0e75350985 | |||
| cf7a8d114a | |||
| ef1591d596 | |||
| 429c7e4e50 | |||
| 3bc612c44e | |||
| 2c83113040 | |||
| fae5a87ce2 | |||
| 145e6326c9 | |||
| 5def18a9af | |||
| 8656ae947a | |||
| c27376c89b | |||
| 01cbee824a | |||
| 337d3e9ae1 | |||
| 60a70d2d83 | |||
| 1f62ace524 | |||
| 13028db287 | |||
| 1f550c2470 | |||
| 359b07b562 | |||
| 45c3bce7ff | |||
| 6eee02d90a | |||
|
dfc91a86a1
|
|||
|
dd86ec4ca8
|
|||
| fce1ab6c02 | |||
| 381de28e83 | |||
| 56cec1580a | |||
|
ebfdb504ce
|
|||
|
fc7dade6f8
|
|||
|
5e94050865
|
|||
|
fe86b50ee3
|
|||
| a4a8997f57 | |||
| 1f6c0e8c4b | |||
|
6b370599fa
|
|||
|
9216cc5d6a
|
|||
| 53576dc916 | |||
| 05ff163386 | |||
| 302ebcb394 | |||
|
0242dfcb0f
|
|||
|
29971c36a0
|
|||
|
2158dc851c
|
|||
|
a36e80db99
|
|||
|
b0c241ae98
|
|||
| a74d214121 | |||
| e064f18730 | |||
| 7b2100c568 | |||
| c9ba7aef20 | |||
|
16514b3151
|
|||
| 635c6d6080 | |||
|
dee013e4e4
|
|||
|
a60ebf8710
|
|||
|
cfe2f70151
|
|||
|
bd9bc530d1
|
|||
|
f7059dbe98
|
|||
|
8c5e25bd01
|
|||
|
6caa176308
|
|||
|
7d5db5fee1
|
|||
|
37c06c82bf
|
|||
|
a2bb0ed027
|
|||
|
38f610bdec
|
|||
|
e13948f37e
|
|||
|
d1f7e8011d
|
|||
|
2134f57dd0
|
|||
|
6c748922b4
|
|||
|
b9e06f2310
|
|||
|
be46695d82
|
|||
|
ae68f3aa95
|
|||
|
5e1b076bf9
|
|||
| c65ae974dd | |||
| 9b8f16345c | |||
| 4884c14ab3 | |||
| a9d9d9de2f | |||
| 980f2f7684 | |||
| 3b1dfb7562 | |||
| 0a6ffd48cb | |||
| 567dae83cf | |||
| 462a4d296f | |||
| 8373dea7fb | |||
| b13081d1a6 | |||
| 8b38b89647 | |||
| 3a96f48ec5 | |||
| aa8db280e5 | |||
|
a78bb9123a
|
|||
|
881ccfd820
|
|||
|
1adca5ca0e
|
|||
|
9a0bd6dc11
|
|||
|
2aa9029893
|
|||
|
a2a836c2a9
|
|||
|
a7d748cb1f
|
|||
|
9e52e9f676
|
@@ -0,0 +1,7 @@
|
|||||||
|
*.swo
|
||||||
|
*.swp
|
||||||
|
.dockerignore
|
||||||
|
Dockerfile
|
||||||
|
abra
|
||||||
|
dist
|
||||||
|
tags
|
||||||
+100
-67
@@ -1,88 +1,121 @@
|
|||||||
---
|
---
|
||||||
kind: pipeline
|
kind: pipeline
|
||||||
name: linters
|
name: coopcloud.tech/abra
|
||||||
steps:
|
steps:
|
||||||
- name: run shellcheck
|
- name: make check
|
||||||
image: koalaman/shellcheck-alpine
|
image: golang:1.26
|
||||||
commands:
|
commands:
|
||||||
- shellcheck abra
|
- make check
|
||||||
- shellcheck bin/*.sh
|
|
||||||
- shellcheck deploy/install.abra.coopcloud.tech/installer
|
|
||||||
|
|
||||||
- name: run flake8
|
- name: xgettext-go
|
||||||
image: alpine/flake8
|
image: git.coopcloud.tech/toolshed/drone-xgettext-go:latest
|
||||||
commands:
|
settings:
|
||||||
- flake8 --max-line-length 100 bin/*.py
|
keyword: i18n.G
|
||||||
|
keyword_ctx: i18n.GC
|
||||||
|
out: pkg/i18n/locales/abra.pot
|
||||||
|
comments_tag: translators
|
||||||
|
depends_on:
|
||||||
|
- make check
|
||||||
|
when:
|
||||||
|
event:
|
||||||
|
exclude:
|
||||||
|
- tag
|
||||||
|
|
||||||
- name: run unit tests
|
- name: xgettext-go status
|
||||||
image: decentral1se/docker-dind-bats-kcov
|
image: golang:1.26-alpine3.22
|
||||||
commands:
|
commands:
|
||||||
- bats tests
|
- apk add patchutils git make
|
||||||
|
- cd /drone/src
|
||||||
|
- sed -i "s/charset=CHARSET/charset=UTF-8/g" pkg/i18n/locales/*.pot
|
||||||
|
- git diff pkg/i18n/locales/abra.pot | grepdiff --output-matching=hunk POT-Creation-Date | git apply --reverse --allow-empty
|
||||||
|
- git diff
|
||||||
|
- git diff-files --exit-code
|
||||||
|
depends_on:
|
||||||
|
- xgettext-go
|
||||||
|
when:
|
||||||
|
event:
|
||||||
|
exclude:
|
||||||
|
- tag
|
||||||
|
|
||||||
- name: test installation script
|
- name: make test
|
||||||
image: debian:buster
|
image: golang:1.26
|
||||||
|
environment:
|
||||||
|
ABRA_DIR: /root/.abra_test
|
||||||
commands:
|
commands:
|
||||||
- apt update && apt install -yqq sudo lsb-release
|
- make test
|
||||||
- deploy/install.abra.coopcloud.tech/installer --no-prompt
|
depends_on:
|
||||||
- ~/.local/bin/abra version
|
- make check
|
||||||
|
|
||||||
- name: publish image
|
- name: publish image
|
||||||
image: plugins/docker
|
image: plugins/docker
|
||||||
settings:
|
settings:
|
||||||
auto_tag: true
|
auto_tag: true
|
||||||
username: thecoopcloud
|
username: abra-bot
|
||||||
password:
|
password:
|
||||||
from_secret: thecoopcloud_password
|
from_secret: git_coopcloud_tech_token_abra_bot
|
||||||
repo: thecoopcloud/abra
|
repo: git.coopcloud.tech/toolshed/abra
|
||||||
tags: latest
|
tags: dev
|
||||||
|
registry: git.coopcloud.tech
|
||||||
|
when:
|
||||||
|
branch:
|
||||||
|
- main
|
||||||
depends_on:
|
depends_on:
|
||||||
- run shellcheck
|
- make check
|
||||||
- run flake8
|
- make test
|
||||||
- run unit tests
|
|
||||||
- test installation script
|
- name: on-demand integration test
|
||||||
|
image: appleboy/drone-ssh
|
||||||
|
settings:
|
||||||
|
host:
|
||||||
|
- int.coopcloud.tech
|
||||||
|
username: abra
|
||||||
|
key:
|
||||||
|
from_secret: abra_int_private_key
|
||||||
|
port: 22
|
||||||
|
command_timeout: 60m
|
||||||
|
script_stop: true
|
||||||
|
request_pty: true
|
||||||
|
script:
|
||||||
|
- |
|
||||||
|
wget https://git.coopcloud.tech/toolshed/abra/raw/branch/main/scripts/tests/run-ci-int -O run-ci-int
|
||||||
|
chmod +x run-ci-int
|
||||||
|
sh run-ci-int
|
||||||
|
when:
|
||||||
|
ref:
|
||||||
|
- refs/heads/int-*
|
||||||
|
depends_on:
|
||||||
|
- make check
|
||||||
|
- make test
|
||||||
|
|
||||||
|
- name: nightly integration test
|
||||||
|
image: appleboy/drone-ssh
|
||||||
|
settings:
|
||||||
|
host:
|
||||||
|
- int.coopcloud.tech
|
||||||
|
username: abra
|
||||||
|
key:
|
||||||
|
from_secret: abra_int_private_key
|
||||||
|
port: 22
|
||||||
|
command_timeout: 60m
|
||||||
|
script_stop: true
|
||||||
|
request_pty: true
|
||||||
|
script:
|
||||||
|
- |
|
||||||
|
wget https://git.coopcloud.tech/toolshed/abra/raw/branch/main/scripts/tests/run-ci-int -O run-ci-int
|
||||||
|
chmod +x run-ci-int
|
||||||
|
sh run-ci-int
|
||||||
when:
|
when:
|
||||||
event:
|
event:
|
||||||
exclude:
|
- cron:
|
||||||
- pull_request
|
cron:
|
||||||
|
# @daily https://docs.drone.io/cron/
|
||||||
|
- integration
|
||||||
|
|
||||||
- name: trigger downstream builds
|
volumes:
|
||||||
image: plugins/downstream
|
- name: deps
|
||||||
settings:
|
temp: {}
|
||||||
server: https://drone.autonomic.zone
|
|
||||||
token:
|
|
||||||
from_secret: decentral1se_token
|
|
||||||
fork: true
|
|
||||||
repositories:
|
|
||||||
- coop-cloud/drone-abra
|
|
||||||
depends_on:
|
|
||||||
- run shellcheck
|
|
||||||
- run flake8
|
|
||||||
- run unit tests
|
|
||||||
- test installation script
|
|
||||||
- publish image
|
|
||||||
when:
|
|
||||||
event:
|
|
||||||
exclude:
|
|
||||||
- pull_request
|
|
||||||
|
|
||||||
- name: notify on failure
|
|
||||||
image: plugins/matrix
|
|
||||||
settings:
|
|
||||||
homeserver: https://matrix.autonomic.zone
|
|
||||||
roomid: "IFazIpLtxiScqbHqoa:autonomic.zone"
|
|
||||||
userid: "@autono-bot:autonomic.zone"
|
|
||||||
accesstoken:
|
|
||||||
from_secret: autono_bot_access_token
|
|
||||||
depends_on:
|
|
||||||
- run shellcheck
|
|
||||||
- run flake8
|
|
||||||
- run unit tests
|
|
||||||
- test installation script
|
|
||||||
- publish image
|
|
||||||
- trigger downstream builds
|
|
||||||
when:
|
|
||||||
status:
|
|
||||||
- failure
|
|
||||||
trigger:
|
trigger:
|
||||||
branch:
|
action:
|
||||||
- main
|
exclude:
|
||||||
|
- synchronized
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
# integration test suite
|
||||||
|
# export ABRA_DIR="$HOME/.abra_test"
|
||||||
|
# export TEST_SERVER=test.example.com
|
||||||
|
# export ABRA_CI=1
|
||||||
|
|
||||||
|
# release automation
|
||||||
|
# export GITEA_TOKEN=
|
||||||
+9
-5
@@ -1,5 +1,9 @@
|
|||||||
*.json
|
*.tar.gz
|
||||||
*.pyc
|
*fmtcoverage.html
|
||||||
/.venv
|
.e2e.env
|
||||||
__pycache__
|
.envrc
|
||||||
coverage/
|
.vscode/
|
||||||
|
/abra
|
||||||
|
/bin
|
||||||
|
dist/
|
||||||
|
tests/integration/.bats
|
||||||
|
|||||||
@@ -0,0 +1,56 @@
|
|||||||
|
---
|
||||||
|
version: 2
|
||||||
|
|
||||||
|
gitea_urls:
|
||||||
|
api: https://git.coopcloud.tech/api/v1
|
||||||
|
download: https://git.coopcloud.tech/
|
||||||
|
skip_tls_verify: false
|
||||||
|
|
||||||
|
before:
|
||||||
|
hooks:
|
||||||
|
- go mod tidy
|
||||||
|
|
||||||
|
builds:
|
||||||
|
- id: abra
|
||||||
|
binary: abra
|
||||||
|
dir: cmd/abra
|
||||||
|
env:
|
||||||
|
- CGO_ENABLED=0
|
||||||
|
goos:
|
||||||
|
- linux
|
||||||
|
- darwin
|
||||||
|
goarch:
|
||||||
|
- 386
|
||||||
|
- amd64
|
||||||
|
- arm
|
||||||
|
- arm64
|
||||||
|
goarm:
|
||||||
|
- 5
|
||||||
|
- 6
|
||||||
|
- 7
|
||||||
|
flags:
|
||||||
|
- -v
|
||||||
|
- -trimpath
|
||||||
|
ldflags:
|
||||||
|
- "-X 'main.Commit={{ .Commit }}'"
|
||||||
|
- "-X 'main.Version={{ .Version }}'"
|
||||||
|
- "-s"
|
||||||
|
- "-w"
|
||||||
|
|
||||||
|
checksum:
|
||||||
|
name_template: "checksums.txt"
|
||||||
|
|
||||||
|
snapshot:
|
||||||
|
name_template: "{{ incpatch .Version }}-next"
|
||||||
|
|
||||||
|
changelog:
|
||||||
|
sort: desc
|
||||||
|
filters:
|
||||||
|
exclude:
|
||||||
|
- "^Merge"
|
||||||
|
- "^Revert"
|
||||||
|
- "^WIP:"
|
||||||
|
- "^chore(deps):"
|
||||||
|
- "^style:"
|
||||||
|
- "^test:"
|
||||||
|
- "^tests:"
|
||||||
+28
@@ -0,0 +1,28 @@
|
|||||||
|
# authors
|
||||||
|
|
||||||
|
> If you're looking at this and you hack on `abra` and you're not listed here,
|
||||||
|
> please do add yourself! This is a community project, let's show some 💞
|
||||||
|
|
||||||
|
- 3wordchant
|
||||||
|
- ammaratef45
|
||||||
|
- apfelwurm
|
||||||
|
- basebuilder
|
||||||
|
- cassowary
|
||||||
|
- chasqui
|
||||||
|
- codegod100
|
||||||
|
- cyrnel
|
||||||
|
- decentral1se
|
||||||
|
- fauno
|
||||||
|
- frando
|
||||||
|
- iexos
|
||||||
|
- jade
|
||||||
|
- kawaiipunk
|
||||||
|
- knoflook
|
||||||
|
- mayel
|
||||||
|
- moritz
|
||||||
|
- namnatulco
|
||||||
|
- p4u1
|
||||||
|
- rix
|
||||||
|
- roxxers
|
||||||
|
- vera
|
||||||
|
- yksflip
|
||||||
-195
@@ -1,195 +0,0 @@
|
|||||||
> 🔥 🔥 🔥 Please note, while we are still in
|
|
||||||
> [public alpha](https://docs.cloud.autonomic.zone/roadmap/), the `abra` release
|
|
||||||
> versioning scheme is not following [semver](https://semver.org/) conventions
|
|
||||||
> because we are still in the exploratory phases of building this tool. Please
|
|
||||||
> read the changes before upgrading your `abra` installation as there are
|
|
||||||
> **most likely** breaking changes coming each release. Sorry for any
|
|
||||||
> inconvenience caused, we're working hard to make this tool stable. Semver
|
|
||||||
> will be respected when we reach public beta. 🔥 🔥 🔥
|
|
||||||
|
|
||||||
# abra x.x.x (UNRELEASED)
|
|
||||||
|
|
||||||
# abra 10.0.1 (2021-07-30)
|
|
||||||
|
|
||||||
- New `recipe .. lint` command ([#202](https://git.autonomic.zone/coop-cloud/abra/issues/202))
|
|
||||||
- `abra-capsul` plugin ([e85bcc4](https://git.autonomic.zone/coop-cloud/abra/commit/e85bcc4))
|
|
||||||
- Fix `run <service> <cmd>` invocations ([#204](https://git.autonomic.zone/coop-cloud/abra/issues/204))
|
|
||||||
- Point installer at our [half-migrated new Gitea instance](https://git.coopcloud.tech) ([#207](https://git.coopcloud.tech/coop-cloud/abra/pulls/207) thanks @nicksellen!)
|
|
||||||
|
|
||||||
# abra 10.0.0 (2021-07-06)
|
|
||||||
|
|
||||||
- Add `--bump` to `deploy` command to allow packagers to make minor package related releases ([#173](https://git.autonomic.zone/coop-cloud/abra/issues/173))
|
|
||||||
- Drop `--skip-version-check`/`--no-domain-poll`/`--no-state-poll` in favour of `--fast` ([#169](https://git.autonomic.zone/coop-cloud/abra/issues/169))
|
|
||||||
- Move `abra` image under the new `thecoopcloud/...` namespace ([#1](https://git.autonomic.zone/coop-cloud/auto-apps-json/issues/1))
|
|
||||||
- Add a `--output` flag to the `app-json.py` app generator for the CI environment ([#2](https://git.autonomic.zone/coop-cloud/auto-apps-json/issues/2))
|
|
||||||
- Support logging in as new `thecoopcloud` Docker account via `skopeo` when generating new `apps.json` ([`7482362`](https://git.autonomic.zone/coop-cloud/abra/commit/7482362af1d01cc02828abd45b1222fa643d1f80))
|
|
||||||
- App deployment checks are somewhat more reliable (see [#193](https://git.autonomic.zone/coop-cloud/abra/issues/193) for remaining work) ([#165](https://git.autonomic.zone/coop-cloud/abra/issues/165))
|
|
||||||
- Skip generation of commented out secrets and correctly fail deploy when secret generation fails ([#133](https://git.autonomic.zone/coop-cloud/abra/issues/133))
|
|
||||||
- Fix logging for chaos deploys and recipe selection logic ([#185](https://git.autonomic.zone/coop-cloud/abra/issues/185))
|
|
||||||
- Improve reliability of selecting when to download a new `apps.json` ([#170](https://git.autonomic.zone/coop-cloud/abra/issues/170))
|
|
||||||
- Implement `pwgen`/`pwqgen` native fallback for password generation ([#167](https://git.autonomic.zone/coop-cloud/abra/issues/167) / [#197](https://git.autonomic.zone/coop-cloud/abra/issues/197))
|
|
||||||
- `abra` installer script will now try to install system requirements ([#196](https://git.autonomic.zone/coop-cloud/abra/issues/196))
|
|
||||||
- Use latest [v4.9.6](https://github.com/mikefarah/yq/releases/tag/v4.9.6) install of `yq` for vendoring (**upgrade HOWTO:** `rm -rf ~/.abra/vendor/*`)
|
|
||||||
- Support overriding `$ARGS` from `abra.sh` custom commands and error out correctly when these commands fail ([#1](https://github.com/Coop-Cloud/peertube/issues/1))
|
|
||||||
- Add `abra <app> restart <service>` to support restarting individual services ([#200](https://git.autonomic.zone/coop-cloud/abra/issues/200))
|
|
||||||
- Output diff of proposed changes when asking to commit during release logic ([`4b82045`](https://git.autonomic.zone/coop-cloud/abra/commit/4b820457defe1511208b6caa8b9feb9603ffc8be))
|
|
||||||
- Add validation for generated output when making new release labels ([#186](https://git.autonomic.zone/coop-cloud/abra/issues/186))
|
|
||||||
|
|
||||||
# abra 9.0.0 (2021-06-10)
|
|
||||||
|
|
||||||
- Add Docker image for `abra` ([`64d578cf`](https://git.autonomic.zone/coop-cloud/abra/commit/64d578cf914bd2bad378ea4ef375747d10b33191))
|
|
||||||
- Support unattended mode for recipe releasing ([`3759bcd6`](https://git.autonomic.zone/coop-cloud/abra/commit/3759bcd641cf60611c13927e83425e773d2bb629))
|
|
||||||
- Add Renovate bot configuraiton script ([`9fadc430`](https://git.autonomic.zone/coop-cloud/abra/commit/9fadc430a7bb2d554c0ee26c0f9b6c51dc5b0475))
|
|
||||||
- Add release automation via [drone-abra](https://git.autonomic.zone/coop-cloud/drone-abra) ([#56](https://git.autonomic.zone/coop-cloud/organising/issues/56))
|
|
||||||
- Move `apps.json` generation to [auto-apps-json](https://git.autonomic.zone/coop-cloud/auto-apps-json) ([#125](https://git.autonomic.zone/coop-cloud/abra/issues/125))
|
|
||||||
- Add Github mirroring script ([`4ef43331`](https://git.autonomic.zone/coop-cloud/abra/commit/4ef433312dd0b0ace91b3c285f82f3973093d92d))
|
|
||||||
- Add `--chaos` flag to deploy (always choose latest Git commit) ([#178](https://git.autonomic.zone/coop-cloud/abra/issues/178))
|
|
||||||
|
|
||||||
# abra 8.0.1 (2021-05-31)
|
|
||||||
|
|
||||||
- Fix help for `... app ... volume ls` ([`efad71c4`](https://git.autonomic.zone/coop-cloud/abra/commit/efad71c470d6d65f7e4bfe39c5f68ff1097f80a2))
|
|
||||||
- Only output secrets warnings once ([#143](https://git.autonomic.zone/coop-cloud/abra/issues/143))
|
|
||||||
- Migrate `abra` installation script to `coopcloud.tech` domain ([#150](https://git.autonomic.zone/coop-cloud/abra/issues/150))
|
|
||||||
- Add `--no-state-poll` to avoid success/failure forecasting on deployment ([#165](https://git.autonomic.zone/coop-cloud/abra/issues/165))
|
|
||||||
|
|
||||||
# abra 8.0.0 (2021-05-30)
|
|
||||||
|
|
||||||
- Fix secret length generation ([`f537417`](https://git.autonomic.zone/coop-cloud/abra/commit/1b85bf3d37280e9632c315d759c0f2d09c039fef))
|
|
||||||
- Fix checking out new apps ([#164](https://git.autonomic.zone/coop-cloud/abra/issues/164)
|
|
||||||
- Give up if YAML is invalid ([#154](https://git.autonomic.zone/coop-cloud/abra/issues/154))
|
|
||||||
- Switch from wget to cURL ([`fc0caaa`](https://git.autonomic.zone/coop-cloud/abra/commit/fc0caaa))
|
|
||||||
- Add Bash completion for `recipe ..` ([`8c93d1a`](https://git.autonomic.zone/coop-cloud/abra/commit/8c93d1a))
|
|
||||||
- Tweak README parsing in `app-json.py` ([`b14219b`](https://git.autonomic.zone/coop-cloud/abra/commit/b14219b))
|
|
||||||
- Add fallback names to `app.json` ([#157](https://git.autonomic.zone/coop-cloud/abra/issues/157))
|
|
||||||
- Remove duplicate message ([#155](https://git.autonomic.zone/coop-cloud/abra/issues/155))
|
|
||||||
- Add `deploy --fast` ([`a7f7c96`](https://git.autonomic.zone/coop-cloud/abra/commit/a7f7c96))
|
|
||||||
- Add `app .. volume` commands, fix volume deletion with `app .. delete --volumes` ([#161](https://git.autonomic.zone/coop-cloud/abra/issues/161))
|
|
||||||
|
|
||||||
# abra 0.7.4 (2021-05-10)
|
|
||||||
|
|
||||||
- Sort `apps.json` when publishing ([`39a7fc0`](https://git.autonomic.zone/coop-cloud/abra/commit/39a7fc04fb5df1a6d78b84f51838530ab3eb76db))
|
|
||||||
- Fix publishing of rating for new apps ([`0e28af9`](https://git.autonomic.zone/coop-cloud/abra/commit/0e28af9eb1af6c6da705b4614ddd173c60576629))
|
|
||||||
- Detect compose filenames in `n+1` release generation ([`ffc569e`](https://git.autonomic.zone/coop-cloud/abra/commit/ffc569e275df7ca784a4db1a3331e17975fd8c87))
|
|
||||||
- Fix secret generation when specifying length ([`3a353f4`](https://git.autonomic.zone/coop-cloud/abra/commit/3a353f4062baccde2c9f175b03afb2db6d462ae4))
|
|
||||||
|
|
||||||
# abra 0.7.3 (2021-04-28)
|
|
||||||
|
|
||||||
- Only check for pw(q)gen if we're actually trying to use them ([#147](https://git.autonomic.zone/coop-cloud/abra/issues/147))
|
|
||||||
- Use apps.coopcloud.tech for app data hosting & download ([`75bd599`](https://git.autonomic.zone/coop-cloud/abra/commit/75bd599))
|
|
||||||
- Choose latest commit messages for new tags ([#144](https://git.autonomic.zone/coop-cloud/abra/issues/144))
|
|
||||||
- Handle recipes without an `app` service in `recipe .. release` ([#151](https://git.autonomic.zone/coop-cloud/abra/issues/151))
|
|
||||||
|
|
||||||
# abra 0.7.2 (2021-04-07)
|
|
||||||
|
|
||||||
- Fix installation script development installs (again! Thanks Bash!) ([`4747d9b7`](https://git.autonomic.zone/coop-cloud/abra/commit/4747d9b7fb5fba914f210b6570bfe2db0b53da23))
|
|
||||||
|
|
||||||
# abra 0.7.1 (2021-04-07)
|
|
||||||
|
|
||||||
- Fix installation script development installs ([`8f2fadb3c`](https://git.autonomic.zone/coop-cloud/abra/commit/8f2fadb3c43c5915520f5ea531ea3815c2ba8531))
|
|
||||||
|
|
||||||
# abra 0.7.0 (2021-04-07)
|
|
||||||
|
|
||||||
- Add `--force` to the `deploy` command to allow overriding deployment logic ([#105](https://git.autonomic.zone/coop-cloud/abra/issues/105))
|
|
||||||
- Handle undeployed apps in version summaries when deploying ([#104](https://git.autonomic.zone/coop-cloud/abra/issues/104))
|
|
||||||
- Add `--force` to `undeploy` command ([`e5e98d5`](https://git.autonomic.zone/coop-cloud/abra/commit/e5e98d5))
|
|
||||||
- Rename "app type" back to "stack" in the deployment overview ([`54b6acc`](https://git.autonomic.zone/coop-cloud/abra/commit/54b6acc))
|
|
||||||
- Show context connection details on `abra server ls` ([#110](https://git.autonomic.zone/coop-cloud/abra/issues/110))
|
|
||||||
- Allow to debug the SSH connection details on swarm init ([#109](https://git.autonomic.zone/coop-cloud/abra/issues/109))
|
|
||||||
- Show correct status for apps deployed on servers with missing context ([#99](https://git.autonomic.zone/coop-cloud/abra/issues/99))
|
|
||||||
- Search for subcommands in descending order of how many components there are ([#108](https://git.autonomic.zone/coop-cloud/abra/issues/108))
|
|
||||||
- Add specific app version checking command (`abra app <app> version`) ([#108](https://git.autonomic.zone/coop-cloud/abra/issues/108))
|
|
||||||
- Add docker version check (guestimating < v19 is a bad idea) ([#15](https://git.autonomic.zone/coop-cloud/abra/issues/15))
|
|
||||||
- Fix git branch handling when not passing `-b <branch>` ([#122](https://git.autonomic.zone/coop-cloud/abra/issues/122))
|
|
||||||
- Add work-around to correctly git clone non-master default branch app repositories ([#122](https://git.autonomic.zone/coop-cloud/abra/issues/122))
|
|
||||||
- Replace `--force` (except for the `deploy` command) with a global `--no-prompt` for avoiding interactive questions ([#118](https://git.autonomic.zone/coop-cloud/abra/issues/118))
|
|
||||||
- Use [docker-stack-wait-deploy](https://github.com/vitalets/docker-stack-wait-deploy) inspired logic to deploy apps ([#116](https://git.autonomic.zone/coop-cloud/abra/issues/116))
|
|
||||||
- Add a domain polling check when deploying apps ([#113](https://git.autonomic.zone/coop-cloud/abra/issues/113))
|
|
||||||
- Recognise when apps are already undeployed with `abra app <app> undeploy` ([#123](https://git.autonomic.zone/coop-cloud/abra/issues/123))
|
|
||||||
- Add `abra doctor` command to help diagnose setup issues ([#119](https://git.autonomic.zone/coop-cloud/abra/issues/119))
|
|
||||||
- Add apps version and feature catalogue generation script ([#121](https://git.autonomic.zone/coop-cloud/abra/issues/121))
|
|
||||||
- New `--skip-version-check` option to `deploy` ([`df4e504`](https://git.autonomic.zone/coop-cloud/abra/commit/df4e504))
|
|
||||||
- Look up local available version from compose files instead of `abra.sh` ([#131](https://git.autonomic.zone/coop-cloud/abra/issues/131))
|
|
||||||
- Improve domain polling logging and allow to skip the check altogether with `--no-domain-poll` ([#140](https://git.autonomic.zone/coop-cloud/abra/issues/140), [#141](https://git.autonomic.zone/coop-cloud/abra/issues/141))
|
|
||||||
- Support `ABRA_DIR` in the installer script ([`4e94a424e`](https://git.autonomic.zone/coop-cloud/abra/commit/4e94a424e94a42))
|
|
||||||
- Support [abra-hetzner](https://git.autonomic.zone/coop-cloud/abra-hetzner) plugin ([#88](https://git.autonomic.zone/coop-cloud/abra/issues/88))
|
|
||||||
|
|
||||||
# abra 0.6.0 (2021-03-17)
|
|
||||||
|
|
||||||
- Show version and digest of app if labelled ([`98e674b8e`](https://git.autonomic.zone/coop-cloud/abra/commit/98e674b8e83458a83dcbf331e8e34c7188559c4a))
|
|
||||||
- Implement basic version checking on deployment ([#82](https://git.autonomic.zone/coop-cloud/abra/issues/82))
|
|
||||||
- New `app-catalogue.sh` script to auto-generate app list for documentation ([`f163d4b0f`](https://git.autonomic.zone/coop-cloud/abra/commit/f163d4b0fa920232e9d995a22d20fe78b174b3a9))
|
|
||||||
- Support app service rollbacks with `abra <app> rollback <service>` ([#76](https://git.autonomic.zone/coop-cloud/abra/issues/76))
|
|
||||||
- Detect when latest version is deployed and perform a no-op ([#87](https://git.autonomic.zone/coop-cloud/abra/issues/87))
|
|
||||||
- Allow cloning of app repos with different main branches using `-b, --branch=<branch>` ([#80](https://git.autonomic.zone/coop-cloud/abra/issues/80))
|
|
||||||
- Protect against lengthy app names which gives Docker trouble later on ([#83](https://git.autonomic.zone/coop-cloud/abra/issues/83))
|
|
||||||
- Support removal of secrets and volumes when `rm`'ing apps ([#44](https://git.autonomic.zone/coop-cloud/abra/issues/44))
|
|
||||||
- Always choose the default IPv4 address with `abra server <host> init` ([#91](https://git.autonomic.zone/coop-cloud/abra/issues/91))
|
|
||||||
- Add `--type=<type>` filtering option to `abra <app> ls` ([`0828189`](https://git.autonomic.zone/coop-cloud/abra/commit/0828189))
|
|
||||||
- Check for bash 4+ ([#96](https://git.autonomic.zone/coop-cloud/abra/commit/0828189))
|
|
||||||
- Add `--dev` option to installer using `git clone` ([`88d2a75`](https://git.autonomic.zone/coop-cloud/abra/commit/88d2a75))
|
|
||||||
- Support `--dev` on the `abra upgrade` command also ([`bcc15ec`](https://git.autonomic.zone/coop-cloud/abra/commit/bcc15ec))
|
|
||||||
- Vendor [yq](https://github.com/mikefarah/yq/releases) automatically ([`3b59adf`](https://git.autonomic.zone/coop-cloud/abra/commit/3b59adf))
|
|
||||||
- Extend version handling logic to support all underlying services ([#90](https://git.autonomic.zone/coop-cloud/abra/issues/90))
|
|
||||||
- Fix development installation script symlink issue ([#98](https://git.autonomic.zone/coop-cloud/abra/issues/98))
|
|
||||||
- Add `app-version.sh` script to help packagers version apps ([`28618bd`](https://git.autonomic.zone/coop-cloud/abra/commit/28618bd))
|
|
||||||
- Add git digest to `abra version` output ([`8b41416`](https://git.autonomic.zone/coop-cloud/abra/commit/8b41416))
|
|
||||||
|
|
||||||
# abra 0.5.0 (2021-03-01)
|
|
||||||
|
|
||||||
- `secret auto` merged into `secret generate` and `app new --auto` is now `app new --secrets` ([#64](https://git.autonomic.zone/coop-cloud/abra/pulls/64))
|
|
||||||
- Avoid outputting length during secret generation when not in use ([#67](https://git.autonomic.zone/coop-cloud/abra/issues/67))
|
|
||||||
- Support graceful failure when missing secret generation commands ([`44d3ac3`](https://git.autonomic.zone/coop-cloud/abra/commit/44d3ac3a1cb86edc9b9e91eea1a00e70eae14965))
|
|
||||||
- Fix secret detection when using new `.env` file format in apps ([`5532452`](https://git.autonomic.zone/coop-cloud/abra/commit/55324524ca77141666ffe6cc41b62cc71cf89ace))
|
|
||||||
- Support choosing an `$EDITOR` when editing configs ([`29cc392`](https://git.autonomic.zone/coop-cloud/abra/commit/29cc392dff3e93e48e0e2edd3ce11b405c66a95a))
|
|
||||||
- "server" shell completion fixed ([`8839bd4`](https://git.autonomic.zone/coop-cloud/abra/commit/8839bd45951d00dccf4ef81ece445bcc49e13ee6))
|
|
||||||
- Drop `multilogs` command ([#56](https://git.autonomic.zone/coop-cloud/abra/pulls/56))
|
|
||||||
- Remove `server use` command ([#51](https://git.autonomic.zone/coop-cloud/abra/issues/51))
|
|
||||||
- `new <app>` becomes `new <type>` ([#48](https://git.autonomic.zone/coop-cloud/abra/issues/48))
|
|
||||||
- `check` is run on `deploy` now and configurable ([`77ba565`](https://git.autonomic.zone/coop-cloud/abra/commit/77ba5652b2fe15820f5edfa0f642636f7b8eae7e))
|
|
||||||
- App configurations are always updated now ([#42](https://git.autonomic.zone/coop-cloud/abra/issues/42))
|
|
||||||
- We use docker format `.env` files (no "export" syntax) from now now ([#55](https://git.autonomic.zone/coop-cloud/abra/pulls/55))
|
|
||||||
- Rename `<domain>` option to `<app>` and `APP` variable to `TYPE`, see ([#47](https://git.autonomic.zone/coop-cloud/abra/issues/47))
|
|
||||||
- Use Docker-in-Docker (dind), and `dind-bats-kcov` Docker image, for `make test` ([`1600b62`](https://git.autonomic.zone/coop-cloud/abra/commit/1600b6277fbbffc4c6de1e4ba799c7bbe72ec6a0))
|
|
||||||
- Add built-in documentation using `abra help <subcommand>...`, see ([#50](https://git.autonomic.zone/coop-cloud/abra/issues/50))
|
|
||||||
- `version` subcommand ([e6b24fe](https://git.autonomic.zone/coop-cloud/abra/commit/e6b24fe))
|
|
||||||
- Use `# length=x` comments to generate passwords with `pwgen` and drop `KEY`/`PASSWORD` logic ([#68](https://git.autonomic.zone/coop-cloud/abra/issues/68))
|
|
||||||
- Global `--skip-update|-U` / `--skip-check|-C` options to make things quicker ([`37e8b00`](https://git.autonomic.zone/coop-cloud/abra/commit/37e8b00))
|
|
||||||
- `app backup` and `app restore` commands; requires per-app definition ([#70](https://git.autonomic.zone/coop-cloud/abra/issues/70))
|
|
||||||
- Rename per-type `abra-commands.sh` to `abra.sh`, and include config versions as type-level instead of app-level config ([#43](https://git.autonomic.zone/coop-cloud/abra/issues/43))
|
|
||||||
- Show per-subcommand help by adding `-h/--help` to a command line ([#38](https://git.autonomic.zone/coop-cloud/abra/issues/78))
|
|
||||||
|
|
||||||
# abra 0.4.1 (2020-12-24)
|
|
||||||
|
|
||||||
- Bug-fixes on `app ls --status` & custom commands
|
|
||||||
- Add `app ls --server=...` and alias
|
|
||||||
|
|
||||||
# abra 0.4.0 (2020-12-24)
|
|
||||||
|
|
||||||
- New command-line interface based on docopt
|
|
||||||
- `~/.abra` directory instead of expecting local `.env` files
|
|
||||||
- Integration tests & code coverage
|
|
||||||
|
|
||||||
# abra 0.3.1 (2020-09-27)
|
|
||||||
|
|
||||||
- Fix installer version
|
|
||||||
|
|
||||||
# abra 0.3.0 (2020-09-27)
|
|
||||||
|
|
||||||
- Add multilogs stack logs implementation ([#8](https://git.autonomic.zone/compose-stacks/abra/issues/8))
|
|
||||||
- Add beginnings of "monorepo" functionality
|
|
||||||
|
|
||||||
# abra 0.2.0 (2020-09-24)
|
|
||||||
|
|
||||||
- Prepare for swarm install script using script.d ([#12](https://git.autonomic.zone/compose-stacks/planning/issues/12))
|
|
||||||
|
|
||||||
# abra 0.1.2 (2020-09-22)
|
|
||||||
|
|
||||||
- Add upgrade command ([#10](https://git.autonomic.zone/autonomic-cooperative/abra/issues/10))
|
|
||||||
|
|
||||||
# abra 0.1.1 (2020-09-22)
|
|
||||||
|
|
||||||
- Add installer script ([#9](https://git.autonomic.zone/autonomic-cooperative/abra/issues/9))
|
|
||||||
|
|
||||||
# abra 0.1.0 (2020-09-22)
|
|
||||||
|
|
||||||
- Initial pre-alpha release
|
|
||||||
+20
-24
@@ -1,33 +1,29 @@
|
|||||||
FROM alpine:latest
|
# Build image
|
||||||
|
FROM golang:1.26-alpine AS build
|
||||||
|
|
||||||
RUN apk add --upgrade --no-cache \
|
ENV GOPRIVATE=coopcloud.tech
|
||||||
bash \
|
|
||||||
curl \
|
RUN apk add --no-cache \
|
||||||
|
gcc \
|
||||||
git \
|
git \
|
||||||
grep \
|
make \
|
||||||
openssh-client \
|
musl-dev
|
||||||
py3-requests \
|
|
||||||
skopeo \
|
|
||||||
util-linux
|
|
||||||
|
|
||||||
RUN mkdir -p ~./local/bin
|
COPY . /app
|
||||||
RUN mkdir -p ~/.abra/apps
|
|
||||||
RUN mkdir -p ~/.abra/vendor
|
|
||||||
RUN mkdir -p ~/.ssh/
|
|
||||||
|
|
||||||
RUN ssh-keyscan -p 2222 git.autonomic.zone > ~/.ssh/known_hosts
|
WORKDIR /app
|
||||||
|
|
||||||
RUN curl -L https://github.com/stedolan/jq/releases/download/jq-1.6/jq-linux64 --output ~/.abra/vendor/jq
|
RUN CGO_ENABLED=0 make build
|
||||||
RUN chmod +x ~/.abra/vendor/jq
|
|
||||||
|
|
||||||
RUN curl -L https://github.com/mikefarah/yq/releases/download/v4.9.3/yq_linux_amd64 --output ~/.abra/vendor/yq
|
FROM alpine:3.22
|
||||||
RUN chmod +x ~/.abra/vendor/yq
|
|
||||||
|
|
||||||
# Note(decentral1se): it is fine to always use the development branch because
|
RUN apk add --no-cache \
|
||||||
# our Drone CI docker auto-tagger will publish official release tags and
|
ca-certificates \
|
||||||
# otherwise give us the latest abra on the latest tag
|
git \
|
||||||
RUN curl https://install.abra.coopcloud.tech | bash -s -- --dev
|
openssh
|
||||||
|
|
||||||
COPY bin/* /root/.local/bin/
|
RUN update-ca-certificates
|
||||||
|
|
||||||
ENTRYPOINT ["/root/.local/bin/abra"]
|
COPY --from=build /app/abra /abra
|
||||||
|
|
||||||
|
ENTRYPOINT ["/abra"]
|
||||||
|
|||||||
@@ -0,0 +1,15 @@
|
|||||||
|
Abra: The Co-op Cloud utility belt
|
||||||
|
Copyright (C) 2022 Co-op Cloud <helo@coopcloud.tech>
|
||||||
|
|
||||||
|
This program is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
@@ -0,0 +1,91 @@
|
|||||||
|
ABRA := ./cmd/abra
|
||||||
|
XGETTEXT := ./bin/xgettext-go
|
||||||
|
COMMIT := $(shell git rev-list -1 HEAD)
|
||||||
|
GOPATH := $(shell go env GOPATH)
|
||||||
|
GOVERSION := 1.26
|
||||||
|
LDFLAGS := "-X 'main.Commit=$(COMMIT)'"
|
||||||
|
DIST_LDFLAGS := $(LDFLAGS)" -s -w"
|
||||||
|
BFLAGS := -v -trimpath
|
||||||
|
GCFLAGS := "all=-l -B"
|
||||||
|
DOMAIN := abra
|
||||||
|
POFILES := $(wildcard pkg/i18n/locales/*.po)
|
||||||
|
MOFILES := $(patsubst %.po,%.mo,$(POFILES))
|
||||||
|
LINGUAS := $(basename $(POFILES))
|
||||||
|
|
||||||
|
export GOPRIVATE=coopcloud.tech
|
||||||
|
|
||||||
|
all: format check build
|
||||||
|
|
||||||
|
run:
|
||||||
|
@go run -gcflags=$(GCFLAGS) -ldflags=$(LDFLAGS) $(ABRA)
|
||||||
|
|
||||||
|
install:
|
||||||
|
@go install -gcflags=$(GCFLAGS) -ldflags=$(LDFLAGS) $(ABRA)
|
||||||
|
|
||||||
|
build:
|
||||||
|
@go build $(BFLAGS) -gcflags=$(GCFLAGS) -ldflags=$(DIST_LDFLAGS) $(ABRA)
|
||||||
|
|
||||||
|
build-docker:
|
||||||
|
@docker run -it -v $(PWD):/abra golang:$(GOVERSION) \
|
||||||
|
bash -c 'cd /abra; ./scripts/docker/build.sh'
|
||||||
|
|
||||||
|
clean:
|
||||||
|
@rm '$(GOPATH)/bin/abra'
|
||||||
|
|
||||||
|
format:
|
||||||
|
@gofmt -s -w $$(find . -type f -name '*.go' | grep -v "/vendor/")
|
||||||
|
|
||||||
|
check:
|
||||||
|
@test -z $$(gofmt -l $$(find . -type f -name '*.go' | grep -v "/vendor/")) || \
|
||||||
|
(echo "gofmt: formatting issue - run 'make format' to resolve" && exit 1)
|
||||||
|
|
||||||
|
test:
|
||||||
|
@go test ./... -cover -v -p 1
|
||||||
|
|
||||||
|
find-tests:
|
||||||
|
@find . -name "*_test.go"
|
||||||
|
|
||||||
|
loc:
|
||||||
|
@find . -name "*.go" | xargs wc -l
|
||||||
|
|
||||||
|
deps:
|
||||||
|
@go get -t -u ./...
|
||||||
|
|
||||||
|
.PHONY: i18n
|
||||||
|
i18n: update-pot update-pot-po-metadata update-po build-mo
|
||||||
|
|
||||||
|
.PHONY: update-po
|
||||||
|
update-po:
|
||||||
|
@set -eu; \
|
||||||
|
for lang in $(LINGUAS); do \
|
||||||
|
msgmerge --backup=none -U $$lang.po pkg/i18n/locales/$(DOMAIN).pot; \
|
||||||
|
done
|
||||||
|
|
||||||
|
.PHONY: update-pot
|
||||||
|
update-pot: $(XGETTEXT)
|
||||||
|
@${XGETTEXT} \
|
||||||
|
-o pkg/i18n/locales/$(DOMAIN).pot \
|
||||||
|
--keyword=i18n.G \
|
||||||
|
--keyword-ctx=i18n.GC \
|
||||||
|
--sort-output \
|
||||||
|
--add-comments-tag="translators" \
|
||||||
|
$$(find . -name "*.go" -not -path "*vendor*" | sort)
|
||||||
|
|
||||||
|
${XGETTEXT}:
|
||||||
|
@mkdir -p ./bin && \
|
||||||
|
wget -O ./bin/xgettext-go https://git.coopcloud.tech/toolshed/xgettext-go/raw/branch/main/xgettext-go && \
|
||||||
|
chmod +x ./bin/xgettext-go
|
||||||
|
|
||||||
|
.PHONY: update-pot-po-metadata
|
||||||
|
update-pot-po-metadata:
|
||||||
|
@sed -i "s/charset=CHARSET/charset=UTF-8/g" pkg/i18n/locales/*.po pkg/i18n/locales/*.pot
|
||||||
|
|
||||||
|
.PHONY: build-mo
|
||||||
|
build-mo:
|
||||||
|
@set -eu; \
|
||||||
|
for lang in $(POFILES); do \
|
||||||
|
msgfmt $$lang -o $$(echo $$lang | sed 's/.po/.mo/g') --statistics; \
|
||||||
|
done
|
||||||
|
|
||||||
|
release:
|
||||||
|
@goreleaser release --clean
|
||||||
@@ -1,92 +1,14 @@
|
|||||||
# abra
|
# `abra`
|
||||||
|
|
||||||
[](https://drone.autonomic.zone/coop-cloud/abra)
|
[](https://build.coopcloud.tech/toolshed/abra)
|
||||||
|
[](https://goreportcard.com/report/git.coopcloud.tech/toolshed/abra)
|
||||||
> https://coopcloud.tech
|
[](https://pkg.go.dev/coopcloud.tech/abra)
|
||||||
|
[](https://translate.coopcloud.tech/engage/co-op-cloud/)
|
||||||
|
|
||||||
The Co-op Cloud utility belt 🎩🐇
|
The Co-op Cloud utility belt 🎩🐇
|
||||||
|
|
||||||
`abra` is a command-line tool for managing your own [Co-op Cloud](https://coopcloud.tech). It can provision new servers, create applications, deploy them, run backup and restore operations and a whole lot of other things. It is the go-to tool for day-to-day operations when managing a Co-op Cloud instance.
|
<a href="https://github.com/egonelbre/gophers"><img align="right" width="150" src="https://github.com/egonelbre/gophers/raw/master/.thumb/sketch/adventure/poking-fire.png"/></a>
|
||||||
|
|
||||||
## Change log
|
`abra` is the flagship client & command-line tool for Co-op Cloud. It has been developed specifically for the purpose of making the day-to-day operations of [operators](https://docs.coopcloud.tech/operators/) and [maintainers](https://docs.coopcloud.tech/maintainers/) pleasant & convenient. It is libre software, written in [Go](https://go.dev) and maintained and extended by the community 💖
|
||||||
|
|
||||||
> 🔥 🔥 🔥 Please note, while we are still in [public
|
Please see [docs.coopcloud.tech/abra](https://docs.coopcloud.tech/abra) for help on install, upgrade, hacking, troubleshooting & more!
|
||||||
> alpha](https://docs.coopcloud.tech/roadmap/), the `abra` release
|
|
||||||
> versioning scheme is not following [semver](https://semver.org/) conventions
|
|
||||||
> because we are still in the exploratory phases of building this tool. Please
|
|
||||||
> read the changes before upgrading your `abra` installation as there are
|
|
||||||
> **most likely** breaking changes coming each release. Sorry for any
|
|
||||||
> inconvenience caused, we're working hard to make this tool stable. Semver
|
|
||||||
> will be respected when we reach public beta. 🔥 🔥 🔥
|
|
||||||
|
|
||||||
See [CHANGELOG.md](./CHANGELOG.md).
|
|
||||||
|
|
||||||
## Documentation
|
|
||||||
|
|
||||||
> [docs.coopcloud.tech](https://docs.coopcloud.tech)
|
|
||||||
|
|
||||||
## Requirements
|
|
||||||
|
|
||||||
- `curl`
|
|
||||||
- `docker`
|
|
||||||
- `bash` >= 4
|
|
||||||
|
|
||||||
## Install
|
|
||||||
|
|
||||||
Install the latest stable release:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
curl https://install.abra.coopcloud.tech | bash
|
|
||||||
```
|
|
||||||
|
|
||||||
The source for this script is [here](./deploy/install.abra.coopcloud.tech/installer).
|
|
||||||
|
|
||||||
You can pass options to the script like so (e.g. install the bleeding edge development version):
|
|
||||||
|
|
||||||
```sh
|
|
||||||
curl https://install.abra.coopcloud.tech | bash -s -- --dev
|
|
||||||
```
|
|
||||||
|
|
||||||
Other options available are as follows:
|
|
||||||
|
|
||||||
- **--no-prompt**: non-interactive installation
|
|
||||||
- **--no-deps**: do not attempt to install [requirements](#requirements)
|
|
||||||
|
|
||||||
## Container
|
|
||||||
|
|
||||||
An [image](https://hub.docker.com/r/thecoopcloud/abra) is also provided.
|
|
||||||
|
|
||||||
```
|
|
||||||
docker run thecoopcloud/abra app ls
|
|
||||||
```
|
|
||||||
|
|
||||||
## Update
|
|
||||||
|
|
||||||
Run `abra upgrade` to automatically download and install the latest release
|
|
||||||
version.
|
|
||||||
|
|
||||||
To update the development version, run `abra upgrade --dev`.
|
|
||||||
|
|
||||||
## Hack
|
|
||||||
|
|
||||||
It's written in Bash version 4 or greater!
|
|
||||||
|
|
||||||
Install it via `curl https://install.abra.coopcloud.tech | bash -s -- --dev`, then you can hack on the source in `~/.abra/src`.
|
|
||||||
|
|
||||||
The command-line interface is generated via [docopt](http://docopt.org/). If you add arguments then you need to run `make docopt` ro regenerate the parser.
|
|
||||||
|
|
||||||
Please remember to update the [CHANGELOG](./CHANGELOG.md) when you make a change.
|
|
||||||
|
|
||||||
## Releasing
|
|
||||||
|
|
||||||
### `abra`
|
|
||||||
|
|
||||||
> [install.abra.coopcloud.tech](https://install.abra.coopcloud.tech)
|
|
||||||
|
|
||||||
- Change the `x.x.x` header in [CHANGELOG.md](./CHANGELOG.md) to reflect new version and mark date
|
|
||||||
- Update the version in [abra](./abra)
|
|
||||||
- Update the version in [deploy/install.abra.coopcloud.tech/installer](./deploy/install.abra.coopcloud.tech/installer)
|
|
||||||
- `git commit` the above changes and then tag it with `git tag <your-new-version>`
|
|
||||||
- `git push` and `git push --tags`
|
|
||||||
- Deploy a new installer script `make release-installer`
|
|
||||||
- Tell the world (CoTech forum, Matrix public channel, Autonomic mastodon, etc.)
|
|
||||||
|
|||||||
-112
@@ -1,112 +0,0 @@
|
|||||||
"""Shared utilities for bin/*.py scripts."""
|
|
||||||
|
|
||||||
from logging import DEBUG, basicConfig, getLogger
|
|
||||||
from os import chdir, mkdir
|
|
||||||
from os.path import exists, expanduser
|
|
||||||
from pathlib import Path
|
|
||||||
from shlex import split
|
|
||||||
from subprocess import check_output
|
|
||||||
from sys import exit
|
|
||||||
|
|
||||||
from requests import get
|
|
||||||
|
|
||||||
HOME_PATH = expanduser("~/")
|
|
||||||
CLONES_PATH = Path(f"{HOME_PATH}/.abra/apps").absolute()
|
|
||||||
REPOS_TO_SKIP = (
|
|
||||||
"abra",
|
|
||||||
"abra-apps",
|
|
||||||
"abra-capsul",
|
|
||||||
"abra-gandi",
|
|
||||||
"abra-hetzner",
|
|
||||||
"apps",
|
|
||||||
"auto-apps-json",
|
|
||||||
"auto-mirror",
|
|
||||||
"backup-bot",
|
|
||||||
"coopcloud.tech",
|
|
||||||
"coturn",
|
|
||||||
"docker-cp-deploy",
|
|
||||||
"docker-dind-bats-kcov",
|
|
||||||
"docs.coopcloud.tech",
|
|
||||||
"example",
|
|
||||||
"gardening",
|
|
||||||
"go-abra",
|
|
||||||
"organising",
|
|
||||||
"pyabra",
|
|
||||||
"radicle-seed-node",
|
|
||||||
"stack-ssh-deploy",
|
|
||||||
"swarm-cronjob",
|
|
||||||
"tyop",
|
|
||||||
)
|
|
||||||
YQ_PATH = Path(f"{HOME_PATH}/.abra/vendor/yq")
|
|
||||||
JQ_PATH = Path(f"{HOME_PATH}/.abra/vendor/jq")
|
|
||||||
|
|
||||||
log = getLogger(__name__)
|
|
||||||
basicConfig()
|
|
||||||
log.setLevel(DEBUG)
|
|
||||||
|
|
||||||
|
|
||||||
def _run_cmd(cmd, shell=False, **kwargs):
|
|
||||||
"""Run a shell command."""
|
|
||||||
args = [split(cmd)]
|
|
||||||
|
|
||||||
if shell:
|
|
||||||
args = [cmd]
|
|
||||||
kwargs = {"shell": shell}
|
|
||||||
|
|
||||||
try:
|
|
||||||
return check_output(*args, **kwargs).decode("utf-8").strip()
|
|
||||||
except Exception as exception:
|
|
||||||
log.error(f"Failed to run {cmd}, saw {str(exception)}")
|
|
||||||
exit(1)
|
|
||||||
|
|
||||||
|
|
||||||
def get_repos_json():
|
|
||||||
""" Retrieve repo list from Gitea """
|
|
||||||
|
|
||||||
url = "https://git.autonomic.zone/api/v1/orgs/coop-cloud/repos"
|
|
||||||
|
|
||||||
log.info(f"Retrieving {url}")
|
|
||||||
|
|
||||||
repos = []
|
|
||||||
response = True
|
|
||||||
page = 1
|
|
||||||
|
|
||||||
try:
|
|
||||||
while response:
|
|
||||||
log.info(f"Trying to fetch page {page}")
|
|
||||||
response = get(url + f"?page={page}", timeout=10).json()
|
|
||||||
repos.extend(response)
|
|
||||||
page += 1
|
|
||||||
|
|
||||||
return repos
|
|
||||||
except Exception as exception:
|
|
||||||
log.error(f"Failed to retrieve {url}, saw {str(exception)}")
|
|
||||||
exit(1)
|
|
||||||
|
|
||||||
|
|
||||||
def clone_all_apps(repos_json, ssh=False):
|
|
||||||
"""Clone all Co-op Cloud apps to ~/.abra/apps."""
|
|
||||||
if not exists(CLONES_PATH):
|
|
||||||
mkdir(CLONES_PATH)
|
|
||||||
|
|
||||||
if ssh:
|
|
||||||
repos = [[p["name"], p["ssh_url"]] for p in repos_json]
|
|
||||||
else:
|
|
||||||
repos = [[p["name"], p["clone_url"]] for p in repos_json]
|
|
||||||
|
|
||||||
for name, url in repos:
|
|
||||||
if name in REPOS_TO_SKIP:
|
|
||||||
continue
|
|
||||||
|
|
||||||
if not exists(f"{CLONES_PATH}/{name}"):
|
|
||||||
log.info(f"Retrieving {url}")
|
|
||||||
_run_cmd(f"git clone {url} {CLONES_PATH}/{name}")
|
|
||||||
|
|
||||||
chdir(f"{CLONES_PATH}/{name}")
|
|
||||||
if not int(_run_cmd("git branch --list | wc -l", shell=True)):
|
|
||||||
log.info(f"Guessing main branch is HEAD for {name}")
|
|
||||||
_run_cmd("git checkout main")
|
|
||||||
else:
|
|
||||||
log.info(f"Updating {name}")
|
|
||||||
chdir(f"{CLONES_PATH}/{name}")
|
|
||||||
_run_cmd("git fetch -a")
|
|
||||||
@@ -1,103 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# shellcheck disable=SC2119
|
|
||||||
|
|
||||||
# Usage: ./app-catalogue.sh
|
|
||||||
#
|
|
||||||
# Gather metadata from Co-op Cloud apps in $ABRA_DIR/apps (default
|
|
||||||
# ~/.abra/apps), and format it as a Markdown table for this page:
|
|
||||||
# https://docs.cloud.autonomic.zone/apps/
|
|
||||||
|
|
||||||
stack_dir="${ABRA_DIR:-$HOME/.abra}/apps/"
|
|
||||||
|
|
||||||
cd "$stack_dir" || exit
|
|
||||||
|
|
||||||
# load all README files into ENV_FILES array
|
|
||||||
mapfile -t readmes < <(find -L . -name "README.md")
|
|
||||||
# FIXME 3wc: requires bash 4, use for loop instead
|
|
||||||
|
|
||||||
base_url="https://git.autonomic.zone/coop-cloud"
|
|
||||||
|
|
||||||
cat_apps=()
|
|
||||||
cat_development=()
|
|
||||||
cat_utilities=()
|
|
||||||
cat_graveyard=()
|
|
||||||
|
|
||||||
get_var() {
|
|
||||||
echo "$1" | grep "$2" | sed 's/^[^:]*: //'
|
|
||||||
}
|
|
||||||
|
|
||||||
# shellcheck disable=SC2120
|
|
||||||
trim() {
|
|
||||||
# accept input as argument or from STDIN, see here:
|
|
||||||
# https://zwbetz.com/passing-input-to-a-bash-function-via-arguments-or-stdin/
|
|
||||||
# shellcheck disable=SC2155
|
|
||||||
local input="$([[ -p /dev/stdin ]] && cat - || echo "$@")"
|
|
||||||
[[ -z "$input" ]] && return 1
|
|
||||||
echo "$input" | tr -d ' '
|
|
||||||
}
|
|
||||||
|
|
||||||
# shellcheck disable=SC2120
|
|
||||||
prettify() {
|
|
||||||
# as above
|
|
||||||
# shellcheck disable=SC2155
|
|
||||||
local input="$([[ -p /dev/stdin ]] && cat - || echo "$@")"
|
|
||||||
[[ -z "$input" ]] && return 1
|
|
||||||
|
|
||||||
echo "$input" | sed -e 's/Yes/✅/' -e 's/No/❌/' -e 's/N\/A/⛔/'
|
|
||||||
}
|
|
||||||
|
|
||||||
for readme in "${readmes[@]}"; do
|
|
||||||
type="$(basename "${readme%README.md}")"
|
|
||||||
if [ "$type" = "example" ]; then
|
|
||||||
continue
|
|
||||||
fi
|
|
||||||
title="$(grep '^# ' "$type/README.md" | sed 's/^# //' )"
|
|
||||||
# find section between 'metadata' and 'endmetadata' comments
|
|
||||||
metadata="$(awk '/-- metadata --/,/-- endmetadata --/' "$type/README.md")"
|
|
||||||
status="$(get_var "$metadata" "Status")"
|
|
||||||
category="$(get_var "$metadata" "Category" | cut -d',' -f2 | trim)"
|
|
||||||
|
|
||||||
if [ -z "$category" ]; then
|
|
||||||
echo "ERROR: missing category for $type"
|
|
||||||
continue
|
|
||||||
fi
|
|
||||||
|
|
||||||
image="$(get_var "$metadata" "Image" | cut -d',' -f2 | trim)"
|
|
||||||
healthcheck="$(get_var "$metadata" "Healthcheck" | prettify)"
|
|
||||||
backups="$(get_var "$metadata" "Backups" | prettify)"
|
|
||||||
email="$(get_var "$metadata" "Email" | prettify)"
|
|
||||||
tests="$(get_var "$metadata" "Tests" | prettify)"
|
|
||||||
sso="$(get_var "$metadata" "SSO" | prettify)"
|
|
||||||
|
|
||||||
row="| [$title]($base_url/$type) | $status | $image | $healthcheck | $backups | $email | $tests | $sso |"
|
|
||||||
|
|
||||||
category_lower="$(echo "$category" | tr '[:upper:]' '[:lower:]')"
|
|
||||||
eval "cat_$category_lower+=( '$row' )"
|
|
||||||
done
|
|
||||||
|
|
||||||
headers="
|
|
||||||
| **Name** | **Status** | **Image** | **Healtcheck** | **Backups** | **Email** | **CI** | **Single-Sign-On** |
|
|
||||||
| --- | --- | --- | --- | --- | --- | --- | --- |"
|
|
||||||
|
|
||||||
echo "## Applications"
|
|
||||||
echo "$headers"
|
|
||||||
printf '%s\n' "${cat_apps[@]}" | sort
|
|
||||||
|
|
||||||
echo
|
|
||||||
|
|
||||||
echo "## Developer tools"
|
|
||||||
echo "$headers"
|
|
||||||
printf '%s\n' "${cat_development[@]}" | sort
|
|
||||||
|
|
||||||
echo
|
|
||||||
|
|
||||||
echo "## Utilities"
|
|
||||||
echo "$headers"
|
|
||||||
printf '%s\n' "${cat_utilities[@]}" | sort
|
|
||||||
|
|
||||||
echo
|
|
||||||
|
|
||||||
echo "## Graveyard"
|
|
||||||
echo "$headers"
|
|
||||||
printf '%s\n' "${cat_graveyard[@]}" | sort
|
|
||||||
-237
@@ -1,237 +0,0 @@
|
|||||||
#!/usr/bin/env python3
|
|
||||||
|
|
||||||
# Usage: ./app-json.py
|
|
||||||
#
|
|
||||||
# Gather metadata from Co-op Cloud apps in $ABRA_DIR/apps (default
|
|
||||||
# ~/.abra/apps), and format it as JSON so that it can be hosted here:
|
|
||||||
# https://apps.coopcloud.tech
|
|
||||||
|
|
||||||
import argparse
|
|
||||||
from json import dump
|
|
||||||
from os import chdir, environ, getcwd, listdir
|
|
||||||
from os.path import basename
|
|
||||||
from pathlib import Path
|
|
||||||
from re import findall, search
|
|
||||||
from subprocess import DEVNULL
|
|
||||||
|
|
||||||
from requests import get
|
|
||||||
|
|
||||||
from abralib import (
|
|
||||||
CLONES_PATH,
|
|
||||||
JQ_PATH,
|
|
||||||
REPOS_TO_SKIP,
|
|
||||||
YQ_PATH,
|
|
||||||
_run_cmd,
|
|
||||||
clone_all_apps,
|
|
||||||
get_repos_json,
|
|
||||||
log,
|
|
||||||
)
|
|
||||||
|
|
||||||
parser = argparse.ArgumentParser(description="Generate a new apps.json")
|
|
||||||
parser.add_argument("--output", type=Path, default=f"{getcwd()}/apps.json")
|
|
||||||
|
|
||||||
|
|
||||||
def skopeo_login():
|
|
||||||
"""Log into the docker registry to avoid rate limits."""
|
|
||||||
user = environ.get("SKOPEO_USER")
|
|
||||||
password = environ.get("SKOPEO_PASSWORD")
|
|
||||||
registry = environ.get("SKOPEO_REGISTRY", "docker.io")
|
|
||||||
|
|
||||||
if not user or not password:
|
|
||||||
log.info("Failed to log in via Skopeo due to missing env vars")
|
|
||||||
return
|
|
||||||
|
|
||||||
login_cmd = f"skopeo login {registry} -u {user} -p {password}"
|
|
||||||
output = _run_cmd(login_cmd, shell=True)
|
|
||||||
log.info(f"Skopeo login attempt: {output}")
|
|
||||||
|
|
||||||
|
|
||||||
def get_published_apps_json():
|
|
||||||
"""Retrieve already published apps json."""
|
|
||||||
url = "https://apps.coopcloud.tech"
|
|
||||||
|
|
||||||
log.info(f"Retrieving {url}")
|
|
||||||
|
|
||||||
try:
|
|
||||||
return get(url, timeout=5).json()
|
|
||||||
except Exception as exception:
|
|
||||||
log.error(f"Failed to retrieve {url}, saw {str(exception)}")
|
|
||||||
return {}
|
|
||||||
|
|
||||||
|
|
||||||
def generate_apps_json(repos_json):
|
|
||||||
"""Generate the abra-apps.json application versions file."""
|
|
||||||
apps_json = {}
|
|
||||||
cached_apps_json = get_published_apps_json()
|
|
||||||
|
|
||||||
for app in listdir(CLONES_PATH):
|
|
||||||
if app in REPOS_TO_SKIP:
|
|
||||||
log.info(f"Skipping {app}")
|
|
||||||
continue
|
|
||||||
|
|
||||||
repo_details = next(filter(lambda x: x["name"] == app, repos_json), {})
|
|
||||||
|
|
||||||
app_path = f"{CLONES_PATH}/{app}"
|
|
||||||
chdir(app_path)
|
|
||||||
|
|
||||||
metadata = get_app_metadata(app_path)
|
|
||||||
|
|
||||||
name = metadata.pop("name", app)
|
|
||||||
|
|
||||||
log.info(f"Processing {app}")
|
|
||||||
apps_json[app] = {
|
|
||||||
"name": name,
|
|
||||||
"category": metadata.get("category", ""),
|
|
||||||
"repository": repo_details.get("clone_url", ""),
|
|
||||||
"default_branch": repo_details.get("default_branch", ""),
|
|
||||||
"description": repo_details.get("description", ""),
|
|
||||||
"website": repo_details.get("website", ""),
|
|
||||||
"features": metadata,
|
|
||||||
"versions": get_app_versions(app_path, cached_apps_json),
|
|
||||||
"icon": repo_details.get("avatar_url", ""),
|
|
||||||
}
|
|
||||||
|
|
||||||
return apps_json
|
|
||||||
|
|
||||||
|
|
||||||
def get_app_metadata(app_path):
|
|
||||||
"""Parse metadata from app repo README files."""
|
|
||||||
metadata = {}
|
|
||||||
|
|
||||||
chdir(app_path)
|
|
||||||
|
|
||||||
try:
|
|
||||||
with open(f"{app_path}/README.md", "r") as handle:
|
|
||||||
log.info(f"{app_path}/README.md")
|
|
||||||
contents = handle.read()
|
|
||||||
except Exception:
|
|
||||||
log.info(f"No {app_path}/README.md discovered, moving on")
|
|
||||||
return {}
|
|
||||||
|
|
||||||
try:
|
|
||||||
for match in findall(r"\*\*.*", contents):
|
|
||||||
title = search(r"(?<=\*\*).*(?=\*\*)", match).group().lower()
|
|
||||||
|
|
||||||
if title == "image":
|
|
||||||
value = {
|
|
||||||
"image": search(r"(?<=`).*(?=`)", match).group(),
|
|
||||||
"url": search(r"(?<=\().*(?=\))", match).group(),
|
|
||||||
"rating": match.split(",")[1].strip(),
|
|
||||||
"source": match.split(",")[-1].replace("*", "").strip(),
|
|
||||||
}
|
|
||||||
elif title == "status":
|
|
||||||
value = {"❶💚": 1, "❷💛": 2, "❸🍎": 3, "❹💣": 4, "?": 5, "": 5}[
|
|
||||||
match.split(":")[-1].replace("*", "").strip()
|
|
||||||
]
|
|
||||||
else:
|
|
||||||
value = match.split(":")[-1].replace("*", "").strip()
|
|
||||||
|
|
||||||
metadata[title] = value
|
|
||||||
metadata["name"] = findall(r"^# (.*)", contents)[0]
|
|
||||||
except (IndexError, AttributeError):
|
|
||||||
log.info(f"Can't parse {app_path}/README.md")
|
|
||||||
return {}
|
|
||||||
finally:
|
|
||||||
_run_cmd("git checkout HEAD")
|
|
||||||
|
|
||||||
log.info(f"Parsed {metadata}")
|
|
||||||
|
|
||||||
return metadata
|
|
||||||
|
|
||||||
|
|
||||||
def get_app_versions(app_path, cached_apps_json):
|
|
||||||
versions = {}
|
|
||||||
|
|
||||||
chdir(app_path)
|
|
||||||
|
|
||||||
tags = _run_cmd("git tag --list").split()
|
|
||||||
|
|
||||||
if not tags:
|
|
||||||
log.info("No tags discovered, moving on")
|
|
||||||
return {}
|
|
||||||
|
|
||||||
initial_branch = _run_cmd("git rev-parse --abbrev-ref HEAD")
|
|
||||||
|
|
||||||
app_name = basename(app_path)
|
|
||||||
|
|
||||||
try:
|
|
||||||
existing_tags = cached_apps_json[app_name]["versions"].keys()
|
|
||||||
except KeyError:
|
|
||||||
existing_tags = []
|
|
||||||
|
|
||||||
for tag in tags:
|
|
||||||
_run_cmd(f"git checkout {tag}", stderr=DEVNULL)
|
|
||||||
|
|
||||||
services_cmd = f"{YQ_PATH} e '.services | keys | .[]' compose*.yml"
|
|
||||||
services = _run_cmd(services_cmd, shell=True).split()
|
|
||||||
|
|
||||||
parsed_services = []
|
|
||||||
service_versions = {}
|
|
||||||
for service in services:
|
|
||||||
if service in ("null", "---"):
|
|
||||||
continue
|
|
||||||
|
|
||||||
if (
|
|
||||||
tag in existing_tags
|
|
||||||
and service in cached_apps_json[app_name]["versions"][tag]
|
|
||||||
):
|
|
||||||
log.info(f"Skipping {tag} because we've already processed it")
|
|
||||||
existing_versions = cached_apps_json[app_name]["versions"][tag][service]
|
|
||||||
service_versions[service] = existing_versions
|
|
||||||
_run_cmd(f"git checkout {initial_branch}")
|
|
||||||
continue
|
|
||||||
|
|
||||||
if service in parsed_services:
|
|
||||||
log.info(f"Skipped {service}, we've already parsed it locally")
|
|
||||||
continue
|
|
||||||
|
|
||||||
services_cmd = f"{YQ_PATH} e '.services.{service}.image' compose*.yml"
|
|
||||||
images = _run_cmd(services_cmd, shell=True).split()
|
|
||||||
|
|
||||||
for image in images:
|
|
||||||
if image in ("null", "---"):
|
|
||||||
continue
|
|
||||||
|
|
||||||
images_cmd = f"skopeo inspect docker://{image} | {JQ_PATH} '.Digest'"
|
|
||||||
output = _run_cmd(images_cmd, shell=True)
|
|
||||||
|
|
||||||
service_version_info = {
|
|
||||||
"image": image.split(":")[0],
|
|
||||||
"tag": image.split(":")[-1],
|
|
||||||
"digest": output.split(":")[-1][:8],
|
|
||||||
}
|
|
||||||
|
|
||||||
log.info(f"Parsed {service_version_info}")
|
|
||||||
service_versions[service] = service_version_info
|
|
||||||
|
|
||||||
parsed_services.append(service)
|
|
||||||
|
|
||||||
versions[tag] = service_versions
|
|
||||||
|
|
||||||
_run_cmd(f"git checkout {initial_branch}")
|
|
||||||
|
|
||||||
return versions
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
"""Run the script."""
|
|
||||||
args = parser.parse_args()
|
|
||||||
|
|
||||||
skopeo_login()
|
|
||||||
|
|
||||||
repos_json = get_repos_json()
|
|
||||||
clone_all_apps(repos_json)
|
|
||||||
|
|
||||||
with open(args.output, "w", encoding="utf-8") as handle:
|
|
||||||
dump(
|
|
||||||
generate_apps_json(repos_json),
|
|
||||||
handle,
|
|
||||||
ensure_ascii=False,
|
|
||||||
indent=4,
|
|
||||||
sort_keys=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
log.info(f"Successfully generated {args.output}")
|
|
||||||
|
|
||||||
|
|
||||||
main()
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
#!/usr/bin/env python3
|
|
||||||
|
|
||||||
# Usage: ./clone-all-apps.py
|
|
||||||
#
|
|
||||||
# Clone all available apps into ~/.abra/apps using ssh:// URLs
|
|
||||||
|
|
||||||
from abralib import clone_all_apps, get_repos_json
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
"""Run the script."""
|
|
||||||
repos_json = get_repos_json()
|
|
||||||
clone_all_apps(repos_json, ssh=True)
|
|
||||||
|
|
||||||
|
|
||||||
main()
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
#!/usr/bin/env python3
|
|
||||||
|
|
||||||
# Usage: ./github-sync.py
|
|
||||||
#
|
|
||||||
# Mirror repositories to Github (Fuck M$, get it straight)
|
|
||||||
|
|
||||||
from os import chdir, environ, listdir
|
|
||||||
|
|
||||||
from abralib import CLONES_PATH, _run_cmd, clone_all_apps, get_repos_json, log
|
|
||||||
|
|
||||||
REPOS_TO_SKIP = (
|
|
||||||
"apps",
|
|
||||||
"backup-bot",
|
|
||||||
"docker-dind-bats-kcov",
|
|
||||||
"docs.coopcloud.tech",
|
|
||||||
"pyabra",
|
|
||||||
"radicle-seed-node",
|
|
||||||
"swarm-cronjob",
|
|
||||||
"tyop",
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
"""Run the script."""
|
|
||||||
repos_json = get_repos_json()
|
|
||||||
clone_all_apps(repos_json)
|
|
||||||
|
|
||||||
for app in listdir(CLONES_PATH):
|
|
||||||
if app in REPOS_TO_SKIP:
|
|
||||||
log.info(f"Skipping {app}")
|
|
||||||
continue
|
|
||||||
|
|
||||||
app_path = f"{CLONES_PATH}/{app}"
|
|
||||||
chdir(app_path)
|
|
||||||
|
|
||||||
log.info(f"Mirroring {app}...")
|
|
||||||
|
|
||||||
token = environ.get("GITHUB_ACCESS_TOKEN")
|
|
||||||
remote = f"https://coopcloudbot:{token}@github.com/Coop-Cloud/{app}.git"
|
|
||||||
|
|
||||||
_run_cmd(
|
|
||||||
f"git remote add github {remote} || true",
|
|
||||||
shell=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
_run_cmd("git push github --all")
|
|
||||||
|
|
||||||
|
|
||||||
main()
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
#!/usr/bin/env python3
|
|
||||||
|
|
||||||
# Usage: ./renovate-ls-apps.py
|
|
||||||
#
|
|
||||||
# Output list of apps for Renovate bot configuration
|
|
||||||
|
|
||||||
from abralib import REPOS_TO_SKIP, get_repos_json
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
"""Run the script."""
|
|
||||||
repos = [p["full_name"] for p in get_repos_json()]
|
|
||||||
repos.sort()
|
|
||||||
for repo in repos:
|
|
||||||
if repo.split("/")[-1] in REPOS_TO_SKIP:
|
|
||||||
continue
|
|
||||||
print(f'"{repo}",')
|
|
||||||
|
|
||||||
|
|
||||||
main()
|
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appAliases = i18n.GC("a", "abra app")
|
||||||
|
|
||||||
|
var AppCommand = &cobra.Command{
|
||||||
|
// translators: `app` command group
|
||||||
|
Use: i18n.G("app [cmd] [args] [flags]"),
|
||||||
|
Aliases: strings.Split(appAliases, ","),
|
||||||
|
// translators: Short description for `app` command group
|
||||||
|
Short: i18n.G("Manage apps"),
|
||||||
|
}
|
||||||
@@ -0,0 +1,339 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app backup list` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appBackupListAliases = i18n.G("ls")
|
||||||
|
|
||||||
|
var AppBackupListCommand = &cobra.Command{
|
||||||
|
// translators: `app backup list` command
|
||||||
|
Use: i18n.G("list <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appBackupListAliases, ","),
|
||||||
|
// translators: Short description for `app backup list` command
|
||||||
|
Short: i18n.G("List the contents of a snapshot"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
targetContainer, err := internal.RetrieveBackupBotContainer(cl)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
execEnv := []string{
|
||||||
|
fmt.Sprintf("SERVICE=%s", app.Domain),
|
||||||
|
"MACHINE_LOGS=true",
|
||||||
|
}
|
||||||
|
|
||||||
|
if snapshot != "" {
|
||||||
|
log.Debug(i18n.G("including SNAPSHOT=%s in backupbot exec invocation", snapshot))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("SNAPSHOT=%s", snapshot))
|
||||||
|
}
|
||||||
|
|
||||||
|
if showAllPaths {
|
||||||
|
log.Debug(i18n.G("including SHOW_ALL=%v in backupbot exec invocation", showAllPaths))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("SHOW_ALL=%v", showAllPaths))
|
||||||
|
}
|
||||||
|
|
||||||
|
if timestamps {
|
||||||
|
log.Debug(i18n.G("including TIMESTAMPS=%v in backupbot exec invocation", timestamps))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("TIMESTAMPS=%v", timestamps))
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err = internal.RunBackupCmdRemote(cl, "ls", targetContainer.ID, execEnv); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app backup download` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appBackupDownloadAliases = i18n.G("d")
|
||||||
|
|
||||||
|
var AppBackupDownloadCommand = &cobra.Command{
|
||||||
|
// translators: `app backup download` command
|
||||||
|
Use: i18n.G("download <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appBackupDownloadAliases, ","),
|
||||||
|
// translators: Short description for `app backup download` command
|
||||||
|
Short: i18n.G("Download a snapshot"),
|
||||||
|
Long: i18n.G(`Downloads a backup.tar.gz to the current working directory.
|
||||||
|
|
||||||
|
"--volumes/-v" includes data contained in volumes alongide paths specified in
|
||||||
|
"backupbot.backup.path" labels.`),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
targetContainer, err := internal.RetrieveBackupBotContainer(cl)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
execEnv := []string{
|
||||||
|
fmt.Sprintf("SERVICE=%s", app.Domain),
|
||||||
|
"MACHINE_LOGS=true",
|
||||||
|
}
|
||||||
|
|
||||||
|
if snapshot != "" {
|
||||||
|
log.Debug(i18n.G("including SNAPSHOT=%s in backupbot exec invocation", snapshot))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("SNAPSHOT=%s", snapshot))
|
||||||
|
}
|
||||||
|
|
||||||
|
if includePath != "" {
|
||||||
|
log.Debug(i18n.G("including INCLUDE_PATH=%s in backupbot exec invocation", includePath))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("INCLUDE_PATH=%s", includePath))
|
||||||
|
}
|
||||||
|
|
||||||
|
if includeSecrets {
|
||||||
|
log.Debug(i18n.G("including SECRETS=%v in backupbot exec invocation", includeSecrets))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("SECRETS=%v", includeSecrets))
|
||||||
|
}
|
||||||
|
|
||||||
|
if includeVolumes {
|
||||||
|
log.Debug(i18n.G("including VOLUMES=%v in backupbot exec invocation", includeVolumes))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("VOLUMES=%v", includeVolumes))
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := internal.RunBackupCmdRemote(cl, "download", targetContainer.ID, execEnv); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
remoteBackupDir := "/tmp/backup.tar.gz"
|
||||||
|
currentWorkingDir := "."
|
||||||
|
if err = CopyFromContainer(cl, targetContainer.ID, remoteBackupDir, currentWorkingDir); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app backup create` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appBackupCreateAliases = i18n.G("c")
|
||||||
|
|
||||||
|
var AppBackupCreateCommand = &cobra.Command{
|
||||||
|
// translators: `app backup create` command
|
||||||
|
Use: i18n.G("create <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appBackupCreateAliases, ","),
|
||||||
|
// translators: Short description for `app backup create` command
|
||||||
|
Short: i18n.G("Create a new snapshot"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
targetContainer, err := internal.RetrieveBackupBotContainer(cl)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
execEnv := []string{
|
||||||
|
fmt.Sprintf("SERVICE=%s", app.Domain),
|
||||||
|
"MACHINE_LOGS=true",
|
||||||
|
}
|
||||||
|
|
||||||
|
if retries != "" {
|
||||||
|
log.Debug(i18n.G("including RETRIES=%s in backupbot exec invocation", retries))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("RETRIES=%s", retries))
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := internal.RunBackupCmdRemote(cl, "create", targetContainer.ID, execEnv); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app backup snapshots` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appBackupSnapshotsAliases = i18n.G("s")
|
||||||
|
|
||||||
|
var AppBackupSnapshotsCommand = &cobra.Command{
|
||||||
|
// translators: `app backup snapshots` command
|
||||||
|
Use: i18n.G("snapshots <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appBackupSnapshotsAliases, ","),
|
||||||
|
// translators: Short description for `app backup snapshots` command
|
||||||
|
Short: i18n.G("List all snapshots"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
targetContainer, err := internal.RetrieveBackupBotContainer(cl)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
execEnv := []string{
|
||||||
|
fmt.Sprintf("SERVICE=%s", app.Domain),
|
||||||
|
"MACHINE_LOGS=true",
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err = internal.RunBackupCmdRemote(cl, "snapshots", targetContainer.ID, execEnv); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app backup` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appBackupAliases = i18n.G("b")
|
||||||
|
|
||||||
|
var AppBackupCommand = &cobra.Command{
|
||||||
|
// translators: `app backup` command group
|
||||||
|
Use: i18n.G("backup [cmd] [args] [flags]"),
|
||||||
|
Aliases: strings.Split(appBackupAliases, ","),
|
||||||
|
// translators: Short description for `app backup` command group
|
||||||
|
Short: i18n.G("Manage app backups"),
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
snapshot string
|
||||||
|
retries string
|
||||||
|
includePath string
|
||||||
|
showAllPaths bool
|
||||||
|
timestamps bool
|
||||||
|
includeSecrets bool
|
||||||
|
includeVolumes bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppBackupListCommand.Flags().StringVarP(
|
||||||
|
&snapshot,
|
||||||
|
i18n.G("snapshot"),
|
||||||
|
i18n.G("s"),
|
||||||
|
"",
|
||||||
|
i18n.G("list specific snapshot"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupListCommand.Flags().BoolVarP(
|
||||||
|
&showAllPaths,
|
||||||
|
i18n.G("all"),
|
||||||
|
i18n.GC("a", "app backup list"),
|
||||||
|
false,
|
||||||
|
i18n.G("show all paths"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupListCommand.Flags().BoolVarP(
|
||||||
|
×tamps,
|
||||||
|
i18n.G("timestamps"),
|
||||||
|
i18n.G("t"),
|
||||||
|
false,
|
||||||
|
i18n.G("include timestamps"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupDownloadCommand.Flags().StringVarP(
|
||||||
|
&snapshot,
|
||||||
|
i18n.G("snapshot"),
|
||||||
|
i18n.G("s"),
|
||||||
|
"",
|
||||||
|
i18n.G("list specific snapshot"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupDownloadCommand.Flags().StringVarP(
|
||||||
|
&includePath,
|
||||||
|
i18n.G("path"),
|
||||||
|
i18n.G("p"),
|
||||||
|
"",
|
||||||
|
i18n.G("volumes path"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupDownloadCommand.Flags().BoolVarP(
|
||||||
|
&includeSecrets,
|
||||||
|
i18n.G("secrets"),
|
||||||
|
i18n.G("S"),
|
||||||
|
false,
|
||||||
|
i18n.G("include secrets"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupDownloadCommand.Flags().BoolVarP(
|
||||||
|
&includeVolumes,
|
||||||
|
i18n.G("volumes"),
|
||||||
|
i18n.G("v"),
|
||||||
|
false,
|
||||||
|
i18n.G("include volumes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupDownloadCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupCreateCommand.Flags().StringVarP(
|
||||||
|
&retries,
|
||||||
|
i18n.G("retries"),
|
||||||
|
i18n.G("r"),
|
||||||
|
"1",
|
||||||
|
i18n.G("number of retry attempts"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppBackupCreateCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,99 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/charmbracelet/lipgloss"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app check` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appCheckAliases = i18n.G("chk")
|
||||||
|
|
||||||
|
var AppCheckCommand = &cobra.Command{
|
||||||
|
// translators: `app check` command
|
||||||
|
Use: i18n.G("check <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appCheckAliases, ","),
|
||||||
|
// translators: Short description for `app check` command
|
||||||
|
Short: i18n.G("Ensure an app is well configured"),
|
||||||
|
Long: i18n.G(`Compare env vars in both the app ".env" and recipe ".env.sample" file.
|
||||||
|
|
||||||
|
The goal is to ensure that recipe ".env.sample" env vars are defined in your
|
||||||
|
app ".env" file. Only env var definitions in the ".env.sample" which are
|
||||||
|
uncommented, e.g. "FOO=bar" are checked. If an app ".env" file does not include
|
||||||
|
these env vars, then "check" will complain.
|
||||||
|
|
||||||
|
Recipe maintainers may or may not provide defaults for env vars within their
|
||||||
|
recipes regardless of commenting or not (e.g. through the use of
|
||||||
|
${FOO:<default>} syntax). "check" does not confirm or deny this for you.`),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.
|
||||||
|
Headers(
|
||||||
|
fmt.Sprintf("%s .env.sample", app.Recipe.Name),
|
||||||
|
fmt.Sprintf("%s.env", app.Name),
|
||||||
|
).
|
||||||
|
StyleFunc(func(row, col int) lipgloss.Style {
|
||||||
|
switch {
|
||||||
|
case col == 1:
|
||||||
|
return lipgloss.NewStyle().Padding(0, 1, 0, 1).Align(lipgloss.Center)
|
||||||
|
default:
|
||||||
|
return lipgloss.NewStyle().Padding(0, 1, 0, 1)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
envVars, err := appPkg.CheckEnv(app)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, envVar := range envVars {
|
||||||
|
if envVar.Present {
|
||||||
|
val := []string{envVar.Name, "✅"}
|
||||||
|
table.Row(val...)
|
||||||
|
} else {
|
||||||
|
val := []string{envVar.Name, "❌"}
|
||||||
|
table.Row(val...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppCheckCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
+289
@@ -0,0 +1,289 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"slices"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app cmd` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appCmdAliases = i18n.G("cmd")
|
||||||
|
|
||||||
|
var AppCmdCommand = &cobra.Command{
|
||||||
|
// translators: `app command` command
|
||||||
|
Use: i18n.G("command <domain> [service | --local] <cmd> [[args] [flags] | [flags] -- [args]]"),
|
||||||
|
Aliases: strings.Split(appCmdAliases, ","),
|
||||||
|
// translators: Short description for `app cmd` command
|
||||||
|
Short: i18n.G("Run app commands"),
|
||||||
|
Long: i18n.G(`Run an app specific command.
|
||||||
|
|
||||||
|
These commands are bash functions, defined in the abra.sh of the recipe itself.
|
||||||
|
They can be run within the context of a service (e.g. app) or locally on your
|
||||||
|
work station by passing "--local/-l".
|
||||||
|
|
||||||
|
N.B. If using the "--" style to pass arguments, flags (e.g. "--local/-l") must
|
||||||
|
be passed *before* the "--". It is possible to pass arguments without the "--"
|
||||||
|
as long as no dashes are present (i.e. "foo" works without "--", "-foo"
|
||||||
|
does not).`),
|
||||||
|
Example: i18n.G(` # pass <cmd> args/flags without "--"
|
||||||
|
abra app cmd 1312.net app my_cmd_arg foo --user bar
|
||||||
|
|
||||||
|
# pass <cmd> args/flags with "--"
|
||||||
|
abra app cmd 1312.net app my_cmd_args --user bar -- foo -vvv
|
||||||
|
|
||||||
|
# drop the [service] arg if using "--local/-l"
|
||||||
|
abra app cmd 1312.net my_cmd --local`),
|
||||||
|
Args: func(cmd *cobra.Command, args []string) error {
|
||||||
|
if local {
|
||||||
|
if !(len(args) >= 2) {
|
||||||
|
return errors.New(i18n.G("requires at least 2 arguments with --local/-l"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if slices.Contains(os.Args, "--") {
|
||||||
|
if cmd.ArgsLenAtDash() > 2 {
|
||||||
|
return errors.New(i18n.G("accepts at most 2 args with --local/-l"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE(d1): it is unclear how to correctly validate this case
|
||||||
|
//
|
||||||
|
// abra app cmd 1312.net app test_cmd_args foo --local
|
||||||
|
// FATAL <recipe> doesn't have a app function
|
||||||
|
//
|
||||||
|
// "app" should not be there, but there is no reliable way to detect arg
|
||||||
|
// count when the user can pass an arbitrary amount of recipe command
|
||||||
|
// arguments
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !(len(args) >= 3) {
|
||||||
|
return errors.New(i18n.G("requires at least 3 arguments"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
if !local {
|
||||||
|
return autocomplete.ServiceNameComplete(args[0])
|
||||||
|
}
|
||||||
|
return autocomplete.CommandNameComplete(args[0])
|
||||||
|
case 2:
|
||||||
|
if !local {
|
||||||
|
return autocomplete.CommandNameComplete(args[0])
|
||||||
|
}
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if local && remoteUser != "" {
|
||||||
|
log.Fatal(i18n.G("cannot use --local & --user together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
hasCmdArgs, parsedCmdArgs := parseCmdArgs(args, local)
|
||||||
|
|
||||||
|
if _, err := os.Stat(app.Recipe.AbraShPath); err != nil {
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
log.Fatal(i18n.G("%s does not exist for %s?", app.Recipe.AbraShPath, app.Name))
|
||||||
|
}
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if local {
|
||||||
|
cmdName := args[1]
|
||||||
|
if err := internal.EnsureCommand(app.Recipe.AbraShPath, app.Recipe.Name, cmdName); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("--local detected, running %s on local work station", cmdName))
|
||||||
|
|
||||||
|
var exportEnv string
|
||||||
|
for k, v := range app.Env {
|
||||||
|
exportEnv = exportEnv + fmt.Sprintf("%s='%s'; ", k, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
var sourceAndExec string
|
||||||
|
if hasCmdArgs {
|
||||||
|
log.Debug(i18n.G("parsed following command arguments: %s", parsedCmdArgs))
|
||||||
|
sourceAndExec = fmt.Sprintf("TARGET=local; APP_NAME=%s; STACK_NAME=%s; %s . %s; %s %s", app.Name, app.StackName(), exportEnv, app.Recipe.AbraShPath, cmdName, parsedCmdArgs)
|
||||||
|
} else {
|
||||||
|
log.Debug(i18n.G("did not detect any command arguments"))
|
||||||
|
sourceAndExec = fmt.Sprintf("TARGET=local; APP_NAME=%s; STACK_NAME=%s; %s . %s; %s", app.Name, app.StackName(), exportEnv, app.Recipe.AbraShPath, cmdName)
|
||||||
|
}
|
||||||
|
|
||||||
|
shell := "/bin/bash"
|
||||||
|
if _, err := os.Stat(shell); errors.Is(err, os.ErrNotExist) {
|
||||||
|
log.Debug(i18n.G("%s does not exist locally, use /bin/sh as fallback", shell))
|
||||||
|
shell = "/bin/sh"
|
||||||
|
}
|
||||||
|
cmd := exec.Command(shell, "-c", sourceAndExec)
|
||||||
|
|
||||||
|
if err := internal.RunCmd(cmd); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
cmdName := args[2]
|
||||||
|
if err := internal.EnsureCommand(app.Recipe.AbraShPath, app.Recipe.Name, cmdName); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
serviceNames, err := appPkg.GetAppServiceNames(app.Name)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
matchingServiceName := false
|
||||||
|
targetServiceName := args[1]
|
||||||
|
for _, serviceName := range serviceNames {
|
||||||
|
if serviceName == targetServiceName {
|
||||||
|
matchingServiceName = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !matchingServiceName {
|
||||||
|
log.Fatal(i18n.G("no service %s for %s?", targetServiceName, app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("running command %s within the context of %s_%s", cmdName, app.StackName(), targetServiceName))
|
||||||
|
|
||||||
|
if hasCmdArgs {
|
||||||
|
log.Debug(i18n.G("parsed following command arguments: %s", parsedCmdArgs))
|
||||||
|
} else {
|
||||||
|
log.Debug(i18n.G("did not detect any command arguments"))
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := internal.RunCmdRemote(
|
||||||
|
cl,
|
||||||
|
app,
|
||||||
|
disableTTY,
|
||||||
|
app.Recipe.AbraShPath,
|
||||||
|
targetServiceName, cmdName, parsedCmdArgs, remoteUser); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app command list` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var appCmdListAliases = i18n.G("ls")
|
||||||
|
|
||||||
|
var AppCmdListCommand = &cobra.Command{
|
||||||
|
// translators: `app cmd list` command
|
||||||
|
Use: i18n.G("list <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appCmdListAliases, ","),
|
||||||
|
// translators: Short description for `app cmd list` command
|
||||||
|
Short: i18n.G("List all available commands"),
|
||||||
|
Args: cobra.MinimumNArgs(1),
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cmdNames, err := appPkg.ReadAbraShCmdNames(app.Recipe.AbraShPath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Strings(cmdNames)
|
||||||
|
|
||||||
|
for _, cmdName := range cmdNames {
|
||||||
|
fmt.Println(cmdName)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseCmdArgs(args []string, isLocal bool) (bool, string) {
|
||||||
|
var (
|
||||||
|
parsedCmdArgs string
|
||||||
|
hasCmdArgs bool
|
||||||
|
)
|
||||||
|
|
||||||
|
if isLocal {
|
||||||
|
if len(args) > 2 {
|
||||||
|
return true, fmt.Sprintf("%s ", strings.Join(args[2:], " "))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if len(args) > 3 {
|
||||||
|
return true, fmt.Sprintf("%s ", strings.Join(args[3:], " "))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return hasCmdArgs, parsedCmdArgs
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
local bool
|
||||||
|
remoteUser string
|
||||||
|
disableTTY bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppCmdCommand.Flags().BoolVarP(
|
||||||
|
&local,
|
||||||
|
i18n.G("local"),
|
||||||
|
i18n.G("l"),
|
||||||
|
false,
|
||||||
|
i18n.G("run command locally"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppCmdCommand.Flags().StringVarP(
|
||||||
|
&remoteUser,
|
||||||
|
i18n.G("user"),
|
||||||
|
i18n.G("u"),
|
||||||
|
"",
|
||||||
|
i18n.G("request remote user"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppCmdCommand.Flags().BoolVarP(
|
||||||
|
&disableTTY,
|
||||||
|
i18n.G("tty"),
|
||||||
|
i18n.G("T"),
|
||||||
|
false,
|
||||||
|
i18n.G("disable remote TTY"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppCmdCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestParseCmdArgs(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
input []string
|
||||||
|
shouldParse bool
|
||||||
|
expectedOutput string
|
||||||
|
}{
|
||||||
|
// `--` is not parsed when passed in from the command-line e.g. -- foo bar baz
|
||||||
|
// so we need to eumlate that as missing when testing if bash args are passed in
|
||||||
|
// see https://git.coopcloud.tech/toolshed/organising/issues/336 for more
|
||||||
|
{[]string{"foo.com", "app", "test"}, false, ""},
|
||||||
|
{[]string{"foo.com", "app", "test", "foo"}, true, "foo "},
|
||||||
|
{[]string{"foo.com", "app", "test", "foo", "bar", "baz"}, true, "foo bar baz "},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, test := range tests {
|
||||||
|
ok, parsed := parseCmdArgs(test.input, false)
|
||||||
|
if ok != test.shouldParse {
|
||||||
|
t.Fatalf("[%s] should not parse", strings.Join(test.input, " "))
|
||||||
|
}
|
||||||
|
if parsed != test.expectedOutput {
|
||||||
|
t.Fatalf("%s does not match %s", parsed, test.expectedOutput)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,65 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app config` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var appConfigAliases = i18n.G("cfg")
|
||||||
|
|
||||||
|
var AppConfigCommand = &cobra.Command{
|
||||||
|
// translators: `app config` command
|
||||||
|
Use: i18n.G("config <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appConfigAliases, ","),
|
||||||
|
// translators: Short description for `app config` command
|
||||||
|
Short: i18n.G("Edit app config"),
|
||||||
|
Example: i18n.G(" abra config 1312.net"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
files, err := appPkg.LoadAppFiles("")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
appName := args[0]
|
||||||
|
appFile, exists := files[appName]
|
||||||
|
if !exists {
|
||||||
|
log.Fatal(i18n.G("cannot find app with name %s", appName))
|
||||||
|
}
|
||||||
|
|
||||||
|
ed, ok := os.LookupEnv("EDITOR")
|
||||||
|
if !ok {
|
||||||
|
edPrompt := &survey.Select{
|
||||||
|
Message: i18n.G("which editor do you wish to use?"),
|
||||||
|
Options: []string{"vi", "vim", "nvim", "nano", "pico", "emacs"},
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(edPrompt, &ed); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
c := exec.Command(ed, appFile.Path)
|
||||||
|
c.Stdin = os.Stdin
|
||||||
|
c.Stdout = os.Stdout
|
||||||
|
c.Stderr = os.Stderr
|
||||||
|
if err := c.Run(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
+389
@@ -0,0 +1,389 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
containerPkg "coopcloud.tech/abra/pkg/container"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/container"
|
||||||
|
"github.com/docker/cli/cli/command"
|
||||||
|
containertypes "github.com/docker/docker/api/types/container"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/docker/docker/errdefs"
|
||||||
|
"github.com/docker/docker/pkg/archive"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app cp` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appCpAliases = i18n.G("c")
|
||||||
|
|
||||||
|
var AppCpCommand = &cobra.Command{
|
||||||
|
// translators: `app cp` command
|
||||||
|
Use: i18n.G("cp <domain> <src> <dst> [flags]"),
|
||||||
|
Aliases: strings.Split(appCpAliases, ","),
|
||||||
|
// translators: Short description for `app cp` command
|
||||||
|
Short: i18n.G("Copy files to/from a deployed app service"),
|
||||||
|
Example: i18n.G(` # copy myfile.txt to the root of the app service
|
||||||
|
abra app cp 1312.net myfile.txt app:/
|
||||||
|
|
||||||
|
# copy that file back to your current working directory locally
|
||||||
|
abra app cp 1312.net app:/myfile.txt ./`),
|
||||||
|
Args: cobra.ExactArgs(3),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
src := args[1]
|
||||||
|
dst := args[2]
|
||||||
|
srcPath, dstPath, service, toContainer, err := parseSrcAndDst(src, dst)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
container, err := containerPkg.GetContainerFromStackAndService(cl, app.StackName(), service)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
log.Debug(i18n.G("retrieved %s as target container on %s", formatter.ShortenID(container.ID), app.Server))
|
||||||
|
|
||||||
|
if toContainer {
|
||||||
|
err = CopyToContainer(cl, container.ID, srcPath, dstPath)
|
||||||
|
} else {
|
||||||
|
err = CopyFromContainer(cl, container.ID, srcPath, dstPath)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var errServiceMissing = errors.New(i18n.G("one of <src>/<dest> arguments must take $SERVICE:$PATH form"))
|
||||||
|
|
||||||
|
// parseSrcAndDst parses src and dest string. One of src or dst must be of the form $SERVICE:$PATH
|
||||||
|
func parseSrcAndDst(src, dst string) (srcPath string, dstPath string, service string, toContainer bool, err error) {
|
||||||
|
parsedSrc := strings.SplitN(src, ":", 2)
|
||||||
|
parsedDst := strings.SplitN(dst, ":", 2)
|
||||||
|
if len(parsedSrc)+len(parsedDst) != 3 {
|
||||||
|
return "", "", "", false, errServiceMissing
|
||||||
|
}
|
||||||
|
if len(parsedSrc) == 2 {
|
||||||
|
return parsedSrc[1], dst, parsedSrc[0], false, nil
|
||||||
|
}
|
||||||
|
if len(parsedDst) == 2 {
|
||||||
|
return src, parsedDst[1], parsedDst[0], true, nil
|
||||||
|
}
|
||||||
|
return "", "", "", false, errServiceMissing
|
||||||
|
}
|
||||||
|
|
||||||
|
// CopyToContainer copies a file or directory from the local file system to the container.
|
||||||
|
// See the possible copy modes and their documentation.
|
||||||
|
func CopyToContainer(cl *dockerClient.Client, containerID, srcPath, dstPath string) error {
|
||||||
|
srcStat, err := os.Stat(srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("local %s ", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
dstStat, err := cl.ContainerStatPath(context.Background(), containerID, dstPath)
|
||||||
|
dstExists := true
|
||||||
|
if err != nil {
|
||||||
|
if errdefs.IsNotFound(err) {
|
||||||
|
dstExists = false
|
||||||
|
} else {
|
||||||
|
return errors.New(i18n.G("remote path: %s", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mode, err := copyMode(srcPath, dstPath, srcStat.Mode(), dstStat.Mode, dstExists)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
movePath := ""
|
||||||
|
switch mode {
|
||||||
|
case CopyModeDirToDir:
|
||||||
|
// Add the src directory to the destination path
|
||||||
|
_, srcDir := path.Split(srcPath)
|
||||||
|
dstPath = path.Join(dstPath, srcDir)
|
||||||
|
|
||||||
|
// Make sure the dst directory exits.
|
||||||
|
dcli, err := command.NewDockerCli()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if _, err := container.RunExec(dcli, cl, containerID, &containertypes.ExecOptions{
|
||||||
|
AttachStderr: true,
|
||||||
|
AttachStdin: true,
|
||||||
|
AttachStdout: true,
|
||||||
|
Cmd: []string{"mkdir", "-p", dstPath},
|
||||||
|
Detach: false,
|
||||||
|
Tty: true,
|
||||||
|
}); err != nil {
|
||||||
|
return errors.New(i18n.G("create remote directory: %s", err))
|
||||||
|
}
|
||||||
|
case CopyModeFileToFile:
|
||||||
|
// Remove the file component from the path, since docker can only copy
|
||||||
|
// to a directory.
|
||||||
|
dstPath, _ = path.Split(dstPath)
|
||||||
|
case CopyModeFileToFileRename:
|
||||||
|
// Copy the file to the temp directory and move it to its dstPath
|
||||||
|
// afterwards.
|
||||||
|
movePath = dstPath
|
||||||
|
dstPath = "/tmp"
|
||||||
|
}
|
||||||
|
|
||||||
|
toTarOpts := &archive.TarOptions{IncludeSourceDir: true, NoOverwriteDirNonDir: true, Compression: archive.Gzip}
|
||||||
|
content, err := archive.TarWithOptions(srcPath, toTarOpts)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("copy %s from local to %s on container", srcPath, dstPath))
|
||||||
|
copyOpts := containertypes.CopyToContainerOptions{AllowOverwriteDirWithFile: false, CopyUIDGID: false}
|
||||||
|
if err := cl.CopyToContainer(context.Background(), containerID, dstPath, content, copyOpts); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if movePath != "" {
|
||||||
|
_, srcFile := path.Split(srcPath)
|
||||||
|
dcli, err := command.NewDockerCli()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if _, err := container.RunExec(dcli, cl, containerID, &containertypes.ExecOptions{
|
||||||
|
AttachStderr: true,
|
||||||
|
AttachStdin: true,
|
||||||
|
AttachStdout: true,
|
||||||
|
Cmd: []string{"mv", path.Join("/tmp", srcFile), movePath},
|
||||||
|
Detach: false,
|
||||||
|
Tty: true,
|
||||||
|
}); err != nil {
|
||||||
|
return errors.New(i18n.G("create remote directory: %s", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CopyFromContainer copies a file or directory from the given container to the local file system.
|
||||||
|
// See the possible copy modes and their documentation.
|
||||||
|
func CopyFromContainer(cl *dockerClient.Client, containerID, srcPath, dstPath string) error {
|
||||||
|
srcStat, err := cl.ContainerStatPath(context.Background(), containerID, srcPath)
|
||||||
|
if err != nil {
|
||||||
|
if errdefs.IsNotFound(err) {
|
||||||
|
return errors.New(i18n.G("remote: %s does not exist", srcPath))
|
||||||
|
} else {
|
||||||
|
return errors.New(i18n.G("remote path: %s", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dstStat, err := os.Stat(dstPath)
|
||||||
|
dstExists := true
|
||||||
|
var dstMode os.FileMode
|
||||||
|
if err != nil {
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
dstExists = false
|
||||||
|
} else {
|
||||||
|
return errors.New(i18n.G("remote path: %s", err))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
dstMode = dstStat.Mode()
|
||||||
|
}
|
||||||
|
|
||||||
|
mode, err := copyMode(srcPath, dstPath, srcStat.Mode, dstMode, dstExists)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
moveDstDir := ""
|
||||||
|
moveDstFile := ""
|
||||||
|
switch mode {
|
||||||
|
case CopyModeFileToFile:
|
||||||
|
// Remove the file component from the path, since docker can only copy
|
||||||
|
// to a directory.
|
||||||
|
dstPath, _ = path.Split(dstPath)
|
||||||
|
case CopyModeFileToFileRename:
|
||||||
|
// Copy the file to the temp directory and move it to its dstPath
|
||||||
|
// afterwards.
|
||||||
|
moveDstFile = dstPath
|
||||||
|
dstPath = "/tmp"
|
||||||
|
case CopyModeFilesToDir:
|
||||||
|
// Copy the directory to the temp directory and move it to its
|
||||||
|
// dstPath afterwards.
|
||||||
|
moveDstDir = path.Join(dstPath, "/")
|
||||||
|
dstPath = "/tmp"
|
||||||
|
|
||||||
|
// Make sure the temp directory always gets removed
|
||||||
|
defer os.Remove(path.Join("/tmp"))
|
||||||
|
}
|
||||||
|
|
||||||
|
content, _, err := cl.CopyFromContainer(context.Background(), containerID, srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("copy: %s", err))
|
||||||
|
}
|
||||||
|
defer content.Close()
|
||||||
|
if err := archive.Untar(content, dstPath, &archive.TarOptions{
|
||||||
|
NoOverwriteDirNonDir: true,
|
||||||
|
Compression: archive.Gzip,
|
||||||
|
NoLchown: true,
|
||||||
|
}); err != nil {
|
||||||
|
return errors.New(i18n.G("untar: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if moveDstFile != "" {
|
||||||
|
_, srcFile := path.Split(strings.TrimSuffix(srcPath, "/"))
|
||||||
|
if err := moveFile(path.Join("/tmp", srcFile), moveDstFile); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if moveDstDir != "" {
|
||||||
|
_, srcDir := path.Split(strings.TrimSuffix(srcPath, "/"))
|
||||||
|
if err := moveDir(path.Join("/tmp", srcDir), moveDstDir); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrCopyDirToFile = errors.New(i18n.G("can't copy dir to file"))
|
||||||
|
ErrDstDirNotExist = errors.New(i18n.G("destination directory does not exist"))
|
||||||
|
)
|
||||||
|
|
||||||
|
type CopyMode int
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Copy a src file to a dest file. The src and dest file names are the same.
|
||||||
|
// <dir_src>/<file> + <dir_dst>/<file> -> <dir_dst>/<file>
|
||||||
|
CopyModeFileToFile = CopyMode(iota)
|
||||||
|
// Copy a src file to a dest file. The src and dest file names are not the same.
|
||||||
|
// <dir_src>/<file_src> + <dir_dst>/<file_dst> -> <dir_dst>/<file_dst>
|
||||||
|
CopyModeFileToFileRename
|
||||||
|
// Copy a src file to dest directory. The dest file gets created in the dest
|
||||||
|
// folder with the src filename.
|
||||||
|
// <dir_src>/<file> + <dir_dst> -> <dir_dst>/<file>
|
||||||
|
CopyModeFileToDir
|
||||||
|
// Copy a src directory to dest directory.
|
||||||
|
// <dir_src> + <dir_dst> -> <dir_dst>/<dir_src>
|
||||||
|
CopyModeDirToDir
|
||||||
|
// Copy all files in the src directory to the dest directory. This works recursively.
|
||||||
|
// <dir_src>/ + <dir_dst> -> <dir_dst>/<files_from_dir_src>
|
||||||
|
CopyModeFilesToDir
|
||||||
|
)
|
||||||
|
|
||||||
|
// copyMode takes a src and dest path and file mode to determine the copy mode.
|
||||||
|
// See the possible copy modes and their documentation.
|
||||||
|
func copyMode(srcPath, dstPath string, srcMode os.FileMode, dstMode os.FileMode, dstExists bool) (CopyMode, error) {
|
||||||
|
_, srcFile := path.Split(srcPath)
|
||||||
|
_, dstFile := path.Split(dstPath)
|
||||||
|
if srcMode.IsDir() {
|
||||||
|
if !dstExists {
|
||||||
|
return -1, ErrDstDirNotExist
|
||||||
|
}
|
||||||
|
if dstMode.IsDir() {
|
||||||
|
if strings.HasSuffix(srcPath, "/") {
|
||||||
|
return CopyModeFilesToDir, nil
|
||||||
|
}
|
||||||
|
return CopyModeDirToDir, nil
|
||||||
|
}
|
||||||
|
return -1, ErrCopyDirToFile
|
||||||
|
}
|
||||||
|
|
||||||
|
if dstMode.IsDir() {
|
||||||
|
return CopyModeFileToDir, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if srcFile != dstFile {
|
||||||
|
return CopyModeFileToFileRename, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return CopyModeFileToFile, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// moveDir moves all files from a source path to the destination path recursively.
|
||||||
|
func moveDir(sourcePath, destPath string) error {
|
||||||
|
return filepath.Walk(sourcePath, func(p string, info os.FileInfo, err error) error {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
newPath := path.Join(destPath, strings.TrimPrefix(p, sourcePath))
|
||||||
|
if info.IsDir() {
|
||||||
|
err := os.Mkdir(newPath, info.Mode())
|
||||||
|
if err != nil {
|
||||||
|
if os.IsExist(err) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if info.Mode().IsRegular() {
|
||||||
|
return moveFile(p, newPath)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// moveFile moves a file from a source path to a destination path.
|
||||||
|
func moveFile(sourcePath, destPath string) error {
|
||||||
|
inputFile, err := os.Open(sourcePath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
outputFile, err := os.Create(destPath)
|
||||||
|
if err != nil {
|
||||||
|
inputFile.Close()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer outputFile.Close()
|
||||||
|
_, err = io.Copy(outputFile, inputFile)
|
||||||
|
inputFile.Close()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove file after succesfull copy.
|
||||||
|
err = os.Remove(sourcePath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppCpCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,113 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestParse(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
src string
|
||||||
|
dst string
|
||||||
|
srcPath string
|
||||||
|
dstPath string
|
||||||
|
service string
|
||||||
|
toContainer bool
|
||||||
|
err error
|
||||||
|
}{
|
||||||
|
{src: "foo", dst: "bar", err: errServiceMissing},
|
||||||
|
{src: "app:foo", dst: "app:bar", err: errServiceMissing},
|
||||||
|
{src: "app:foo", dst: "bar", srcPath: "foo", dstPath: "bar", service: "app", toContainer: false},
|
||||||
|
{src: "foo", dst: "app:bar", srcPath: "foo", dstPath: "bar", service: "app", toContainer: true},
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, tc := range tests {
|
||||||
|
srcPath, dstPath, service, toContainer, err := parseSrcAndDst(tc.src, tc.dst)
|
||||||
|
if srcPath != tc.srcPath {
|
||||||
|
t.Errorf("[%d] srcPath: want (%s), got(%s)", i, tc.srcPath, srcPath)
|
||||||
|
}
|
||||||
|
if dstPath != tc.dstPath {
|
||||||
|
t.Errorf("[%d] dstPath: want (%s), got(%s)", i, tc.dstPath, dstPath)
|
||||||
|
}
|
||||||
|
if service != tc.service {
|
||||||
|
t.Errorf("[%d] service: want (%s), got(%s)", i, tc.service, service)
|
||||||
|
}
|
||||||
|
if toContainer != tc.toContainer {
|
||||||
|
t.Errorf("[%d] toConainer: want (%t), got(%t)", i, tc.toContainer, toContainer)
|
||||||
|
}
|
||||||
|
if err == nil && tc.err != nil && err.Error() != tc.err.Error() {
|
||||||
|
t.Errorf("[%d] err: want (%s), got(%s)", i, tc.err, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCopyMode(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
srcPath string
|
||||||
|
dstPath string
|
||||||
|
srcMode os.FileMode
|
||||||
|
dstMode os.FileMode
|
||||||
|
dstExists bool
|
||||||
|
mode CopyMode
|
||||||
|
err error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
srcPath: "foo.txt",
|
||||||
|
dstPath: "foo.txt",
|
||||||
|
srcMode: os.ModePerm,
|
||||||
|
dstMode: os.ModePerm,
|
||||||
|
dstExists: true,
|
||||||
|
mode: CopyModeFileToFile,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
srcPath: "foo.txt",
|
||||||
|
dstPath: "bar.txt",
|
||||||
|
srcMode: os.ModePerm,
|
||||||
|
dstExists: true,
|
||||||
|
mode: CopyModeFileToFileRename,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
srcPath: "foo",
|
||||||
|
dstPath: "foo",
|
||||||
|
srcMode: os.ModeDir,
|
||||||
|
dstMode: os.ModeDir,
|
||||||
|
dstExists: true,
|
||||||
|
mode: CopyModeDirToDir,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
srcPath: "foo/",
|
||||||
|
dstPath: "foo",
|
||||||
|
srcMode: os.ModeDir,
|
||||||
|
dstMode: os.ModeDir,
|
||||||
|
dstExists: true,
|
||||||
|
mode: CopyModeFilesToDir,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
srcPath: "foo",
|
||||||
|
dstPath: "foo",
|
||||||
|
srcMode: os.ModeDir,
|
||||||
|
dstExists: false,
|
||||||
|
mode: -1,
|
||||||
|
err: ErrDstDirNotExist,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
srcPath: "foo",
|
||||||
|
dstPath: "foo",
|
||||||
|
srcMode: os.ModeDir,
|
||||||
|
dstMode: os.ModePerm,
|
||||||
|
dstExists: true,
|
||||||
|
mode: -1,
|
||||||
|
err: ErrCopyDirToFile,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, tc := range tests {
|
||||||
|
mode, err := copyMode(tc.srcPath, tc.dstPath, tc.srcMode, tc.dstMode, tc.dstExists)
|
||||||
|
if mode != tc.mode {
|
||||||
|
t.Errorf("[%d] mode: want (%d), got(%d)", i, tc.mode, mode)
|
||||||
|
}
|
||||||
|
if err != tc.err {
|
||||||
|
t.Errorf("[%d] err: want (%s), got(%s)", i, tc.err, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,458 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/secret"
|
||||||
|
"coopcloud.tech/tagcmp"
|
||||||
|
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/deploy"
|
||||||
|
"coopcloud.tech/abra/pkg/dns"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/lint"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app deploy` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appDeployAliases = i18n.G("d")
|
||||||
|
|
||||||
|
var AppDeployCommand = &cobra.Command{
|
||||||
|
// translators: `app deploy` command
|
||||||
|
Use: i18n.G("deploy <domain> [version] [flags]"),
|
||||||
|
Aliases: strings.Split(appDeployAliases, ","),
|
||||||
|
// translators: Short description for `app deploy` command
|
||||||
|
Short: i18n.G("Deploy an app"),
|
||||||
|
Long: i18n.G(`Deploy an app.
|
||||||
|
|
||||||
|
This command supports chaos operations. Use "--chaos/-C" to deploy your recipe
|
||||||
|
checkout as-is. Recipe commit hashes are also supported as values for
|
||||||
|
"[version]". Please note, "upgrade"/"rollback" do not support chaos operations.`),
|
||||||
|
Example: i18n.G(` # standard deployment
|
||||||
|
abra app deploy 1312.net
|
||||||
|
|
||||||
|
# chaos deployment
|
||||||
|
abra app deploy 1312.net --chaos
|
||||||
|
|
||||||
|
# deploy specific version
|
||||||
|
abra app deploy 1312.net 2.0.0+1.2.3
|
||||||
|
|
||||||
|
# deploy a specific git hash
|
||||||
|
abra app deploy 1312.net 886db76d`),
|
||||||
|
Args: cobra.RangeArgs(1, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
app, err := appPkg.Get(args[0])
|
||||||
|
if err != nil {
|
||||||
|
errMsg := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{errMsg}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
return autocomplete.RecipeVersionComplete(app.Recipe.Name)
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
var (
|
||||||
|
deployWarnMessages []string
|
||||||
|
toDeployVersion string
|
||||||
|
)
|
||||||
|
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := validateArgsAndFlags(args); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("checking whether %s is already deployed", app.StackName()))
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if deployMeta.IsDeployed && !(internal.Force || internal.Chaos) {
|
||||||
|
log.Fatal(i18n.G("%s is already deployed", app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
toDeployVersion, err = getDeployVersion(args, deployMeta, app)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
isChaosCommit, err := app.Recipe.IsChaosCommit(toDeployVersion)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to determine if %s is a chaos commit: %s", toDeployVersion, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isChaosCommit && !tagcmp.IsParsable(toDeployVersion) {
|
||||||
|
log.Fatal(i18n.G("unable to parse deploy version: %s", toDeployVersion))
|
||||||
|
}
|
||||||
|
|
||||||
|
if !internal.Chaos {
|
||||||
|
isChaosCommit, err := app.Recipe.EnsureVersion(toDeployVersion)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("ensure recipe: %s", err))
|
||||||
|
}
|
||||||
|
if isChaosCommit {
|
||||||
|
log.Warnf(i18n.G("version '%s' appears to be a chaos commit, but --chaos/-C was not provided", toDeployVersion))
|
||||||
|
internal.Chaos = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := lint.LintForErrors(app.Recipe); err != nil {
|
||||||
|
if internal.Chaos {
|
||||||
|
log.Warn(err)
|
||||||
|
} else {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := validateSecrets(cl, app); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := deploy.MergeAbraShEnv(app.Recipe, app.Env); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
stackName := app.StackName()
|
||||||
|
deployOpts := stack.Deploy{
|
||||||
|
Composefiles: composeFiles,
|
||||||
|
Namespace: stackName,
|
||||||
|
Prune: false,
|
||||||
|
ResolveImage: stack.ResolveImageAlways,
|
||||||
|
Detach: false,
|
||||||
|
}
|
||||||
|
compose, err := appPkg.GetAppComposeConfig(app.Name, deployOpts, app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
appPkg.SetRecipeLabel(compose, stackName, app.Recipe.Name)
|
||||||
|
appPkg.SetChaosLabel(compose, stackName, internal.Chaos)
|
||||||
|
if internal.Chaos {
|
||||||
|
appPkg.SetChaosVersionLabel(compose, stackName, toDeployVersion)
|
||||||
|
}
|
||||||
|
|
||||||
|
versionLabel := toDeployVersion
|
||||||
|
if internal.Chaos {
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
labelKey := fmt.Sprintf("coop-cloud.%s.version", stackName)
|
||||||
|
// NOTE(d1): keep non-chaos version labbeling when doing chaos ops
|
||||||
|
versionLabel = service.Deploy.Labels[labelKey]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
appPkg.SetVersionLabel(compose, stackName, versionLabel)
|
||||||
|
|
||||||
|
newRecipeWithDeployVersion := fmt.Sprintf("%s:%s", app.Recipe.Name, toDeployVersion)
|
||||||
|
appPkg.ExposeAllEnv(stackName, compose, app.Env, newRecipeWithDeployVersion)
|
||||||
|
|
||||||
|
envVars, err := appPkg.CheckEnv(app)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, envVar := range envVars {
|
||||||
|
if !envVar.Present {
|
||||||
|
deployWarnMessages = append(deployWarnMessages,
|
||||||
|
i18n.G("%s missing from %s.env", envVar.Name, app.Domain),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !internal.NoDomainChecks {
|
||||||
|
if domainName, ok := app.Env["DOMAIN"]; ok {
|
||||||
|
if _, err = dns.EnsureDomainsResolveSameIPv4(domainName, app.Server); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.Debug(i18n.G("skipping domain checks, no DOMAIN=... configured"))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.Debug(i18n.G("skipping domain checks"))
|
||||||
|
}
|
||||||
|
|
||||||
|
deployedVersion := config.MISSING_DEFAULT
|
||||||
|
if deployMeta.IsDeployed {
|
||||||
|
deployedVersion = deployMeta.Version
|
||||||
|
if deployMeta.IsChaos {
|
||||||
|
deployedVersion = deployMeta.ChaosVersion
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather secrets
|
||||||
|
secretInfo, err := deploy.GatherSecretsForDeploy(cl, app, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather configs
|
||||||
|
configInfo, err := deploy.GatherConfigsForDeploy(cl, app, compose, app.Env, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather images
|
||||||
|
imageInfo, err := deploy.GatherImagesForDeploy(cl, app, compose, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show deploy overview
|
||||||
|
if err := internal.DeployOverview(
|
||||||
|
app,
|
||||||
|
deployedVersion,
|
||||||
|
toDeployVersion,
|
||||||
|
"",
|
||||||
|
deployWarnMessages,
|
||||||
|
secretInfo,
|
||||||
|
configInfo,
|
||||||
|
imageInfo,
|
||||||
|
); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
stack.WaitTimeout, err = appPkg.GetTimeoutFromLabel(compose, stackName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
serviceNames, err := appPkg.GetAppServiceNames(app.Name)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := app.Filters(true, false, serviceNames...)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := stack.RunDeploy(
|
||||||
|
cl,
|
||||||
|
deployOpts,
|
||||||
|
compose,
|
||||||
|
app.Name,
|
||||||
|
app.Server,
|
||||||
|
internal.DontWaitConverge,
|
||||||
|
internal.NoInput,
|
||||||
|
f,
|
||||||
|
); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
postDeployCmds, ok := app.Env["POST_DEPLOY_CMDS"]
|
||||||
|
if ok && !internal.DontWaitConverge {
|
||||||
|
log.Debug(i18n.G("run the following post-deploy commands: %s", postDeployCmds))
|
||||||
|
if err := internal.PostCmds(cl, app, postDeployCmds); err != nil {
|
||||||
|
log.Fatal(i18n.G("attempting to run post deploy commands, saw: %s", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := app.WriteRecipeVersion(toDeployVersion, false); err != nil {
|
||||||
|
log.Fatal(i18n.G("writing recipe version failed: %s", err))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func getLatestVersionOrCommit(app appPkg.App) (string, error) {
|
||||||
|
recipeVersions, warnings, err := app.Recipe.GetRecipeVersions()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, warning := range warnings {
|
||||||
|
log.Warn(warning)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(recipeVersions) > 0 && !internal.Chaos {
|
||||||
|
latest := recipeVersions[len(recipeVersions)-1]
|
||||||
|
for tag := range latest {
|
||||||
|
log.Debug(i18n.G("selected latest recipe version: %s (from %d available versions)", tag, len(recipeVersions)))
|
||||||
|
return tag, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
head, err := app.Recipe.Head()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatter.SmallSHA(head.String()), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// validateArgsAndFlags ensures compatible args/flags.
|
||||||
|
func validateArgsAndFlags(args []string) error {
|
||||||
|
if len(args) == 2 && args[1] != "" && internal.Chaos {
|
||||||
|
return errors.New(i18n.G("cannot use [version] and --chaos together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) == 2 && args[1] != "" && internal.DeployLatest {
|
||||||
|
return errors.New(i18n.G("cannot use [version] and --latest together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.DeployLatest && internal.Chaos {
|
||||||
|
return errors.New(i18n.G("cannot use --chaos and --latest together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func validateSecrets(cl *dockerClient.Client, app appPkg.App) error {
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
secretsConfig, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
secStats, err := secret.PollSecretsStatus(cl, app)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, secStat := range secStats {
|
||||||
|
if !secStat.CreatedOnRemote {
|
||||||
|
secretConfig := secretsConfig[secStat.LocalName]
|
||||||
|
if secretConfig.SkipGenerate {
|
||||||
|
return errors.New(i18n.G("secret not inserted (#generate=false): %s", secStat.LocalName))
|
||||||
|
}
|
||||||
|
return errors.New(i18n.G("secret not generated: %s", secStat.LocalName))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getDeployVersion(cliArgs []string, deployMeta stack.DeployMeta, app appPkg.App) (string, error) {
|
||||||
|
// Chaos mode overrides everything
|
||||||
|
if internal.Chaos {
|
||||||
|
v, err := app.Recipe.ChaosVersion()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
log.Debug(i18n.G("version: taking chaos version: %s", v))
|
||||||
|
return v, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if the deploy version is set with a cli argument
|
||||||
|
if len(cliArgs) == 2 && cliArgs[1] != "" {
|
||||||
|
log.Debug(i18n.G("version: taking version from cli arg: %s", cliArgs[1]))
|
||||||
|
return cliArgs[1], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if the recipe has a version in the .env file
|
||||||
|
if app.Recipe.EnvVersion != "" && !internal.DeployLatest {
|
||||||
|
if strings.HasSuffix(app.Recipe.EnvVersionRaw, "+U") {
|
||||||
|
// NOTE(d1): use double-line 5 spaces ("FATA ") trick to make a more
|
||||||
|
// informative error message. it's ugly but that's our logging situation
|
||||||
|
// atm
|
||||||
|
return "", errors.New(i18n.G(`cannot redeploy previous chaos version (%s), did you mean to use "--chaos"?
|
||||||
|
to return to a regular release, specify a release tag, commit SHA or use "--latest"`,
|
||||||
|
formatter.BoldDirtyDefault(app.Recipe.EnvVersionRaw)))
|
||||||
|
}
|
||||||
|
log.Debug(i18n.G("version: taking version from .env file: %s", app.Recipe.EnvVersion))
|
||||||
|
return app.Recipe.EnvVersion, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Take deployed version
|
||||||
|
if deployMeta.IsDeployed && !internal.DeployLatest {
|
||||||
|
log.Debug(i18n.G("version: taking deployed version: %s", deployMeta.Version))
|
||||||
|
return deployMeta.Version, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
v, err := getLatestVersionOrCommit(app)
|
||||||
|
log.Debug(i18n.G("version: taking new recipe version: %s", v))
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return v, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppDeployCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppDeployCommand.Flags().BoolVarP(
|
||||||
|
&internal.Force,
|
||||||
|
i18n.G("force"),
|
||||||
|
i18n.G("f"),
|
||||||
|
false,
|
||||||
|
i18n.G("perform action without further prompt"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppDeployCommand.Flags().BoolVarP(
|
||||||
|
&internal.NoDomainChecks,
|
||||||
|
i18n.G("no-domain-checks"),
|
||||||
|
i18n.G("D"),
|
||||||
|
false,
|
||||||
|
i18n.G("disable public DNS checks"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppDeployCommand.Flags().BoolVarP(
|
||||||
|
&internal.DontWaitConverge,
|
||||||
|
i18n.G("no-converge-checks"),
|
||||||
|
i18n.G("c"),
|
||||||
|
false,
|
||||||
|
i18n.G("disable converge logic checks"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppDeployCommand.PersistentFlags().BoolVarP(
|
||||||
|
&internal.DeployLatest,
|
||||||
|
i18n.G("latest"),
|
||||||
|
i18n.G("l"),
|
||||||
|
false,
|
||||||
|
i18n.G("deploy latest recipe version"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppDeployCommand.Flags().BoolVarP(
|
||||||
|
&internal.ShowUnchanged,
|
||||||
|
i18n.G("show-unchanged"),
|
||||||
|
i18n.G("U"),
|
||||||
|
false,
|
||||||
|
i18n.G("show all configs & images, including unchanged ones"),
|
||||||
|
)
|
||||||
|
}
|
||||||
+344
@@ -0,0 +1,344 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/app"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
containerPkg "coopcloud.tech/abra/pkg/container"
|
||||||
|
contextPkg "coopcloud.tech/abra/pkg/context"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app env` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var appEnvAliases = i18n.G("e")
|
||||||
|
|
||||||
|
// translators: `abra app env list` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var appEnvListAliases = i18n.G("l,ls")
|
||||||
|
|
||||||
|
// translators: `abra app env pull` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var appEnvPullAliases = i18n.G("pl,p")
|
||||||
|
|
||||||
|
var AppEnvListCommand = &cobra.Command{
|
||||||
|
// translators: `app env list` command
|
||||||
|
Use: i18n.G("list <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appEnvListAliases, ","),
|
||||||
|
// translators: Short description for `app env list` command
|
||||||
|
Short: i18n.G("List all app environment values"),
|
||||||
|
Example: i18n.G(" abra app env list 1312.net"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
var envKeys []string
|
||||||
|
for k := range app.Env {
|
||||||
|
envKeys = append(envKeys, k)
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Strings(envKeys)
|
||||||
|
|
||||||
|
var rows [][]string
|
||||||
|
for _, k := range envKeys {
|
||||||
|
rows = append(rows, []string{k, app.Env[k]})
|
||||||
|
}
|
||||||
|
|
||||||
|
overview := formatter.CreateOverview(i18n.G("ENV OVERVIEW"), rows)
|
||||||
|
fmt.Println(overview)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var AppEnvPullCommand = &cobra.Command{
|
||||||
|
// translators: `app pull` command
|
||||||
|
Use: i18n.G("pull <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appEnvPullAliases, ","),
|
||||||
|
// translators: Short description for `app env pull` command
|
||||||
|
Short: i18n.G("Pull app environment values from a deployed app"),
|
||||||
|
Long: i18n.G(`Pull app environment values from a deploymed app.
|
||||||
|
|
||||||
|
A convenient command for when you've lost your app environment file or want to
|
||||||
|
synchronize your local app environment values with what is deployed live.`),
|
||||||
|
Example: i18n.G(` # pull existing .env file and overwrite local values
|
||||||
|
abra app env pull 1312.net --force
|
||||||
|
|
||||||
|
# pull lost app .env file
|
||||||
|
abra app env pull my.gitea.net --server 1312.net`),
|
||||||
|
Args: cobra.MaximumNArgs(2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
appName := args[0]
|
||||||
|
|
||||||
|
appEnvPath := path.Join(config.ABRA_DIR, "servers", server, fmt.Sprintf("%s.env", appName))
|
||||||
|
if _, err := os.Stat(appEnvPath); !os.IsNotExist(err) {
|
||||||
|
log.Fatal(i18n.G("%s already exists?", appEnvPath))
|
||||||
|
}
|
||||||
|
|
||||||
|
if server == "" {
|
||||||
|
log.Fatal(i18n.G("unable to determine server of app %s, please pass --server/-s", appName))
|
||||||
|
}
|
||||||
|
|
||||||
|
serverDir := filepath.Join(config.SERVERS_DIR, server)
|
||||||
|
if _, err := os.Stat(serverDir); os.IsNotExist(err) {
|
||||||
|
log.Fatal(i18n.G("unknown server %s, run \"abra server add %s\"?", server, server))
|
||||||
|
}
|
||||||
|
|
||||||
|
store := contextPkg.NewDefaultDockerContextStore()
|
||||||
|
contexts, err := store.Store.List()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to look up server context for %s: %s", server, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
var contextCreated bool
|
||||||
|
if server == "default" {
|
||||||
|
contextCreated = true
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, context := range contexts {
|
||||||
|
if context.Name == server {
|
||||||
|
contextCreated = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !contextCreated {
|
||||||
|
log.Fatal(i18n.G("%s missing context, run \"abra server add %s\"?", server, server))
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, appPkg.StackName(appName))
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s is not deployed?", appName))
|
||||||
|
}
|
||||||
|
|
||||||
|
filters := filters.NewArgs()
|
||||||
|
filters.Add("name", fmt.Sprintf("^%s_%s", app.StackName(appName), "app"))
|
||||||
|
targetContainer, err := containerPkg.GetContainer(context.Background(), cl, filters, internal.NoInput)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to retrieve container for %s: %s", appName, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
inspectResult, err := cl.ContainerInspect(context.Background(), targetContainer.ID)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to inspect container for %s: %s", appName, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
deploymentEnv := make(map[string]string)
|
||||||
|
for _, envVar := range inspectResult.Config.Env {
|
||||||
|
split := strings.SplitN(envVar, "=", 2)
|
||||||
|
if len(split) != 2 {
|
||||||
|
log.Debug(i18n.G("no value attached to %s", envVar))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
key, val := split[0], split[1]
|
||||||
|
deploymentEnv[key] = val
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("pulled env values from %s deployment: %s", appName, deploymentEnv))
|
||||||
|
|
||||||
|
var (
|
||||||
|
recipeEnvVar string
|
||||||
|
recipeKey string
|
||||||
|
)
|
||||||
|
|
||||||
|
if r, ok := deploymentEnv["TYPE"]; ok {
|
||||||
|
recipeKey = "TYPE"
|
||||||
|
recipeEnvVar = r
|
||||||
|
}
|
||||||
|
|
||||||
|
if r, ok := deploymentEnv["RECIPE"]; ok {
|
||||||
|
recipeKey = "RECIPE"
|
||||||
|
recipeEnvVar = r
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeEnvVar == "" {
|
||||||
|
log.Fatal(i18n.G("unable to determine recipe type from %s, env: %v", appName, inspectResult.Config.Env))
|
||||||
|
}
|
||||||
|
|
||||||
|
var recipeName = recipeEnvVar
|
||||||
|
if strings.Contains(recipeEnvVar, ":") {
|
||||||
|
split := strings.Split(recipeEnvVar, ":")
|
||||||
|
recipeName = split[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
recipe := internal.ValidateRecipe(
|
||||||
|
[]string{recipeName},
|
||||||
|
cmd.Name(),
|
||||||
|
)
|
||||||
|
|
||||||
|
version := deployMeta.Version
|
||||||
|
if deployMeta.IsChaos {
|
||||||
|
version = deployMeta.ChaosVersion
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := recipe.EnsureVersion(version); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
mergedEnv, err := recipe.SampleEnv()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("retrieved env values from .env.sample of %s: %s", recipe.Name, mergedEnv))
|
||||||
|
|
||||||
|
for k, v := range deploymentEnv {
|
||||||
|
mergedEnv[k] = v
|
||||||
|
}
|
||||||
|
|
||||||
|
if !strings.Contains(recipeEnvVar, ":") {
|
||||||
|
mergedEnv[recipeKey] = fmt.Sprintf("%s:%s", mergedEnv[recipeKey], version)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("final merged env values for %s are: %s", appName, mergedEnv))
|
||||||
|
|
||||||
|
envSample, err := os.ReadFile(recipe.SampleEnvPath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = os.WriteFile(appEnvPath, envSample, 0o664)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to write new env %s: %s", appEnvPath, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
read, err := os.ReadFile(appEnvPath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to read new env %s: %s", appEnvPath, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
sampleEnv, err := recipe.SampleEnv()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var composeFileUpdated bool
|
||||||
|
newContents := string(read)
|
||||||
|
for key, val := range mergedEnv {
|
||||||
|
if sampleEnv[key] == val {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if key == "COMPOSE_FILE" {
|
||||||
|
composeFileUpdated = true
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if m, _ := regexp.MatchString(fmt.Sprintf(`#%s=`, key), newContents); m {
|
||||||
|
log.Debug(i18n.G("uncommenting %s", key))
|
||||||
|
re := regexp.MustCompile(fmt.Sprintf(`#%s=`, key))
|
||||||
|
newContents = re.ReplaceAllString(newContents, fmt.Sprintf("%s=", key))
|
||||||
|
}
|
||||||
|
|
||||||
|
if m, _ := regexp.MatchString(fmt.Sprintf(`# %s=`, key), newContents); m {
|
||||||
|
log.Debug(i18n.G("uncommenting %s", key))
|
||||||
|
re := regexp.MustCompile(fmt.Sprintf(`# %s=`, key))
|
||||||
|
newContents = re.ReplaceAllString(newContents, fmt.Sprintf("%s=", key))
|
||||||
|
}
|
||||||
|
|
||||||
|
if m, _ := regexp.MatchString(fmt.Sprintf(`%s=".*"`, key), newContents); m {
|
||||||
|
log.Debug(i18n.G(`inserting %s="%s" (double quotes)`, key, val))
|
||||||
|
re := regexp.MustCompile(fmt.Sprintf(`%s=".*"`, key))
|
||||||
|
newContents = re.ReplaceAllString(newContents, fmt.Sprintf(`%s="%s"`, key, val))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if m, _ := regexp.MatchString(fmt.Sprintf(`%s='.*'`, key), newContents); m {
|
||||||
|
log.Debug(i18n.G(`inserting %s='%s' (single quotes)`, key, val))
|
||||||
|
re := regexp.MustCompile(fmt.Sprintf(`%s='.*'`, key))
|
||||||
|
newContents = re.ReplaceAllString(newContents, fmt.Sprintf(`%s='%s'`, key, val))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if m, _ := regexp.MatchString(fmt.Sprintf("%s=.*", key), newContents); m {
|
||||||
|
log.Debug(i18n.G("inserting %s=%s (no quotes)", key, val))
|
||||||
|
re := regexp.MustCompile(fmt.Sprintf("%s=.*", key))
|
||||||
|
newContents = re.ReplaceAllString(newContents, fmt.Sprintf("%s=%s", key, val))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err = os.WriteFile(appEnvPath, []byte(newContents), 0)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to write new env %s: %s", appEnvPath, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("%s successfully created", appEnvPath))
|
||||||
|
|
||||||
|
if composeFileUpdated {
|
||||||
|
log.Warn(i18n.G("manual update required: COMPOSE_FILE=\"%s\"", mergedEnv["COMPOSE_FILE"]))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var AppEnvCommand = &cobra.Command{
|
||||||
|
// translators: `app env` command group
|
||||||
|
Use: i18n.G("env [cmd] [args] [flags]"),
|
||||||
|
Aliases: strings.Split(appEnvAliases, ","),
|
||||||
|
// translators: Short description for `app env` command group
|
||||||
|
Short: i18n.G("Manage app environment values"),
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
server string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppEnvPullCommand.Flags().BoolVarP(
|
||||||
|
&internal.Force,
|
||||||
|
i18n.G("force"),
|
||||||
|
i18n.G("f"),
|
||||||
|
false,
|
||||||
|
i18n.G("perform action without further prompt"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppEnvPullCommand.Flags().StringVarP(
|
||||||
|
&server,
|
||||||
|
i18n.G("server"),
|
||||||
|
i18n.G("s"),
|
||||||
|
"",
|
||||||
|
i18n.G("server associated with deployed app"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppEnvPullCommand.RegisterFlagCompletionFunc(
|
||||||
|
i18n.G("server"),
|
||||||
|
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.ServerNameComplete()
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,147 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/convert"
|
||||||
|
composetypes "github.com/docker/cli/cli/compose/types"
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app labels` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var appLabelsAliases = i18n.G("lb")
|
||||||
|
|
||||||
|
var AppLabelsCommand = &cobra.Command{
|
||||||
|
// translators: `app labels` command
|
||||||
|
Use: i18n.G("labels <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appLabelsAliases, ","),
|
||||||
|
// translators: Short description for `app labels` command
|
||||||
|
Short: i18n.G("Show deployment labels"),
|
||||||
|
Long: i18n.G("Both local recipe and live deployment labels are shown."),
|
||||||
|
Example: " " + i18n.G("abra app labels 1312.net"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
remoteLabels, err := getLabels(cl, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
rows := [][]string{
|
||||||
|
{i18n.G("DEPLOYED LABELS"), "---"},
|
||||||
|
}
|
||||||
|
|
||||||
|
remoteLabelKeys := make([]string, 0, len(remoteLabels))
|
||||||
|
for k := range remoteLabels {
|
||||||
|
remoteLabelKeys = append(remoteLabelKeys, k)
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Strings(remoteLabelKeys)
|
||||||
|
|
||||||
|
for _, k := range remoteLabelKeys {
|
||||||
|
rows = append(rows, []string{
|
||||||
|
k,
|
||||||
|
remoteLabels[k],
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(remoteLabelKeys) == 0 {
|
||||||
|
rows = append(rows, []string{i18n.G("unknown")})
|
||||||
|
}
|
||||||
|
|
||||||
|
rows = append(rows, []string{i18n.G("RECIPE LABELS"), "---"})
|
||||||
|
|
||||||
|
config, err := app.Recipe.GetComposeConfig(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var localLabelKeys []string
|
||||||
|
var appServiceConfig composetypes.ServiceConfig
|
||||||
|
for _, service := range config.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
appServiceConfig = service
|
||||||
|
|
||||||
|
for k := range service.Deploy.Labels {
|
||||||
|
localLabelKeys = append(localLabelKeys, k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Strings(localLabelKeys)
|
||||||
|
|
||||||
|
for _, k := range localLabelKeys {
|
||||||
|
rows = append(rows, []string{
|
||||||
|
k,
|
||||||
|
appServiceConfig.Deploy.Labels[k],
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
overview := formatter.CreateOverview(i18n.G("LABELS OVERVIEW"), rows)
|
||||||
|
fmt.Println(overview)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// getLabels reads docker labels from running services in the format of "coop-cloud.${STACK_NAME}.${LABEL}".
|
||||||
|
func getLabels(cl *dockerClient.Client, stackName string) (map[string]string, error) {
|
||||||
|
labels := make(map[string]string)
|
||||||
|
|
||||||
|
filter := filters.NewArgs()
|
||||||
|
filter.Add("label", fmt.Sprintf("%s=%s", convert.LabelNamespace, stackName))
|
||||||
|
|
||||||
|
services, err := cl.ServiceList(context.Background(), types.ServiceListOptions{Filters: filter})
|
||||||
|
if err != nil {
|
||||||
|
return labels, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, service := range services {
|
||||||
|
if service.Spec.Name != fmt.Sprintf("%s_app", stackName) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for k, v := range service.Spec.Labels {
|
||||||
|
labels[k] = v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return labels, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppLabelsCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
+346
@@ -0,0 +1,346 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/tagcmp"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
type appStatus struct {
|
||||||
|
Server string `json:"server"`
|
||||||
|
Recipe string `json:"recipe"`
|
||||||
|
AppName string `json:"appName"`
|
||||||
|
Domain string `json:"domain"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
Chaos string `json:"chaos"`
|
||||||
|
ChaosVersion string `json:"chaosVersion"`
|
||||||
|
Version string `json:"version"`
|
||||||
|
Upgrade string `json:"upgrade"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type serverStatus struct {
|
||||||
|
Apps []appStatus `json:"apps"`
|
||||||
|
AppCount int `json:"appCount"`
|
||||||
|
VersionCount int `json:"versionCount"`
|
||||||
|
UnversionedCount int `json:"unversionedCount"`
|
||||||
|
LatestCount int `json:"latestCount"`
|
||||||
|
UpgradeCount int `json:"upgradeCount"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app list` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appListAliases = i18n.G("ls")
|
||||||
|
|
||||||
|
var AppListCommand = &cobra.Command{
|
||||||
|
// translators: `app list` command
|
||||||
|
Use: i18n.G("list [flags]"),
|
||||||
|
Aliases: strings.Split(appListAliases, ","),
|
||||||
|
// translators: Short description for `app list` command
|
||||||
|
Short: i18n.G("List all managed apps"),
|
||||||
|
Long: i18n.G(`Generate a report of all managed apps.
|
||||||
|
|
||||||
|
Use "--status/-S" flag to query all servers for the live deployment status.`),
|
||||||
|
Example: i18n.G(` # list apps of all servers without live status
|
||||||
|
abra app ls
|
||||||
|
|
||||||
|
# list apps of a specific server with live status
|
||||||
|
abra app ls -s 1312.net -S
|
||||||
|
|
||||||
|
# list apps of all servers which match a specific recipe
|
||||||
|
abra app ls -r gitea`),
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
appFiles, err := appPkg.LoadAppFiles(listAppServer)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
apps, err := appPkg.GetApps(appFiles, recipeFilter)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Sort(appPkg.ByServerAndRecipe(apps))
|
||||||
|
|
||||||
|
statuses := make(map[string]map[string]string)
|
||||||
|
if status {
|
||||||
|
alreadySeen := make(map[string]bool)
|
||||||
|
for _, app := range apps {
|
||||||
|
if _, ok := alreadySeen[app.Server]; !ok {
|
||||||
|
alreadySeen[app.Server] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
statuses, err = appPkg.GetAppStatuses(apps, internal.MachineReadable)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var totalServersCount int
|
||||||
|
var totalAppsCount int
|
||||||
|
allStats := make(map[string]serverStatus)
|
||||||
|
for _, app := range apps {
|
||||||
|
var stats serverStatus
|
||||||
|
var ok bool
|
||||||
|
if stats, ok = allStats[app.Server]; !ok {
|
||||||
|
stats = serverStatus{}
|
||||||
|
if recipeFilter == "" {
|
||||||
|
// count server, no filtering
|
||||||
|
totalServersCount++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if app.Recipe.Name == recipeFilter || recipeFilter == "" {
|
||||||
|
if recipeFilter != "" {
|
||||||
|
// only count server if matches filter
|
||||||
|
totalServersCount++
|
||||||
|
}
|
||||||
|
|
||||||
|
appStats := appStatus{}
|
||||||
|
stats.AppCount++
|
||||||
|
totalAppsCount++
|
||||||
|
|
||||||
|
if status {
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
status := i18n.G("unknown")
|
||||||
|
version := i18n.G("unknown")
|
||||||
|
chaos := i18n.G("unknown")
|
||||||
|
chaosVersion := i18n.G("unknown")
|
||||||
|
if statusMeta, ok := statuses[app.StackName()]; ok {
|
||||||
|
if currentVersion, exists := statusMeta["version"]; exists {
|
||||||
|
if currentVersion != "" {
|
||||||
|
version = currentVersion
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if chaosDeploy, exists := statusMeta["chaos"]; exists {
|
||||||
|
chaos = chaosDeploy
|
||||||
|
}
|
||||||
|
if chaosDeployVersion, exists := statusMeta["chaosVersion"]; exists {
|
||||||
|
chaosVersion = chaosDeployVersion
|
||||||
|
}
|
||||||
|
if statusMeta["status"] != "" {
|
||||||
|
status = statusMeta["status"]
|
||||||
|
}
|
||||||
|
stats.VersionCount++
|
||||||
|
} else {
|
||||||
|
stats.UnversionedCount++
|
||||||
|
}
|
||||||
|
|
||||||
|
appStats.Status = status
|
||||||
|
appStats.Chaos = chaos
|
||||||
|
appStats.ChaosVersion = chaosVersion
|
||||||
|
appStats.Version = version
|
||||||
|
|
||||||
|
var newUpdates []string
|
||||||
|
if version != "unknown" && chaos == "false" {
|
||||||
|
if err := app.Recipe.EnsureExists(); err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to clone %s: %s", app.Name, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
updates, err := app.Recipe.Tags()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to retrieve tags for %s: %s", app.Name, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
parsedVersion, err := tagcmp.Parse(version)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, update := range updates {
|
||||||
|
if ok := tagcmp.IsParsable(update); !ok {
|
||||||
|
log.Debug(i18n.G("unable to parse %s, skipping as upgrade option", update))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
parsedUpdate, err := tagcmp.Parse(update)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if update != version && parsedUpdate.IsGreaterThan(parsedVersion) {
|
||||||
|
newUpdates = append(newUpdates, update)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(newUpdates) == 0 {
|
||||||
|
if version == "unknown" {
|
||||||
|
appStats.Upgrade = i18n.G("unknown")
|
||||||
|
} else {
|
||||||
|
appStats.Upgrade = i18n.G("latest")
|
||||||
|
stats.LatestCount++
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
newUpdates = internal.SortVersionsDesc(newUpdates)
|
||||||
|
appStats.Upgrade = strings.Join(newUpdates, "\n")
|
||||||
|
stats.UpgradeCount++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
appStats.Server = app.Server
|
||||||
|
appStats.Recipe = app.Recipe.Name
|
||||||
|
appStats.AppName = app.Name
|
||||||
|
appStats.Domain = app.Domain
|
||||||
|
|
||||||
|
stats.Apps = append(stats.Apps, appStats)
|
||||||
|
}
|
||||||
|
allStats[app.Server] = stats
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.MachineReadable {
|
||||||
|
jsonstring, err := json.Marshal(allStats)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
} else {
|
||||||
|
fmt.Println(string(jsonstring))
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
alreadySeen := make(map[string]bool)
|
||||||
|
for _, app := range apps {
|
||||||
|
if _, ok := alreadySeen[app.Server]; ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
serverStat := allStats[app.Server]
|
||||||
|
|
||||||
|
headers := []string{i18n.G("RECIPE"), i18n.G("DOMAIN"), i18n.G("SERVER")}
|
||||||
|
if status {
|
||||||
|
headers = append(headers, []string{
|
||||||
|
i18n.G("STATUS"),
|
||||||
|
i18n.G("CHAOS"),
|
||||||
|
i18n.G("VERSION"),
|
||||||
|
i18n.G("UPGRADE"),
|
||||||
|
}...,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Headers(headers...)
|
||||||
|
|
||||||
|
var rows [][]string
|
||||||
|
for _, appStat := range serverStat.Apps {
|
||||||
|
row := []string{appStat.Recipe, appStat.Domain, appStat.Server}
|
||||||
|
if status {
|
||||||
|
chaosStatus := appStat.Chaos
|
||||||
|
if chaosStatus != "unknown" {
|
||||||
|
chaosEnabled, err := strconv.ParseBool(chaosStatus)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if chaosEnabled && appStat.ChaosVersion != "unknown" {
|
||||||
|
chaosStatus = appStat.ChaosVersion
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
row = append(row, []string{
|
||||||
|
appStat.Status,
|
||||||
|
chaosStatus,
|
||||||
|
appStat.Version,
|
||||||
|
appStat.Upgrade}...,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
rows = append(rows, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Rows(rows...)
|
||||||
|
|
||||||
|
if len(rows) > 0 {
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(allStats) > 1 && len(rows) > 0 {
|
||||||
|
fmt.Println() // newline separator for multiple servers
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
alreadySeen[app.Server] = true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
status bool
|
||||||
|
recipeFilter string
|
||||||
|
listAppServer string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppListCommand.Flags().BoolVarP(
|
||||||
|
&status,
|
||||||
|
i18n.G("status"),
|
||||||
|
i18n.G("S"),
|
||||||
|
false,
|
||||||
|
i18n.G("show app deployment status"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppListCommand.Flags().StringVarP(
|
||||||
|
&recipeFilter,
|
||||||
|
i18n.G("recipe"),
|
||||||
|
i18n.G("r"),
|
||||||
|
"",
|
||||||
|
i18n.G("show apps of a specific recipe"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppListCommand.RegisterFlagCompletionFunc(
|
||||||
|
i18n.G("recipe"),
|
||||||
|
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
AppListCommand.Flags().BoolVarP(
|
||||||
|
&internal.MachineReadable,
|
||||||
|
i18n.G("machine"),
|
||||||
|
i18n.G("m"),
|
||||||
|
false,
|
||||||
|
i18n.G("print machine-readable output"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppListCommand.Flags().StringVarP(
|
||||||
|
&listAppServer,
|
||||||
|
i18n.G("server"),
|
||||||
|
i18n.G("s"),
|
||||||
|
"",
|
||||||
|
i18n.G("show apps of a specific server"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppListCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppListCommand.RegisterFlagCompletionFunc(
|
||||||
|
i18n.G("server"),
|
||||||
|
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.ServerNameComplete()
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
+113
@@ -0,0 +1,113 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/logs"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app logs` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appLogsAliases = i18n.G("l")
|
||||||
|
|
||||||
|
var AppLogsCommand = &cobra.Command{
|
||||||
|
// translators: `app logs` command
|
||||||
|
Use: i18n.G("logs <domain> [service] [flags]"),
|
||||||
|
Aliases: strings.Split(appLogsAliases, ","),
|
||||||
|
// translators: Short description for `app logs` command
|
||||||
|
Short: i18n.G("Tail app logs"),
|
||||||
|
Args: cobra.RangeArgs(1, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
app, err := appPkg.Get(args[0])
|
||||||
|
if err != nil {
|
||||||
|
return []string{i18n.G("autocomplete failed: %s", err)}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
return autocomplete.ServiceNameComplete(app.Name)
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
stackName := app.StackName()
|
||||||
|
|
||||||
|
if err := app.Recipe.EnsureExists(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, stackName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s is not deployed?", app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
var serviceNames []string
|
||||||
|
if len(args) == 2 {
|
||||||
|
serviceNames = []string{args[1]}
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := app.Filters(true, false, serviceNames...)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := logs.TailOpts{
|
||||||
|
AppName: app.Name,
|
||||||
|
Services: serviceNames,
|
||||||
|
StdErr: stdErr,
|
||||||
|
Since: sinceLogs,
|
||||||
|
Filters: f,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := logs.TailLogs(cl, opts); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
stdErr bool
|
||||||
|
sinceLogs string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppLogsCommand.Flags().BoolVarP(
|
||||||
|
&stdErr,
|
||||||
|
i18n.G("stderr"),
|
||||||
|
i18n.G("s"),
|
||||||
|
false,
|
||||||
|
i18n.G("only tail stderr"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppLogsCommand.Flags().StringVarP(
|
||||||
|
&sinceLogs,
|
||||||
|
i18n.G("since"),
|
||||||
|
i18n.G("S"),
|
||||||
|
"",
|
||||||
|
i18n.G("tail logs since YYYY-MM-DDTHH:MM:SSZ"),
|
||||||
|
)
|
||||||
|
}
|
||||||
+354
@@ -0,0 +1,354 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/app"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
containerPkg "coopcloud.tech/abra/pkg/container"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/secret"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
containertypes "github.com/docker/docker/api/types/container"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
"github.com/docker/docker/api/types/mount"
|
||||||
|
"github.com/docker/docker/api/types/swarm"
|
||||||
|
"github.com/docker/docker/api/types/volume"
|
||||||
|
dockerclient "github.com/docker/docker/client"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app move` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var appMoveAliases = i18n.G("m")
|
||||||
|
|
||||||
|
var AppMoveCommand = &cobra.Command{
|
||||||
|
// translators: `app move` command
|
||||||
|
Use: i18n.G("move <domain> <server> [flags]"),
|
||||||
|
Aliases: strings.Split(appMoveAliases, ","),
|
||||||
|
// translators: Short description for `app move` command
|
||||||
|
Short: i18n.G("Moves an app to a different server"),
|
||||||
|
Long: i18n.G(`Move an app to a differnt server.
|
||||||
|
|
||||||
|
This command will migrate an app config and copy secrets and volumes from the
|
||||||
|
old server to the new one. The app MUST be deployed on the old server before
|
||||||
|
doing the move. The app will be undeployed from the current server but not
|
||||||
|
deployed on the new server.
|
||||||
|
|
||||||
|
The "tar" command is required on both the old and new server as well as "sudo"
|
||||||
|
permissions. The "rsync" command is required on your local machine for
|
||||||
|
transferring volumes.
|
||||||
|
|
||||||
|
Do not forget to update your DNS records. Don't panic, it might take a while
|
||||||
|
for the dust to settle after you move an app. If anything goes wrong, you can
|
||||||
|
always move the app config file to the original server and deploy it there
|
||||||
|
again. No data is removed from the old server.
|
||||||
|
|
||||||
|
Use "--dry-run/-r" to see which secrets and volumes will be moved.`),
|
||||||
|
Example: i18n.G(` # move an app
|
||||||
|
abra app move nextcloud.1312.net myserver.com`),
|
||||||
|
Args: cobra.RangeArgs(1, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
return autocomplete.ServerNameComplete()
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if len(args) <= 1 {
|
||||||
|
log.Fatal(i18n.G("no server provided?"))
|
||||||
|
}
|
||||||
|
newServer := internal.ValidateServer([]string{args[1]})
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
currentServerClient, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), currentServerClient, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s must first be deployed on %s before moving", app.Name, app.Server))
|
||||||
|
}
|
||||||
|
|
||||||
|
resources, err := getAppResources(currentServerClient, app)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to retrieve %s resources on %s: %s", app.Name, app.Server, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
internal.MoveOverview(app, newServer, resources.SecretNames(), resources.VolumeNames())
|
||||||
|
if err := internal.PromptProcced(); err != nil {
|
||||||
|
log.Fatal(i18n.G("bailing out: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("undeploying %s on %s", app.Name, app.Server))
|
||||||
|
rmOpts := stack.Remove{
|
||||||
|
Namespaces: []string{app.StackName()},
|
||||||
|
Detach: false,
|
||||||
|
}
|
||||||
|
if err := stack.RunRemove(context.Background(), currentServerClient, rmOpts); err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to remove app from %s: %s", err, app.Server))
|
||||||
|
}
|
||||||
|
|
||||||
|
newServerClient, err := client.New(newServer)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, s := range resources.SecretList {
|
||||||
|
sname := strings.Split(strings.TrimPrefix(s.Spec.Name, app.StackName()+"_"), "_")
|
||||||
|
secretName := strings.Join(sname[:len(sname)-1], "_")
|
||||||
|
data := resources.Secrets[secretName]
|
||||||
|
if err := client.StoreSecret(newServerClient, s.Spec.Name, data); err != nil {
|
||||||
|
if strings.Contains(err.Error(), "already exists") {
|
||||||
|
log.Info(i18n.G("skipping secret (because it already exists) on %s: %s", s.Spec.Name, newServer))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
log.Fatal(i18n.G("failed to store secret on %s: %s", err, newServer))
|
||||||
|
}
|
||||||
|
log.Info(i18n.G("created secret on %s: %s", s.Spec.Name, newServer))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range resources.Volumes {
|
||||||
|
log.Info(i18n.G("moving volume %s from %s to %s", v.Name, app.Server, newServer))
|
||||||
|
|
||||||
|
// NOTE(p4u1): Need to create the volume before copying the data, because
|
||||||
|
// when docker creates a new volume it set the folder permissions to
|
||||||
|
// root, which might be wrong. This ensures we always have the correct
|
||||||
|
// folder permissions inside the volume.
|
||||||
|
log.Debug(i18n.G("creating volume %s on %s", v.Name, newServer))
|
||||||
|
_, err := newServerClient.VolumeCreate(context.Background(), volume.CreateOptions{
|
||||||
|
Name: v.Name,
|
||||||
|
Driver: v.Driver,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to create volume %s on %s: %s", v.Name, newServer, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
filename := fmt.Sprintf("%s_outgoing.tar.gz", v.Name)
|
||||||
|
log.Debug(i18n.G("creating %s on %s", filename, app.Server))
|
||||||
|
tarCmd := fmt.Sprintf("sudo tar --same-owner -czhpf %s -C /var/lib/docker/volumes %s", filename, v.Name)
|
||||||
|
cmd := exec.Command("ssh", app.Server, "-tt", tarCmd)
|
||||||
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
|
log.Fatal(i18n.G("%s failed on %s: output:%s err:%s", tarCmd, app.Server, string(out), err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("rsyncing %s from %s to local machine", filename, app.Server))
|
||||||
|
cmd = exec.Command("rsync", "-a", "-v", fmt.Sprintf("%s:%s", app.Server, filename), filename)
|
||||||
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to copy %s from %s to local machine: output:%s err:%s", filename, app.Server, string(out), err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("rsyncing %s to %s from local machine", filename, filename, newServer))
|
||||||
|
cmd = exec.Command("rsync", "-a", "-v", filename, fmt.Sprintf("%s:%s", newServer, filename))
|
||||||
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to copy %s from local machine to %s: output:%s err:%s", filename, newServer, string(out), err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("extracting %s on %s", filename, newServer))
|
||||||
|
tarExtractCmd := fmt.Sprintf("sudo tar --same-owner -xzpf %s -C /var/lib/docker/volumes", filename)
|
||||||
|
cmd = exec.Command("ssh", newServer, "-tt", tarExtractCmd)
|
||||||
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
|
log.Fatal(i18n.G("%s failed to extract %s on %s: output:%s err:%s", tarExtractCmd, filename, newServer, string(out), err))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove tar files
|
||||||
|
log.Debug(i18n.G("removing %s from %s", filename, newServer))
|
||||||
|
cmd = exec.Command("ssh", newServer, "-tt", fmt.Sprintf("sudo rm -rf %s", filename))
|
||||||
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to remove %s from %s: output:%s err:%s", filename, newServer, string(out), err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("removing %s from %s", filename, app.Server))
|
||||||
|
cmd = exec.Command("ssh", app.Server, "-tt", fmt.Sprintf("sudo rm -rf %s", filename))
|
||||||
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to remove %s from %s: output:%s err:%s", filename, app.Server, string(out), err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("removing %s from local machine", filename))
|
||||||
|
cmd = exec.Command("rm", "-r", "-f", filename)
|
||||||
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to remove %s on local machine: output:%s err:%s", filename, string(out), err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
newServerPath := fmt.Sprintf("%s/servers/%s/%s.env", config.ABRA_DIR, newServer, app.Name)
|
||||||
|
log.Info(i18n.G("migrating app config from %s to %s", app.Server, newServerPath))
|
||||||
|
if err := copyFile(app.Path, newServerPath); err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to migrate app config: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := os.Remove(app.Path); err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to remove %s: %s", app.Path, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("%s was successfully moved from %s to %s 🎉", app.Name, app.Server, newServer))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
type AppResources struct {
|
||||||
|
Secrets map[string]string
|
||||||
|
SecretList []swarm.Secret
|
||||||
|
Volumes map[string]containertypes.MountPoint
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *AppResources) SecretNames() []string {
|
||||||
|
secrets := []string{}
|
||||||
|
for name := range a.Secrets {
|
||||||
|
secrets = append(secrets, name)
|
||||||
|
}
|
||||||
|
return secrets
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *AppResources) VolumeNames() []string {
|
||||||
|
volumes := []string{}
|
||||||
|
for name := range a.Volumes {
|
||||||
|
volumes = append(volumes, name)
|
||||||
|
}
|
||||||
|
return volumes
|
||||||
|
}
|
||||||
|
|
||||||
|
func getAppResources(cl *dockerclient.Client, app app.App) (*AppResources, error) {
|
||||||
|
filter, err := app.Filters(false, false)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
services, err := cl.ServiceList(context.Background(), types.ServiceListOptions{Filters: filter})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
secretList, err := cl.SecretList(context.Background(), types.SecretListOptions{Filters: filter})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
secretConfigs, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := stack.Deploy{Composefiles: composeFiles, Namespace: app.StackName()}
|
||||||
|
compose, err := appPkg.GetAppComposeConfig(app.Name, opts, app.Env)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
resources := &AppResources{
|
||||||
|
Secrets: make(map[string]string),
|
||||||
|
SecretList: secretList,
|
||||||
|
Volumes: make(map[string]containertypes.MountPoint),
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, s := range services {
|
||||||
|
secretNames := map[string]string{}
|
||||||
|
for _, serviceCompose := range compose.Services {
|
||||||
|
stackService := fmt.Sprintf("%s_%s", app.StackName(), serviceCompose.Name)
|
||||||
|
if stackService != s.Spec.Name {
|
||||||
|
log.Debug(i18n.G("skipping %s as it does not match %s", stackService, s.Spec.Name))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, secret := range serviceCompose.Secrets {
|
||||||
|
for _, s := range secretList {
|
||||||
|
stackSecret := fmt.Sprintf("%s_%s_%s", app.StackName(), secret.Source, secretConfigs[secret.Source].Version)
|
||||||
|
if s.Spec.Name == stackSecret {
|
||||||
|
secretNames[secret.Source] = s.ID
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
f := filters.NewArgs()
|
||||||
|
f.Add("name", s.Spec.Name)
|
||||||
|
targetContainer, err := containerPkg.GetContainer(context.Background(), cl, f, true)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.New(i18n.G("unable to get container matching %s: %s", s.Spec.Name, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, m := range targetContainer.Mounts {
|
||||||
|
if m.Type == mount.TypeVolume {
|
||||||
|
resources.Volumes[m.Name] = m
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for secretName, secretID := range secretNames {
|
||||||
|
if _, ok := resources.Secrets[secretName]; ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("extracting secret %s on %s", secretName, app.Server))
|
||||||
|
|
||||||
|
cmd := fmt.Sprintf("sudo cat /var/lib/docker/containers/%s/mounts/secrets/%s", targetContainer.ID, secretID)
|
||||||
|
out, err := exec.Command("ssh", app.Server, "-tt", cmd).Output()
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.New(i18n.G("%s failed on %s: output:%s err:%s", cmd, app.Server, string(out), err))
|
||||||
|
}
|
||||||
|
|
||||||
|
resources.Secrets[secretName] = string(out)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return resources, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func copyFile(src string, dst string) error {
|
||||||
|
// Read all content of src to data, may cause OOM for a large file.
|
||||||
|
data, err := os.ReadFile(src)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write data to dst
|
||||||
|
err = os.WriteFile(dst, data, 0o644)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppMoveCommand.Flags().BoolVarP(
|
||||||
|
&internal.Dry,
|
||||||
|
i18n.G("dry-run"),
|
||||||
|
i18n.G("r"),
|
||||||
|
false,
|
||||||
|
i18n.G("report changes that would be made"),
|
||||||
|
)
|
||||||
|
}
|
||||||
+412
@@ -0,0 +1,412 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/app"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
recipePkg "coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"coopcloud.tech/abra/pkg/secret"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
var appNewDescription = i18n.G(`Creates a new app from a default recipe.
|
||||||
|
|
||||||
|
This new app configuration is stored in your $ABRA_DIR directory under the
|
||||||
|
appropriate server.
|
||||||
|
|
||||||
|
This command does not deploy your app for you. You will need to run "abra app
|
||||||
|
deploy <domain>" to do so.
|
||||||
|
|
||||||
|
You can see what recipes are available (i.e. values for the [recipe] argument)
|
||||||
|
by running "abra recipe ls".
|
||||||
|
|
||||||
|
Recipe commit hashes are supported values for "[version]".
|
||||||
|
|
||||||
|
Passing the "--secrets/-S" flag will automatically generate secrets for your
|
||||||
|
app and store them encrypted at rest on the chosen target server. These
|
||||||
|
generated secrets are only visible at generation time, so please take care to
|
||||||
|
store them somewhere safe.
|
||||||
|
|
||||||
|
You can use the "--pass/-P" to store these generated passwords locally in a
|
||||||
|
pass store (see passwordstore.org for more). The pass command must be available
|
||||||
|
on your $PATH.`)
|
||||||
|
|
||||||
|
// translators: `abra app new` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appNewAliases = i18n.G("n")
|
||||||
|
|
||||||
|
var AppNewCommand = &cobra.Command{
|
||||||
|
// translators: `app new` command
|
||||||
|
Use: i18n.G("new [recipe] [version] [flags]"),
|
||||||
|
Aliases: strings.Split(appNewAliases, ","),
|
||||||
|
// translators: Short description for `app new` command
|
||||||
|
Short: i18n.G("Create a new app"),
|
||||||
|
Long: appNewDescription,
|
||||||
|
Args: cobra.RangeArgs(0, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
case 1:
|
||||||
|
recipe := internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
return autocomplete.RecipeVersionComplete(recipe.Name)
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
recipe := internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
|
||||||
|
if err := recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) == 2 && internal.Chaos {
|
||||||
|
log.Fatal(i18n.G("cannot use [version] and --chaos together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
var recipeVersion string
|
||||||
|
if len(args) == 2 {
|
||||||
|
recipeVersion = args[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
chaosVersion := config.CHAOS_DEFAULT
|
||||||
|
if internal.Chaos {
|
||||||
|
var err error
|
||||||
|
chaosVersion, err = recipe.ChaosVersion()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
recipeVersion = chaosVersion
|
||||||
|
} else {
|
||||||
|
if err := recipe.EnsureIsClean(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var recipeVersions recipePkg.RecipeVersions
|
||||||
|
if recipeVersion == "" {
|
||||||
|
var err error
|
||||||
|
var warnings []string
|
||||||
|
recipeVersions, warnings, err = recipe.GetRecipeVersions()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
for _, warning := range warnings {
|
||||||
|
log.Warn(warning)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(recipeVersions) > 0 {
|
||||||
|
latest := recipeVersions[len(recipeVersions)-1]
|
||||||
|
for tag := range latest {
|
||||||
|
recipeVersion = tag
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("selected recipe version: %s (from %d available versions)", recipeVersion, len(recipeVersions)))
|
||||||
|
|
||||||
|
if _, err := recipe.EnsureVersion(recipeVersion); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := recipe.EnsureLatest(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeVersion == "" {
|
||||||
|
head, err := recipe.Head()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("failed to retrieve latest commit for %s: %s", recipe.Name, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
recipeVersion = formatter.SmallSHA(head.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := ensureServerFlag(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := ensureDomainFlag(recipe, newAppServer); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
sanitisedAppName := appPkg.SanitiseAppName(appDomain)
|
||||||
|
log.Debug(i18n.G("%s sanitised as %s for new app", appDomain, sanitisedAppName))
|
||||||
|
|
||||||
|
if err := appPkg.TemplateAppEnvSample(
|
||||||
|
recipe,
|
||||||
|
appDomain,
|
||||||
|
newAppServer,
|
||||||
|
appDomain,
|
||||||
|
); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
sampleEnv, err := recipe.SampleEnv()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := recipe.GetComposeFiles(sampleEnv)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
secretsConfig, err := secret.ReadSecretsConfig(
|
||||||
|
recipe.SampleEnvPath,
|
||||||
|
composeFiles,
|
||||||
|
appPkg.StackName(appDomain),
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var appSecrets AppSecrets
|
||||||
|
if generateSecrets {
|
||||||
|
if err := promptForSecrets(recipe.Name, secretsConfig); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(newAppServer)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
appSecrets, err = createSecrets(cl, secretsConfig, sanitisedAppName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if newAppServer == "default" {
|
||||||
|
newAppServer = "local"
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("%s created (version: %s)", appDomain, recipeVersion))
|
||||||
|
|
||||||
|
if len(secretsConfig) > 0 {
|
||||||
|
var (
|
||||||
|
hasSecretToGenerate bool
|
||||||
|
hasSecretToSkip bool
|
||||||
|
)
|
||||||
|
|
||||||
|
for _, secretConfig := range secretsConfig {
|
||||||
|
if secretConfig.SkipGenerate {
|
||||||
|
hasSecretToSkip = true
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
hasSecretToGenerate = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if hasSecretToGenerate && !generateSecrets {
|
||||||
|
log.Warn(i18n.G("%s requires secret generation before deploy, run \"abra app secret generate %s --all\"", recipe.Name, appDomain))
|
||||||
|
}
|
||||||
|
|
||||||
|
if hasSecretToSkip {
|
||||||
|
log.Warn(i18n.G("%s requires secret insertion before deploy (#generate=false)", recipe.Name))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(appSecrets) > 0 {
|
||||||
|
rows := [][]string{}
|
||||||
|
for k, v := range appSecrets {
|
||||||
|
rows = append(rows, []string{k, v})
|
||||||
|
}
|
||||||
|
|
||||||
|
overview := formatter.CreateOverview(i18n.G("SECRETS OVERVIEW"), rows)
|
||||||
|
|
||||||
|
fmt.Println(overview)
|
||||||
|
|
||||||
|
log.Warn(i18n.G(
|
||||||
|
"secrets are %s shown again, please save them %s",
|
||||||
|
formatter.BoldUnderlineStyle.Render("NOT"),
|
||||||
|
formatter.BoldUnderlineStyle.Render("NOW"),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
app, err := app.Get(appDomain)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := app.WriteRecipeVersion(recipeVersion, false); err != nil {
|
||||||
|
log.Fatal(i18n.G("writing recipe version failed: %s", err))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// AppSecrets represents all app secrest
|
||||||
|
type AppSecrets map[string]string
|
||||||
|
|
||||||
|
// createSecrets creates all secrets for a new app.
|
||||||
|
func createSecrets(cl *dockerClient.Client, secretsConfig map[string]secret.Secret, sanitisedAppName string) (AppSecrets, error) {
|
||||||
|
// NOTE(d1): trim to match app.StackName() implementation
|
||||||
|
if len(sanitisedAppName) > config.MAX_SANITISED_APP_NAME_LENGTH {
|
||||||
|
log.Debug(i18n.G("trimming %s to %s to avoid runtime limits", sanitisedAppName, sanitisedAppName[:config.MAX_SANITISED_APP_NAME_LENGTH]))
|
||||||
|
sanitisedAppName = sanitisedAppName[:config.MAX_SANITISED_APP_NAME_LENGTH]
|
||||||
|
}
|
||||||
|
|
||||||
|
secrets, err := secret.GenerateSecrets(cl, secretsConfig, newAppServer)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if saveInPass {
|
||||||
|
for secretName := range secrets {
|
||||||
|
secretValue := secrets[secretName]
|
||||||
|
if err := secret.PassInsertSecret(
|
||||||
|
secretValue,
|
||||||
|
secretName,
|
||||||
|
appDomain,
|
||||||
|
newAppServer,
|
||||||
|
); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return secrets, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ensureDomainFlag checks if the domain flag was used. if not, asks the user for it/
|
||||||
|
func ensureDomainFlag(recipe recipePkg.Recipe, server string) error {
|
||||||
|
if appDomain == "" && !internal.NoInput {
|
||||||
|
prompt := &survey.Input{
|
||||||
|
Message: i18n.G("Specify app domain"),
|
||||||
|
Default: fmt.Sprintf("%s.%s", recipe.Name, server),
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(prompt, &appDomain); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if appDomain == "" {
|
||||||
|
return errors.New(i18n.G("no domain provided"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// promptForSecrets asks if we should generate secrets for a new app.
|
||||||
|
func promptForSecrets(recipeName string, secretsConfig map[string]secret.Secret) error {
|
||||||
|
if len(secretsConfig) == 0 {
|
||||||
|
log.Debug(i18n.G("%s has no secrets to generate, skipping...", recipeName))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !generateSecrets && !internal.NoInput {
|
||||||
|
prompt := &survey.Confirm{
|
||||||
|
Message: i18n.G("Generate app secrets?"),
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(prompt, &generateSecrets); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ensureServerFlag checks if the server flag was used. if not, asks the user for it.
|
||||||
|
func ensureServerFlag() error {
|
||||||
|
servers, err := config.GetServers()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(servers) == 1 {
|
||||||
|
newAppServer = servers[0]
|
||||||
|
log.Info(i18n.G("single server detected, choosing %s automatically", newAppServer))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if newAppServer == "" && !internal.NoInput {
|
||||||
|
prompt := &survey.Select{
|
||||||
|
Message: i18n.G("Select app server:"),
|
||||||
|
Options: servers,
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(prompt, &newAppServer); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if newAppServer == "" {
|
||||||
|
return errors.New(i18n.G("no server provided"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
newAppServer string
|
||||||
|
appDomain string
|
||||||
|
saveInPass bool
|
||||||
|
generateSecrets bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppNewCommand.Flags().StringVarP(
|
||||||
|
&newAppServer,
|
||||||
|
i18n.G("server"),
|
||||||
|
i18n.G("s"),
|
||||||
|
"",
|
||||||
|
i18n.G("specify server for new app"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppNewCommand.RegisterFlagCompletionFunc(
|
||||||
|
i18n.G("server"),
|
||||||
|
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.ServerNameComplete()
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
AppNewCommand.Flags().StringVarP(
|
||||||
|
&appDomain,
|
||||||
|
i18n.G("domain"),
|
||||||
|
i18n.G("D"),
|
||||||
|
"",
|
||||||
|
i18n.G("domain name for app"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppNewCommand.Flags().BoolVarP(
|
||||||
|
&saveInPass,
|
||||||
|
i18n.G("pass"),
|
||||||
|
i18n.G("p"),
|
||||||
|
false,
|
||||||
|
i18n.G("store secrets in a local pass store"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppNewCommand.Flags().BoolVarP(
|
||||||
|
&generateSecrets,
|
||||||
|
i18n.G("secrets"),
|
||||||
|
i18n.G("S"),
|
||||||
|
false,
|
||||||
|
i18n.G("automatically generate secrets"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppNewCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
}
|
||||||
+217
@@ -0,0 +1,217 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
abraService "coopcloud.tech/abra/pkg/service"
|
||||||
|
stack "coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
dockerFormatter "github.com/docker/cli/cli/command/formatter"
|
||||||
|
containerTypes "github.com/docker/docker/api/types/container"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app ps` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var appPsAliases = i18n.G("p")
|
||||||
|
|
||||||
|
var AppPsCommand = &cobra.Command{
|
||||||
|
// translators: `app ps` command
|
||||||
|
Use: i18n.G("ps <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appPsAliases, ","),
|
||||||
|
// translators: Short description for `app ps` command
|
||||||
|
Short: i18n.G("Check app deployment status"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s is not deployed?", app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
chaosVersion := config.CHAOS_DEFAULT
|
||||||
|
statuses, err := appPkg.GetAppStatuses([]appPkg.App{app}, true)
|
||||||
|
if statusMeta, ok := statuses[app.StackName()]; ok {
|
||||||
|
if isChaos, exists := statusMeta["chaos"]; exists && isChaos == "true" {
|
||||||
|
if cVersion, exists := statusMeta["chaosVersion"]; exists {
|
||||||
|
chaosVersion = cVersion
|
||||||
|
if strings.HasSuffix(chaosVersion, config.DIRTY_DEFAULT) {
|
||||||
|
chaosVersion = formatter.BoldDirtyDefault(chaosVersion)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
showPSOutput(app, cl, deployMeta.Version, chaosVersion)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// showPSOutput renders ps output.
|
||||||
|
func showPSOutput(app appPkg.App, cl *dockerClient.Client, deployedVersion, chaosVersion string) {
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
deployOpts := stack.Deploy{
|
||||||
|
Composefiles: composeFiles,
|
||||||
|
Namespace: app.StackName(),
|
||||||
|
Prune: false,
|
||||||
|
ResolveImage: stack.ResolveImageAlways,
|
||||||
|
}
|
||||||
|
compose, err := appPkg.GetAppComposeConfig(app.Name, deployOpts, app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
services := compose.Services
|
||||||
|
sort.Slice(services, func(i, j int) bool {
|
||||||
|
return services[i].Name < services[j].Name
|
||||||
|
})
|
||||||
|
|
||||||
|
var rows [][]string
|
||||||
|
allContainerStats := make(map[string]map[string]string)
|
||||||
|
for _, service := range services {
|
||||||
|
filters := filters.NewArgs()
|
||||||
|
filters.Add("name", fmt.Sprintf("^%s_%s", app.StackName(), service.Name))
|
||||||
|
|
||||||
|
containers, err := cl.ContainerList(context.Background(), containerTypes.ListOptions{Filters: filters})
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var containerStats map[string]string
|
||||||
|
if len(containers) == 0 {
|
||||||
|
containerStats = map[string]string{
|
||||||
|
"version": deployedVersion,
|
||||||
|
"chaos": chaosVersion,
|
||||||
|
"service": service.Name,
|
||||||
|
"image": i18n.G("unknown"),
|
||||||
|
"created": i18n.G("unknown"),
|
||||||
|
"status": i18n.G("unknown"),
|
||||||
|
"state": i18n.G("unknown"),
|
||||||
|
"ports": i18n.G("unknown"),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
container := containers[0]
|
||||||
|
containerStats = map[string]string{
|
||||||
|
"version": deployedVersion,
|
||||||
|
"chaos": chaosVersion,
|
||||||
|
"service": abraService.ContainerToServiceName(container.Names, app.StackName()),
|
||||||
|
"image": formatter.RemoveSha(container.Image),
|
||||||
|
"created": formatter.HumanDuration(container.Created),
|
||||||
|
"status": container.Status,
|
||||||
|
"state": container.State,
|
||||||
|
"ports": dockerFormatter.DisplayablePorts(container.Ports),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
allContainerStats[containerStats["service"]] = containerStats
|
||||||
|
|
||||||
|
// NOTE(d1): don't clobber these variables for --machine output
|
||||||
|
dVersion := deployedVersion
|
||||||
|
cVersion := chaosVersion
|
||||||
|
|
||||||
|
if containerStats["service"] != "app" {
|
||||||
|
// NOTE(d1): don't repeat info which only relevant for the "app" service
|
||||||
|
dVersion = ""
|
||||||
|
cVersion = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
row := []string{
|
||||||
|
containerStats["service"],
|
||||||
|
containerStats["status"],
|
||||||
|
containerStats["image"],
|
||||||
|
dVersion,
|
||||||
|
cVersion,
|
||||||
|
}
|
||||||
|
|
||||||
|
rows = append(rows, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.MachineReadable {
|
||||||
|
rendered, err := json.Marshal(allContainerStats)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to convert to JSON: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println(string(rendered))
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{
|
||||||
|
i18n.G("SERVICE"),
|
||||||
|
i18n.G("STATUS"),
|
||||||
|
i18n.G("IMAGE"),
|
||||||
|
i18n.G("VERSION"),
|
||||||
|
i18n.G("CHAOS"),
|
||||||
|
}
|
||||||
|
|
||||||
|
table.
|
||||||
|
Headers(headers...).
|
||||||
|
Rows(rows...)
|
||||||
|
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppPsCommand.Flags().BoolVarP(
|
||||||
|
&internal.MachineReadable,
|
||||||
|
i18n.G("machine"),
|
||||||
|
i18n.G("m"),
|
||||||
|
false,
|
||||||
|
i18n.G("print machine-readable output"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppPsCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,167 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app remove` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appRemoveAliases = i18n.G("rm")
|
||||||
|
|
||||||
|
var AppRemoveCommand = &cobra.Command{
|
||||||
|
// translators: `app remove` command
|
||||||
|
Use: i18n.G("remove <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appRemoveAliases, ","),
|
||||||
|
// translators: Short description for `app remove` command
|
||||||
|
Short: i18n.G("Remove all app data, locally and remotely"),
|
||||||
|
Long: i18n.G(`Remove everything related to an app which is already undeployed.
|
||||||
|
|
||||||
|
By default, it will prompt for confirmation before proceeding. All secrets,
|
||||||
|
volumes and the local app env file will be deleted.
|
||||||
|
|
||||||
|
Only run this command when you are sure you want to completely remove the app
|
||||||
|
and all associated app data. This is a destructive action, Be Careful!
|
||||||
|
|
||||||
|
If you would like to delete specific volumes or secrets, please use removal
|
||||||
|
sub-commands under "app volume" and "app secret" instead.
|
||||||
|
|
||||||
|
Please note, if you delete the local app env file without removing volumes and
|
||||||
|
secrets first, Abra will *not* be able to help you remove them afterwards.
|
||||||
|
|
||||||
|
To delete everything without prompt, use the "--force/-f" or the "--no-input/n"
|
||||||
|
flag.`),
|
||||||
|
Example: i18n.G(" abra app remove 1312.net"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if !internal.Force && !internal.NoInput {
|
||||||
|
log.Warn(i18n.G("ALERTA ALERTA: deleting %s data and config (local/remote)", app.Name))
|
||||||
|
|
||||||
|
response := false
|
||||||
|
prompt := &survey.Confirm{Message: i18n.G("are you sure?")}
|
||||||
|
if err := survey.AskOne(prompt, &response); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !response {
|
||||||
|
log.Fatal(i18n.G("aborting as requested"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s is still deployed. Run \"abra app undeploy %s\"", app.Name, app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
fs, err := app.Filters(false, false)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
configs, err := client.GetConfigs(cl, context.Background(), app.Server, fs)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
configNames := client.GetConfigNames(configs)
|
||||||
|
|
||||||
|
if len(configNames) > 0 {
|
||||||
|
if err := client.RemoveConfigs(cl, context.Background(), configNames, internal.Force); err != nil {
|
||||||
|
log.Fatal(i18n.G("removing configs failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("%d config(s) removed successfully", len(configNames)))
|
||||||
|
} else {
|
||||||
|
log.Info(i18n.G("no configs to remove"))
|
||||||
|
}
|
||||||
|
|
||||||
|
secretList, err := cl.SecretList(context.Background(), types.SecretListOptions{Filters: fs})
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
secrets := make(map[string]string)
|
||||||
|
var secretNames []string
|
||||||
|
|
||||||
|
for _, cont := range secretList {
|
||||||
|
secrets[cont.Spec.Annotations.Name] = cont.ID // we have to map the names to ID's
|
||||||
|
secretNames = append(secretNames, cont.Spec.Annotations.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(secrets) > 0 {
|
||||||
|
for _, name := range secretNames {
|
||||||
|
err := cl.SecretRemove(context.Background(), secrets[name])
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
log.Info(i18n.G("secret: %s removed", name))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.Info(i18n.G("no secrets to remove"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fs, err = app.Filters(false, true)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
volumeList, err := client.GetVolumes(cl, context.Background(), app.Server, fs)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
volumeNames := client.GetVolumeNames(volumeList)
|
||||||
|
|
||||||
|
if len(volumeNames) > 0 {
|
||||||
|
err := client.RemoveVolumes(cl, context.Background(), volumeNames, internal.Force, 5)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("removing volumes failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("%d volume(s) removed successfully", len(volumeNames)))
|
||||||
|
} else {
|
||||||
|
log.Info(i18n.G("no volumes to remove"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = os.Remove(app.Path); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("file: %s removed", app.Path))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppRemoveCommand.Flags().BoolVarP(
|
||||||
|
&internal.Force,
|
||||||
|
i18n.G("force"),
|
||||||
|
i18n.G("f"),
|
||||||
|
false,
|
||||||
|
i18n.G("perform action without further prompt"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,174 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/ui"
|
||||||
|
upstream "coopcloud.tech/abra/pkg/upstream/service"
|
||||||
|
stack "coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app restart` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appRestartAliases = i18n.G("re")
|
||||||
|
|
||||||
|
var AppRestartCommand = &cobra.Command{
|
||||||
|
// translators: `app restart` command
|
||||||
|
Use: i18n.G("restart <domain> [[service] | --all-services] [flags]"),
|
||||||
|
Aliases: strings.Split(appRestartAliases, ","),
|
||||||
|
// translators: Short description for `app restart` command
|
||||||
|
Short: i18n.G("Restart an app"),
|
||||||
|
Long: i18n.G(`This command restarts services within a deployed app.
|
||||||
|
|
||||||
|
Run "abra app ps <domain>" to see a list of service names.
|
||||||
|
|
||||||
|
Pass "--all-services/-a" to restart all services.`),
|
||||||
|
Example: i18n.G(` # restart a single app service
|
||||||
|
abra app restart 1312.net app
|
||||||
|
|
||||||
|
# restart all app services
|
||||||
|
abra app restart 1312.net -a`),
|
||||||
|
Args: cobra.RangeArgs(1, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
if !allServices {
|
||||||
|
return autocomplete.ServiceNameComplete(args[0])
|
||||||
|
}
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var serviceName string
|
||||||
|
if len(args) == 2 {
|
||||||
|
serviceName = args[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
if serviceName == "" && !allServices {
|
||||||
|
log.Fatal(i18n.G("missing [service]"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if serviceName != "" && allServices {
|
||||||
|
log.Fatal(i18n.G("cannot use [service] and --all-services/-a together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
var serviceNames []string
|
||||||
|
if allServices {
|
||||||
|
var err error
|
||||||
|
serviceNames, err = appPkg.GetAppServiceNames(app.Name)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
serviceNames = append(serviceNames, serviceName)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s is not deployed?", app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, serviceName := range serviceNames {
|
||||||
|
stackServiceName := fmt.Sprintf("%s_%s", app.StackName(), serviceName)
|
||||||
|
|
||||||
|
service, _, err := cl.ServiceInspectWithRaw(
|
||||||
|
context.Background(),
|
||||||
|
stackServiceName,
|
||||||
|
types.ServiceInspectOptions{},
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("attempting to scale %s to 0", stackServiceName))
|
||||||
|
|
||||||
|
if err := upstream.RunServiceScale(context.Background(), cl, stackServiceName, 0); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := app.Filters(true, false, serviceName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
waitOpts := stack.WaitOpts{
|
||||||
|
Services: []ui.ServiceMeta{{Name: stackServiceName, ID: service.ID}},
|
||||||
|
AppName: app.Name,
|
||||||
|
ServerName: app.Server,
|
||||||
|
Filters: f,
|
||||||
|
NoInput: internal.NoInput,
|
||||||
|
NoLog: true,
|
||||||
|
Quiet: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := stack.WaitOnServices(cmd.Context(), cl, waitOpts); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("%s has been scaled to 0", stackServiceName))
|
||||||
|
log.Debug(i18n.G("attempting to scale %s to 1", stackServiceName))
|
||||||
|
|
||||||
|
if err := upstream.RunServiceScale(context.Background(), cl, stackServiceName, 1); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := stack.WaitOnServices(cmd.Context(), cl, waitOpts); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("%s has been scaled to 1", stackServiceName))
|
||||||
|
log.Info(i18n.G("%s service successfully restarted", serviceName))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var allServices bool
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppRestartCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
AppRestartCommand.Flags().BoolVarP(
|
||||||
|
&allServices,
|
||||||
|
i18n.G("all-services"),
|
||||||
|
i18n.GC("a", "app restart"),
|
||||||
|
false,
|
||||||
|
i18n.G("restart all services"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,142 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app restore` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var appRestoreAliases = i18n.G("rs")
|
||||||
|
|
||||||
|
var AppRestoreCommand = &cobra.Command{
|
||||||
|
// translators: `app restore` command
|
||||||
|
Use: i18n.G("restore <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appRestoreAliases, ","),
|
||||||
|
// translators: Short description for `app restore` command
|
||||||
|
Short: i18n.G("Restore a snapshot"),
|
||||||
|
Long: i18n.G(`Snapshots are restored while apps are deployed.
|
||||||
|
|
||||||
|
Some restore scenarios may require service / app restarts.`),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
targetContainer, err := internal.RetrieveBackupBotContainer(cl)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
execEnv := []string{
|
||||||
|
fmt.Sprintf("SERVICE=%s", app.Domain),
|
||||||
|
"MACHINE_LOGS=true",
|
||||||
|
}
|
||||||
|
|
||||||
|
if snapshot != "" {
|
||||||
|
log.Debug(i18n.G("including SNAPSHOT=%s in backupbot exec invocation", snapshot))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("SNAPSHOT=%s", snapshot))
|
||||||
|
}
|
||||||
|
|
||||||
|
if targetPath != "" {
|
||||||
|
log.Debug(i18n.G("including TARGET=%s in backupbot exec invocation", targetPath))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("TARGET=%s", targetPath))
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.NoInput {
|
||||||
|
log.Debug(i18n.G("including NONINTERACTIVE=%v in backupbot exec invocation", internal.NoInput))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("NONINTERACTIVE=%v", internal.NoInput))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(volumes) > 0 {
|
||||||
|
allVolumes := strings.Join(volumes, ",")
|
||||||
|
log.Debug(i18n.G("including VOLUMES=%s in backupbot exec invocation", allVolumes))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("VOLUMES=%s", allVolumes))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(services) > 0 {
|
||||||
|
allServices := strings.Join(services, ",")
|
||||||
|
log.Debug(i18n.G("including CONTAINER=%s in backupbot exec invocation", allServices))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("CONTAINER=%s", allServices))
|
||||||
|
}
|
||||||
|
|
||||||
|
if hooks {
|
||||||
|
log.Debug(i18n.G("including NO_COMMANDS=%v in backupbot exec invocation", false))
|
||||||
|
execEnv = append(execEnv, fmt.Sprintf("NO_COMMANDS=%v", false))
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := internal.RunBackupCmdRemote(cl, "restore", targetContainer.ID, execEnv); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
targetPath string
|
||||||
|
hooks bool
|
||||||
|
services []string
|
||||||
|
volumes []string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppRestoreCommand.Flags().StringVarP(
|
||||||
|
&targetPath,
|
||||||
|
i18n.G("target"),
|
||||||
|
i18n.G("t"),
|
||||||
|
"/",
|
||||||
|
i18n.G("target path"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppRestoreCommand.Flags().StringArrayVarP(
|
||||||
|
&services,
|
||||||
|
i18n.G("services"),
|
||||||
|
i18n.G("s"),
|
||||||
|
[]string{},
|
||||||
|
i18n.G("restore specific services"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppRestoreCommand.Flags().StringArrayVarP(
|
||||||
|
&volumes,
|
||||||
|
i18n.G("volumes"),
|
||||||
|
i18n.G("v"),
|
||||||
|
[]string{},
|
||||||
|
i18n.G("restore specific volumes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppRestoreCommand.Flags().BoolVarP(
|
||||||
|
&hooks,
|
||||||
|
i18n.G("hooks"),
|
||||||
|
i18n.G("H"),
|
||||||
|
false,
|
||||||
|
i18n.G("enable pre/post-hook command execution"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppRestoreCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,381 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/deploy"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/lint"
|
||||||
|
stack "coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"coopcloud.tech/tagcmp"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app rollback` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var appRollbackAliases = i18n.G("rl")
|
||||||
|
|
||||||
|
var AppRollbackCommand = &cobra.Command{
|
||||||
|
// translators: `app rollback` command
|
||||||
|
Use: i18n.G("rollback <domain> [version] [flags]"),
|
||||||
|
Aliases: strings.Split(appRollbackAliases, ","),
|
||||||
|
// translators: Short description for `app rollback` command
|
||||||
|
Short: i18n.G("Roll an app back to a previous version"),
|
||||||
|
Long: i18n.G(`This command rolls an app back to a previous version.
|
||||||
|
|
||||||
|
Unlike "abra app deploy", chaos operations are not supported here. Only recipe
|
||||||
|
versions are supported values for "[version]".
|
||||||
|
|
||||||
|
It is possible to "--force/-f" an downgrade if you want to re-deploy a specific
|
||||||
|
version.
|
||||||
|
|
||||||
|
Only the deployed version is consulted when trying to determine what downgrades
|
||||||
|
are available. The live deployment version is the "source of truth" in this
|
||||||
|
case. The stored .env version is not consulted.
|
||||||
|
|
||||||
|
A downgrade can be destructive, please ensure you have a copy of your app data
|
||||||
|
beforehand. See "abra app backup" for more.`),
|
||||||
|
Example: i18n.G(` # standard rollback
|
||||||
|
abra app rollback 1312.net
|
||||||
|
|
||||||
|
# rollback to specific version
|
||||||
|
abra app rollback 1312.net 2.0.0+1.2.3`),
|
||||||
|
Args: cobra.RangeArgs(1, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
app, err := appPkg.Get(args[0])
|
||||||
|
if err != nil {
|
||||||
|
return []string{i18n.G("autocomplete failed: %s", err)}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
return autocomplete.RecipeVersionComplete(app.Recipe.Name)
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
var (
|
||||||
|
downgradeWarnMessages []string
|
||||||
|
chosenDowngrade string
|
||||||
|
availableDowngrades []string
|
||||||
|
)
|
||||||
|
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := ensureDeployed(cl, app)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := lint.LintForErrors(app.Recipe); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
versions, err := app.Recipe.Tags()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE(d1): we've no idea what the live deployment version is, so every
|
||||||
|
// possible downgrade can be shown. it's up to the user to make the choice
|
||||||
|
if deployMeta.Version == config.UNKNOWN_DEFAULT {
|
||||||
|
availableDowngrades = versions
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) == 2 && args[1] != "" {
|
||||||
|
chosenDowngrade = args[1]
|
||||||
|
|
||||||
|
if err := validateDowngradeVersionArg(chosenDowngrade, app, deployMeta); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
availableDowngrades = append(availableDowngrades, chosenDowngrade)
|
||||||
|
}
|
||||||
|
|
||||||
|
if deployMeta.Version != config.UNKNOWN_DEFAULT && chosenDowngrade == "" {
|
||||||
|
downgradeAvailable, err := ensureDowngradesAvailable(versions, &availableDowngrades, deployMeta)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !downgradeAvailable {
|
||||||
|
log.Info(i18n.G("no available downgrades"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.Force || internal.NoInput || chosenDowngrade != "" {
|
||||||
|
if len(availableDowngrades) > 0 {
|
||||||
|
chosenDowngrade = availableDowngrades[len(availableDowngrades)-1]
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := chooseDowngrade(availableDowngrades, deployMeta, &chosenDowngrade); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.Force &&
|
||||||
|
chosenDowngrade == "" &&
|
||||||
|
deployMeta.Version != config.UNKNOWN_DEFAULT {
|
||||||
|
chosenDowngrade = deployMeta.Version
|
||||||
|
}
|
||||||
|
|
||||||
|
if chosenDowngrade == "" {
|
||||||
|
log.Fatal(i18n.G("unknown deployed version, unable to downgrade"))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("choosing %s as version to rollback", chosenDowngrade))
|
||||||
|
|
||||||
|
if _, err := app.Recipe.EnsureVersion(chosenDowngrade); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := deploy.MergeAbraShEnv(app.Recipe, app.Env); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
stackName := app.StackName()
|
||||||
|
deployOpts := stack.Deploy{
|
||||||
|
Composefiles: composeFiles,
|
||||||
|
Namespace: stackName,
|
||||||
|
Prune: false,
|
||||||
|
ResolveImage: stack.ResolveImageAlways,
|
||||||
|
Detach: false,
|
||||||
|
}
|
||||||
|
|
||||||
|
compose, err := appPkg.GetAppComposeConfig(app.Name, deployOpts, app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
newRecipeWithDowngradeVersion := fmt.Sprintf("%s:%s", app.Recipe.Name, chosenDowngrade)
|
||||||
|
appPkg.ExposeAllEnv(stackName, compose, app.Env, newRecipeWithDowngradeVersion)
|
||||||
|
|
||||||
|
appPkg.SetRecipeLabel(compose, stackName, app.Recipe.Name)
|
||||||
|
appPkg.SetChaosLabel(compose, stackName, internal.Chaos)
|
||||||
|
if internal.Chaos {
|
||||||
|
appPkg.SetChaosVersionLabel(compose, stackName, chosenDowngrade)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather secrets
|
||||||
|
secretInfo, err := deploy.GatherSecretsForDeploy(cl, app, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather configs
|
||||||
|
configInfo, err := deploy.GatherConfigsForDeploy(cl, app, compose, app.Env, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather images
|
||||||
|
imageInfo, err := deploy.GatherImagesForDeploy(cl, app, compose, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployedVersion := deployMeta.Version
|
||||||
|
if deployMeta.IsChaos {
|
||||||
|
deployedVersion = deployMeta.ChaosVersion
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE(d1): no release notes implemeneted for rolling back
|
||||||
|
if err := internal.DeployOverview(
|
||||||
|
app,
|
||||||
|
deployedVersion,
|
||||||
|
chosenDowngrade,
|
||||||
|
"",
|
||||||
|
downgradeWarnMessages,
|
||||||
|
secretInfo,
|
||||||
|
configInfo,
|
||||||
|
imageInfo,
|
||||||
|
); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
stack.WaitTimeout, err = appPkg.GetTimeoutFromLabel(compose, stackName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
serviceNames, err := appPkg.GetAppServiceNames(app.Name)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := app.Filters(true, false, serviceNames...)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := stack.RunDeploy(
|
||||||
|
cl,
|
||||||
|
deployOpts,
|
||||||
|
compose,
|
||||||
|
stackName,
|
||||||
|
app.Server,
|
||||||
|
internal.DontWaitConverge,
|
||||||
|
internal.NoInput,
|
||||||
|
f,
|
||||||
|
); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := app.WriteRecipeVersion(chosenDowngrade, false); err != nil {
|
||||||
|
log.Fatal(i18n.G("writing recipe version failed: %s", err))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// chooseDowngrade prompts the user to choose an downgrade interactively.
|
||||||
|
func chooseDowngrade(
|
||||||
|
availableDowngrades []string,
|
||||||
|
deployMeta stack.DeployMeta,
|
||||||
|
chosenDowngrade *string,
|
||||||
|
) error {
|
||||||
|
msg := i18n.G("please select a downgrade (version: %s):", deployMeta.Version)
|
||||||
|
|
||||||
|
if deployMeta.IsChaos {
|
||||||
|
chaosVersion := formatter.BoldDirtyDefault(deployMeta.ChaosVersion)
|
||||||
|
|
||||||
|
msg = i18n.G(
|
||||||
|
"please select a downgrade (version: %s, chaos: %s):",
|
||||||
|
deployMeta.Version,
|
||||||
|
chaosVersion,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
prompt := &survey.Select{
|
||||||
|
Message: msg,
|
||||||
|
Options: internal.SortVersionsDesc(availableDowngrades),
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := survey.AskOne(prompt, chosenDowngrade); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// validateDownpgradeVersionArg validates the specific version.
|
||||||
|
func validateDowngradeVersionArg(
|
||||||
|
specificVersion string,
|
||||||
|
app appPkg.App,
|
||||||
|
deployMeta stack.DeployMeta,
|
||||||
|
) error {
|
||||||
|
parsedDeployedVersion, err := tagcmp.Parse(deployMeta.Version)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("current deployment '%s' is not a known version for %s", deployMeta.Version, app.Recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
parsedSpecificVersion, err := tagcmp.Parse(specificVersion)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("'%s' is not a known version for %s", specificVersion, app.Recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
if parsedSpecificVersion.IsGreaterThan(parsedDeployedVersion) &&
|
||||||
|
!parsedSpecificVersion.Equals(parsedDeployedVersion) {
|
||||||
|
return errors.New(i18n.G("%s is not a downgrade for %s?", deployMeta.Version, specificVersion))
|
||||||
|
}
|
||||||
|
|
||||||
|
if parsedSpecificVersion.Equals(parsedDeployedVersion) && !internal.Force {
|
||||||
|
return errors.New(i18n.G("%s is not a downgrade for %s?", deployMeta.Version, specificVersion))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ensureDowngradesAvailable ensures that there are available downgrades.
|
||||||
|
func ensureDowngradesAvailable(
|
||||||
|
versions []string,
|
||||||
|
availableDowngrades *[]string,
|
||||||
|
deployMeta stack.DeployMeta,
|
||||||
|
) (bool, error) {
|
||||||
|
parsedDeployedVersion, err := tagcmp.Parse(deployMeta.Version)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, version := range versions {
|
||||||
|
parsedVersion, err := tagcmp.Parse(version)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if parsedVersion.IsLessThan(parsedDeployedVersion) &&
|
||||||
|
!(parsedVersion.Equals(parsedDeployedVersion)) {
|
||||||
|
*availableDowngrades = append(*availableDowngrades, version)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(*availableDowngrades) == 0 && !internal.Force {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppRollbackCommand.Flags().BoolVarP(
|
||||||
|
&internal.Force,
|
||||||
|
i18n.G("force"),
|
||||||
|
i18n.G("f"),
|
||||||
|
false,
|
||||||
|
i18n.G("perform action without further prompt"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppRollbackCommand.Flags().BoolVarP(
|
||||||
|
&internal.NoDomainChecks,
|
||||||
|
i18n.G("no-domain-checks"),
|
||||||
|
i18n.G("D"),
|
||||||
|
false,
|
||||||
|
i18n.G("disable public DNS checks"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppRollbackCommand.Flags().BoolVarP(
|
||||||
|
&internal.DontWaitConverge,
|
||||||
|
i18n.G("no-converge-checks"),
|
||||||
|
i18n.G("c"),
|
||||||
|
false,
|
||||||
|
i18n.G("disable converge logic checks"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppRollbackCommand.Flags().BoolVarP(
|
||||||
|
&internal.ShowUnchanged,
|
||||||
|
i18n.G("show-unchanged"),
|
||||||
|
i18n.G("U"),
|
||||||
|
false,
|
||||||
|
i18n.G("show all configs & images, including unchanged ones"),
|
||||||
|
)
|
||||||
|
}
|
||||||
+122
@@ -0,0 +1,122 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
containerPkg "coopcloud.tech/abra/pkg/container"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/container"
|
||||||
|
"github.com/docker/cli/cli/command"
|
||||||
|
containertypes "github.com/docker/docker/api/types/container"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app run` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var appRunAliases = i18n.G("r")
|
||||||
|
|
||||||
|
var AppRunCommand = &cobra.Command{
|
||||||
|
// translators: `app run` command
|
||||||
|
Use: i18n.G("run <domain> <service> <cmd> [[args] [flags] | [flags] -- [args]]"),
|
||||||
|
Aliases: strings.Split(appRunAliases, ","),
|
||||||
|
// translators: Short description for `app run` command
|
||||||
|
Short: i18n.G("Run a command inside a service container"),
|
||||||
|
Example: i18n.G(` # run <cmd> with args/flags
|
||||||
|
abra app run 1312.net app -- ls -lha
|
||||||
|
|
||||||
|
# run <cmd> without args/flags
|
||||||
|
abra app run 1312.net app bash --user nobody
|
||||||
|
|
||||||
|
# run <cmd> with both kinds of args/flags
|
||||||
|
abra app run 1312.net app --user nobody -- ls -lha`),
|
||||||
|
Args: cobra.MinimumNArgs(3),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
return autocomplete.ServiceNameComplete(args[0])
|
||||||
|
case 2:
|
||||||
|
return autocomplete.CommandNameComplete(args[0])
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
serviceName := args[1]
|
||||||
|
stackAndServiceName := fmt.Sprintf("^%s_%s", app.StackName(), serviceName)
|
||||||
|
|
||||||
|
filters := filters.NewArgs()
|
||||||
|
filters.Add("name", stackAndServiceName)
|
||||||
|
|
||||||
|
targetContainer, err := containerPkg.GetContainer(context.Background(), cl, filters, false)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
userCmd := args[2:]
|
||||||
|
execCreateOpts := containertypes.ExecOptions{
|
||||||
|
AttachStderr: true,
|
||||||
|
AttachStdin: true,
|
||||||
|
AttachStdout: true,
|
||||||
|
Cmd: userCmd,
|
||||||
|
Detach: false,
|
||||||
|
Tty: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
if runAsUser != "" {
|
||||||
|
execCreateOpts.User = runAsUser
|
||||||
|
}
|
||||||
|
if noTTY {
|
||||||
|
execCreateOpts.Tty = false
|
||||||
|
}
|
||||||
|
|
||||||
|
dcli, err := command.NewDockerCli()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
noTTY bool
|
||||||
|
runAsUser string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppRunCommand.Flags().BoolVarP(&noTTY,
|
||||||
|
i18n.G("no-tty"),
|
||||||
|
i18n.G("t"),
|
||||||
|
false,
|
||||||
|
i18n.G("do not request a TTY"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppRunCommand.Flags().StringVarP(
|
||||||
|
&runAsUser,
|
||||||
|
i18n.G("user"),
|
||||||
|
i18n.G("u"),
|
||||||
|
"",
|
||||||
|
i18n.G("run command as user"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,653 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"sort"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/secret"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app secret generate` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appSecretGenerateAliases = i18n.G("g")
|
||||||
|
|
||||||
|
var AppSecretGenerateCommand = &cobra.Command{
|
||||||
|
// translators: `app secret generate` command
|
||||||
|
Use: i18n.G("generate <domain> [[secret] [version] | --all] [flags]"),
|
||||||
|
Aliases: strings.Split(appSecretGenerateAliases, ","),
|
||||||
|
// translators: Short description for `app secret generate` command
|
||||||
|
Short: i18n.G("Generate secrets"),
|
||||||
|
Args: cobra.RangeArgs(1, 3),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
app, err := appPkg.Get(args[0])
|
||||||
|
if err != nil {
|
||||||
|
return []string{i18n.G("autocomplete failed: %s", err)}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
return autocomplete.SecretComplete(app.Recipe.Name)
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) <= 2 && !generateAllSecrets {
|
||||||
|
log.Fatal(i18n.G("missing arguments [secret]/[version] or '--all'"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) > 2 && generateAllSecrets {
|
||||||
|
log.Fatal(i18n.G("cannot use '[secret] [version]' and '--all' together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
secrets, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !generateAllSecrets {
|
||||||
|
secretName := args[1]
|
||||||
|
secretVersion := args[2]
|
||||||
|
s, ok := secrets[secretName]
|
||||||
|
if !ok {
|
||||||
|
log.Fatal(i18n.G("%s doesn't exist in the env config?", secretName))
|
||||||
|
}
|
||||||
|
s.Version = secretVersion
|
||||||
|
secrets = map[string]secret.Secret{
|
||||||
|
secretName: s,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
secretVals, err := secret.GenerateSecrets(cl, secrets, app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if storeInPass {
|
||||||
|
for name, data := range secretVals {
|
||||||
|
if err := secret.PassInsertSecret(data, name, app.Name, app.Server); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(secretVals) == 0 {
|
||||||
|
log.Warn(i18n.G("no secrets generated"))
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{i18n.G("NAME"), i18n.G("VALUE")}
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Headers(headers...)
|
||||||
|
|
||||||
|
var rows [][]string
|
||||||
|
for name, val := range secretVals {
|
||||||
|
row := []string{name, val}
|
||||||
|
rows = append(rows, row)
|
||||||
|
table.Row(row...)
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.MachineReadable {
|
||||||
|
out, err := formatter.ToJSON(headers, rows)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to render to JSON: %s", err))
|
||||||
|
}
|
||||||
|
fmt.Println(out)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Warn(i18n.G(
|
||||||
|
"generated secrets %s shown again, please take note of them %s",
|
||||||
|
formatter.BoldStyle.Render(i18n.G("NOT")),
|
||||||
|
formatter.BoldStyle.Render(i18n.G("NOW")),
|
||||||
|
))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app secret insert` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appSecretInsertAliases = i18n.G("i")
|
||||||
|
|
||||||
|
var AppSecretInsertCommand = &cobra.Command{
|
||||||
|
// translators: `app secret insert` command
|
||||||
|
Use: i18n.G("insert <domain> <secret> <version> [<data>] [flags]"),
|
||||||
|
Aliases: strings.Split(appSecretInsertAliases, ","),
|
||||||
|
// translators: Short description for `app secret insert` command
|
||||||
|
Short: i18n.G("Insert secret"),
|
||||||
|
Long: i18n.G(`This command inserts a secret into an app environment.
|
||||||
|
|
||||||
|
Arbitrary secret insertion is not supported. Secrets that are inserted must
|
||||||
|
match those configured in the recipe beforehand.
|
||||||
|
|
||||||
|
This command can be useful when you want to manually generate secrets for an app
|
||||||
|
environment. Typically, you can let Abra generate them for you on app creation
|
||||||
|
(see "abra app new --secrets/-S" for more).`),
|
||||||
|
Example: i18n.G(` # insert regular secret
|
||||||
|
abra app secret insert 1312.net my_secret v1 mySuperSecret
|
||||||
|
|
||||||
|
# insert secret as file
|
||||||
|
abra app secret insert 1312.net my_secret v1 secret.txt -f
|
||||||
|
|
||||||
|
# insert secret from stdin
|
||||||
|
echo "mmySuperSecret" | abra app secret insert 1312.net my_secret v1`),
|
||||||
|
Args: cobra.MinimumNArgs(3),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
app, err := appPkg.Get(args[0])
|
||||||
|
if err != nil {
|
||||||
|
return []string{i18n.G("autocomplete failed: %s", err)}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
return autocomplete.SecretComplete(app.Recipe.Name)
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
name := args[1]
|
||||||
|
version := args[2]
|
||||||
|
data, err := readSecretData(args)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
secrets, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var isRecipeSecret bool
|
||||||
|
for secretName := range secrets {
|
||||||
|
if secretName == name {
|
||||||
|
isRecipeSecret = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !isRecipeSecret {
|
||||||
|
log.Fatal(i18n.G("no secret %s available for recipe %s?", name, app.Recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
if insertFromFile {
|
||||||
|
raw, err := os.ReadFile(data)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("reading secret from file: %s", err))
|
||||||
|
}
|
||||||
|
data = string(raw)
|
||||||
|
}
|
||||||
|
|
||||||
|
if trimInput {
|
||||||
|
data = strings.TrimSpace(data)
|
||||||
|
}
|
||||||
|
|
||||||
|
secretName := fmt.Sprintf("%s_%s_%s", app.StackName(), name, version)
|
||||||
|
if err := client.StoreSecret(cl, secretName, data); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("%s successfully stored on server", secretName))
|
||||||
|
|
||||||
|
if storeInPass {
|
||||||
|
if err := secret.PassInsertSecret(data, name, app.Name, app.Server); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func readSecretData(args []string) (string, error) {
|
||||||
|
if len(args) == 4 {
|
||||||
|
return args[3], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) != 3 {
|
||||||
|
return "", errors.New(i18n.G("need 3 or 4 arguments"))
|
||||||
|
}
|
||||||
|
// First check if data is provided by stdin
|
||||||
|
fi, err := os.Stdin.Stat()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if fi.Mode()&os.ModeNamedPipe != 0 {
|
||||||
|
// Can't insert from stdin and read from file
|
||||||
|
if insertFromFile {
|
||||||
|
return "", errors.New(i18n.G("can not insert from file and read from stdin"))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("reading secret data from stdin"))
|
||||||
|
bytes, err := io.ReadAll(os.Stdin)
|
||||||
|
if err != nil {
|
||||||
|
return "", errors.New(i18n.G("reading data from stdin: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
return string(bytes), nil
|
||||||
|
}
|
||||||
|
if internal.NoInput {
|
||||||
|
return "", errors.New(i18n.G("must provide <data> argument if --no-input is passed"))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("secret data not provided on command-line or stdin, prompting"))
|
||||||
|
var prompt survey.Prompt
|
||||||
|
if !insertFromFile {
|
||||||
|
prompt = &survey.Password{
|
||||||
|
Message: i18n.G("specify secret value"),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
prompt = &survey.Input{
|
||||||
|
Message: i18n.G("specify secret file"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var data string
|
||||||
|
if err := survey.AskOne(prompt, &data); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return data, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// secretRm removes a secret.
|
||||||
|
func secretRm(cl *dockerClient.Client, app appPkg.App, secretName, parsed string) error {
|
||||||
|
if err := cl.SecretRemove(context.Background(), secretName); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("deleted %s successfully from server", secretName))
|
||||||
|
|
||||||
|
if removeFromPass {
|
||||||
|
if err := secret.PassRmSecret(parsed, app.StackName(), app.Server); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("deleted %s successfully from local pass store", secretName))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app secret remove` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appSecretRemoveAliases = i18n.G("rm")
|
||||||
|
|
||||||
|
var AppSecretRmCommand = &cobra.Command{
|
||||||
|
// translators: `app secret remove` command
|
||||||
|
Use: i18n.G("remove <domain> [[secret] | --all] [flags]"),
|
||||||
|
Aliases: strings.Split(appSecretRemoveAliases, ","),
|
||||||
|
// translators: Short description for `app secret remove` command
|
||||||
|
Short: i18n.G("Remove a secret"),
|
||||||
|
Long: i18n.G(`This command removes a secret from an app environment.
|
||||||
|
|
||||||
|
Arbitrary secret removal is not supported. Secrets that are removed must
|
||||||
|
match those configured in the recipe beforehand.`),
|
||||||
|
Example: i18n.G(" abra app secret rm 1312.net oauth_key"),
|
||||||
|
Args: cobra.RangeArgs(1, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
if !rmAllSecrets {
|
||||||
|
app, err := appPkg.Get(args[0])
|
||||||
|
if err != nil {
|
||||||
|
return []string{i18n.G("autocomplete failed: %s", err)}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
return autocomplete.SecretComplete(app.Recipe.Name)
|
||||||
|
}
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
secrets, err := secret.ReadSecretsConfig(app.Path, composeFiles, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) == 2 && rmAllSecrets {
|
||||||
|
log.Fatal(i18n.G("cannot use [secret] and --all/-a together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) != 2 && !rmAllSecrets {
|
||||||
|
log.Fatal(i18n.G("no secret(s) specified?"))
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
filters, err := app.Filters(false, false)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
secretList, err := cl.SecretList(context.Background(), types.SecretListOptions{Filters: filters})
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
remoteSecretNames := make(map[string]bool)
|
||||||
|
for _, cont := range secretList {
|
||||||
|
remoteSecretNames[cont.Spec.Annotations.Name] = true
|
||||||
|
}
|
||||||
|
|
||||||
|
var secretToRm string
|
||||||
|
if len(args) == 2 {
|
||||||
|
secretToRm = args[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
match := false
|
||||||
|
for secretName, val := range secrets {
|
||||||
|
secretRemoteName := fmt.Sprintf("%s_%s_%s", app.StackName(), secretName, val.Version)
|
||||||
|
if _, ok := remoteSecretNames[secretRemoteName]; ok {
|
||||||
|
if secretToRm != "" {
|
||||||
|
if secretName == secretToRm {
|
||||||
|
if err := secretRm(cl, app, secretRemoteName, secretName); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
match = true
|
||||||
|
|
||||||
|
if err := secretRm(cl, app, secretRemoteName, secretName); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !match && secretToRm != "" {
|
||||||
|
log.Fatal(i18n.G("%s doesn't exist on server?", secretToRm))
|
||||||
|
}
|
||||||
|
|
||||||
|
if !match {
|
||||||
|
log.Fatal(i18n.G("no secrets to remove?"))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app secret ls` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appSecretLsAliases = i18n.G("ls")
|
||||||
|
|
||||||
|
var AppSecretLsCommand = &cobra.Command{
|
||||||
|
// translators: `app secret list` command
|
||||||
|
Use: i18n.G("list <domain>"),
|
||||||
|
Aliases: strings.Split(appSecretLsAliases, ","),
|
||||||
|
// translators: Short description for `app secret list` command
|
||||||
|
Short: i18n.G("List all secrets"),
|
||||||
|
Args: cobra.MinimumNArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{i18n.G("NAME"), i18n.G("VERSION"), i18n.G("GENERATED NAME"), i18n.G("CREATED ON SERVER")}
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Headers(headers...)
|
||||||
|
|
||||||
|
secStats, err := secret.PollSecretsStatus(cl, app)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sort secrets to ensure reproducible output
|
||||||
|
sort.Slice(secStats, func(i, j int) bool {
|
||||||
|
return secStats[i].LocalName < secStats[j].LocalName
|
||||||
|
})
|
||||||
|
var rows [][]string
|
||||||
|
for _, secStat := range secStats {
|
||||||
|
row := []string{
|
||||||
|
secStat.LocalName,
|
||||||
|
secStat.Version,
|
||||||
|
secStat.RemoteName,
|
||||||
|
strconv.FormatBool(secStat.CreatedOnRemote),
|
||||||
|
}
|
||||||
|
|
||||||
|
rows = append(rows, row)
|
||||||
|
table.Row(row...)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(rows) > 0 {
|
||||||
|
if internal.MachineReadable {
|
||||||
|
out, err := formatter.ToJSON(headers, rows)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to render to JSON: %s", err))
|
||||||
|
}
|
||||||
|
fmt.Println(out)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Warn(i18n.G("no secrets stored for %s", app.Name))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var AppSecretCommand = &cobra.Command{
|
||||||
|
// translators: `app secret` command group
|
||||||
|
Use: i18n.G("secret [cmd] [args] [flags]"),
|
||||||
|
Aliases: []string{i18n.G("s")},
|
||||||
|
// translators: Short description for `app secret` command group
|
||||||
|
Short: i18n.G("Manage app secrets"),
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
storeInPass bool
|
||||||
|
insertFromFile bool
|
||||||
|
trimInput bool
|
||||||
|
rmAllSecrets bool
|
||||||
|
generateAllSecrets bool
|
||||||
|
removeFromPass bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppSecretGenerateCommand.Flags().BoolVarP(
|
||||||
|
&internal.MachineReadable,
|
||||||
|
i18n.G("machine"),
|
||||||
|
i18n.G("m"),
|
||||||
|
false,
|
||||||
|
i18n.G("print machine-readable output"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretGenerateCommand.Flags().BoolVarP(
|
||||||
|
&storeInPass,
|
||||||
|
i18n.G("pass"),
|
||||||
|
i18n.G("p"),
|
||||||
|
false,
|
||||||
|
i18n.G("store generated secrets in a local pass store"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretGenerateCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretGenerateCommand.Flags().BoolVarP(
|
||||||
|
&generateAllSecrets,
|
||||||
|
i18n.G("all"),
|
||||||
|
i18n.GC("a", "app secret generate"),
|
||||||
|
false,
|
||||||
|
i18n.G("generate all secrets"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretInsertCommand.Flags().BoolVarP(
|
||||||
|
&storeInPass,
|
||||||
|
i18n.G("pass"),
|
||||||
|
i18n.G("p"),
|
||||||
|
false,
|
||||||
|
i18n.G("store generated secrets in a local pass store"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretInsertCommand.Flags().BoolVarP(
|
||||||
|
&insertFromFile,
|
||||||
|
i18n.G("file"),
|
||||||
|
i18n.G("f"),
|
||||||
|
false,
|
||||||
|
i18n.G("treat input as a file"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretInsertCommand.Flags().BoolVarP(
|
||||||
|
&trimInput,
|
||||||
|
i18n.G("trim"),
|
||||||
|
i18n.G("t"),
|
||||||
|
false,
|
||||||
|
i18n.G("trim input"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretInsertCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretRmCommand.Flags().BoolVarP(
|
||||||
|
&rmAllSecrets,
|
||||||
|
i18n.G("all"),
|
||||||
|
i18n.GC("a", "app secret rm"),
|
||||||
|
false,
|
||||||
|
i18n.G("remove all secrets"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretRmCommand.Flags().BoolVarP(
|
||||||
|
&removeFromPass,
|
||||||
|
i18n.G("pass"),
|
||||||
|
i18n.G("p"),
|
||||||
|
false,
|
||||||
|
i18n.G("remove generated secrets from a local pass store"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretRmCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretLsCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppSecretLsCommand.Flags().BoolVarP(
|
||||||
|
&internal.MachineReadable,
|
||||||
|
i18n.G("machine"),
|
||||||
|
i18n.G("m"),
|
||||||
|
false,
|
||||||
|
i18n.G("print machine-readable output"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,103 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/service"
|
||||||
|
stack "coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
containerTypes "github.com/docker/docker/api/types/container"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app services` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var appServicesAliases = i18n.G("sr")
|
||||||
|
|
||||||
|
var AppServicesCommand = &cobra.Command{
|
||||||
|
// translators: `app services` command
|
||||||
|
Use: i18n.G("services <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appServicesAliases, ","),
|
||||||
|
// translators: Short description for `app services` command
|
||||||
|
Short: i18n.G("Display all services of an app"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s is not deployed?", app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
filters, err := app.Filters(true, true)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
containers, err := cl.ContainerList(context.Background(), containerTypes.ListOptions{Filters: filters})
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{i18n.G("SERVICE (SHORT)"), i18n.G("SERVICE (LONG)")}
|
||||||
|
table.Headers(headers...)
|
||||||
|
|
||||||
|
var rows [][]string
|
||||||
|
for _, container := range containers {
|
||||||
|
var containerNames []string
|
||||||
|
for _, containerName := range container.Names {
|
||||||
|
trimmed := strings.TrimPrefix(containerName, "/")
|
||||||
|
containerNames = append(containerNames, trimmed)
|
||||||
|
}
|
||||||
|
|
||||||
|
serviceShortName := service.ContainerToServiceName(container.Names, app.StackName())
|
||||||
|
serviceLongName := fmt.Sprintf("%s_%s", app.StackName(), serviceShortName)
|
||||||
|
|
||||||
|
row := []string{
|
||||||
|
serviceShortName,
|
||||||
|
serviceLongName,
|
||||||
|
}
|
||||||
|
|
||||||
|
rows = append(rows, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Rows(rows...)
|
||||||
|
|
||||||
|
if len(rows) > 0 {
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
@@ -0,0 +1,173 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
stack "coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app undeploy` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appUndeployAliases = i18n.G("un")
|
||||||
|
|
||||||
|
var AppUndeployCommand = &cobra.Command{
|
||||||
|
// translators: `app undeploy` command
|
||||||
|
Use: i18n.G("undeploy <domain> [flags]"),
|
||||||
|
// translators: Short description for `app undeploy` command
|
||||||
|
Aliases: strings.Split(appUndeployAliases, ","),
|
||||||
|
Short: i18n.G("Undeploy a deployed app"),
|
||||||
|
Long: i18n.G(`This does not destroy any application data.
|
||||||
|
|
||||||
|
However, you should remain vigilant, as your swarm installation will consider
|
||||||
|
any previously attached volumes as eligible for pruning once undeployed.
|
||||||
|
|
||||||
|
Passing "--prune/-p" does not remove those volumes.`),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
stackName := app.StackName()
|
||||||
|
|
||||||
|
if err := app.Recipe.EnsureExists(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("checking whether %s is already deployed", stackName))
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, stackName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s is not deployed?", app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
version := deployMeta.Version
|
||||||
|
if deployMeta.IsChaos {
|
||||||
|
version = deployMeta.ChaosVersion
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := internal.DeployOverview(
|
||||||
|
app,
|
||||||
|
version,
|
||||||
|
config.MISSING_DEFAULT,
|
||||||
|
"",
|
||||||
|
nil,
|
||||||
|
nil,
|
||||||
|
nil,
|
||||||
|
nil,
|
||||||
|
); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := stack.Deploy{Composefiles: composeFiles, Namespace: stackName}
|
||||||
|
compose, err := appPkg.GetAppComposeConfig(app.Name, opts, app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
stack.WaitTimeout, err = appPkg.GetTimeoutFromLabel(compose, stackName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
rmOpts := stack.Remove{
|
||||||
|
Namespaces: []string{stackName},
|
||||||
|
Detach: false,
|
||||||
|
}
|
||||||
|
if err := stack.RunRemove(context.Background(), cl, rmOpts); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if prune {
|
||||||
|
if err := pruneApp(cl, app); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("undeploy succeeded 🟢"))
|
||||||
|
|
||||||
|
if err := app.WriteRecipeVersion(version, false); err != nil {
|
||||||
|
log.Fatal(i18n.G("writing recipe version failed: %s", err))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// pruneApp runs the equivalent of a "docker system prune" but only filtering
|
||||||
|
// against resources connected with the app deployment. It is not a system wide
|
||||||
|
// prune. Volumes are not pruned to avoid unwated data loss.
|
||||||
|
func pruneApp(cl *dockerClient.Client, app appPkg.App) error {
|
||||||
|
stackName := app.StackName()
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
pruneFilters := filters.NewArgs()
|
||||||
|
stackSearch := fmt.Sprintf("%s*", stackName)
|
||||||
|
pruneFilters.Add("label", stackSearch)
|
||||||
|
cr, err := cl.ContainersPrune(ctx, pruneFilters)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
cntSpaceReclaimed := formatter.ByteCountSI(cr.SpaceReclaimed)
|
||||||
|
log.Info(i18n.G("containers pruned: %d; space reclaimed: %s", len(cr.ContainersDeleted), cntSpaceReclaimed))
|
||||||
|
|
||||||
|
nr, err := cl.NetworksPrune(ctx, pruneFilters)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("networks pruned: %d", len(nr.NetworksDeleted)))
|
||||||
|
|
||||||
|
ir, err := cl.ImagesPrune(ctx, pruneFilters)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
imgSpaceReclaimed := formatter.ByteCountSI(ir.SpaceReclaimed)
|
||||||
|
log.Info(i18n.G("images pruned: %d; space reclaimed: %s", len(ir.ImagesDeleted), imgSpaceReclaimed))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
prune bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppUndeployCommand.Flags().BoolVarP(
|
||||||
|
&prune,
|
||||||
|
i18n.G("prune"),
|
||||||
|
i18n.G("p"),
|
||||||
|
false,
|
||||||
|
i18n.G("prune unused containers, networks, and dangling images"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,498 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/deploy"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/lint"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
stack "coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"coopcloud.tech/tagcmp"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app upgrade` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appUpgradeAliases = i18n.G("up")
|
||||||
|
|
||||||
|
var AppUpgradeCommand = &cobra.Command{
|
||||||
|
// translators: `app upgrade` command
|
||||||
|
Use: i18n.G("upgrade <domain> [version] [flags]"),
|
||||||
|
Aliases: strings.Split(appUpgradeAliases, ","),
|
||||||
|
// translators: Short description for `app upgrade` command
|
||||||
|
Short: i18n.G("Upgrade an app"),
|
||||||
|
Long: i18n.G(`Upgrade an app.
|
||||||
|
|
||||||
|
Unlike "abra app deploy", chaos operations are not supported here. Only recipe
|
||||||
|
versions are supported values for "[version]".
|
||||||
|
|
||||||
|
It is possible to "--force/-f" an upgrade if you want to re-deploy a specific
|
||||||
|
version.
|
||||||
|
|
||||||
|
Only the deployed version is consulted when trying to determine what upgrades
|
||||||
|
are available. The live deployment version is the "source of truth" in this
|
||||||
|
case. The stored .env version is not consulted.
|
||||||
|
|
||||||
|
An upgrade can be destructive, please ensure you have a copy of your app data
|
||||||
|
beforehand. See "abra app backup" for more.`),
|
||||||
|
Args: cobra.RangeArgs(1, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
case 1:
|
||||||
|
app, err := appPkg.Get(args[0])
|
||||||
|
if err != nil {
|
||||||
|
return []string{i18n.G("autocomplete failed: %s", err)}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
return autocomplete.RecipeVersionComplete(app.Recipe.Name)
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
var (
|
||||||
|
upgradeWarnMessages []string
|
||||||
|
chosenUpgrade string
|
||||||
|
availableUpgrades []string
|
||||||
|
upgradeReleaseNotes string
|
||||||
|
)
|
||||||
|
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
if err := app.Recipe.Ensure(recipe.EnsureContext{
|
||||||
|
Chaos: internal.Chaos,
|
||||||
|
Offline: internal.Offline,
|
||||||
|
// Ignore the env version for now, to make sure we are at the latest commit.
|
||||||
|
// This enables us to get release notes, that were added after a release.
|
||||||
|
IgnoreEnvVersion: true,
|
||||||
|
}); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := ensureDeployed(cl, app)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := lint.LintForErrors(app.Recipe); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
versions, err := app.Recipe.Tags()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE(d1): we've no idea what the live deployment version is, so every
|
||||||
|
// possible upgrade can be shown. it's up to the user to make the choice
|
||||||
|
if deployMeta.Version == config.UNKNOWN_DEFAULT {
|
||||||
|
availableUpgrades = versions
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) == 2 && args[1] != "" {
|
||||||
|
chosenUpgrade = args[1]
|
||||||
|
|
||||||
|
if err := validateUpgradeVersionArg(chosenUpgrade, app, deployMeta); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
availableUpgrades = append(availableUpgrades, chosenUpgrade)
|
||||||
|
}
|
||||||
|
|
||||||
|
if deployMeta.Version != config.UNKNOWN_DEFAULT && chosenUpgrade == "" {
|
||||||
|
upgradeAvailable, err := ensureUpgradesAvailable(app, versions, &availableUpgrades, deployMeta)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !upgradeAvailable {
|
||||||
|
log.Info(i18n.G("no available upgrades"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.Force || internal.NoInput || chosenUpgrade != "" {
|
||||||
|
if len(availableUpgrades) > 0 {
|
||||||
|
chosenUpgrade = availableUpgrades[len(availableUpgrades)-1]
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := chooseUpgrade(availableUpgrades, deployMeta, &chosenUpgrade); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.Force &&
|
||||||
|
chosenUpgrade == "" &&
|
||||||
|
deployMeta.Version != config.UNKNOWN_DEFAULT {
|
||||||
|
chosenUpgrade = deployMeta.Version
|
||||||
|
}
|
||||||
|
|
||||||
|
if chosenUpgrade == "" {
|
||||||
|
log.Fatal(i18n.G("unknown deployed version, unable to upgrade"))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("choosing %s as version to upgrade", chosenUpgrade))
|
||||||
|
|
||||||
|
// Get the release notes before checking out the new version in the
|
||||||
|
// recipe. This enables us to get release notes, that were added after
|
||||||
|
// a release.
|
||||||
|
if err := getReleaseNotes(app, versions, chosenUpgrade, deployMeta, &upgradeReleaseNotes); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := app.Recipe.EnsureVersion(chosenUpgrade); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := deploy.MergeAbraShEnv(app.Recipe, app.Env); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
stackName := app.StackName()
|
||||||
|
deployOpts := stack.Deploy{
|
||||||
|
Composefiles: composeFiles,
|
||||||
|
Namespace: stackName,
|
||||||
|
Prune: false,
|
||||||
|
ResolveImage: stack.ResolveImageAlways,
|
||||||
|
Detach: false,
|
||||||
|
}
|
||||||
|
|
||||||
|
compose, err := appPkg.GetAppComposeConfig(app.Name, deployOpts, app.Env)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
newRecipeWithUpgradeVersion := fmt.Sprintf("%s:%s", app.Recipe.Name, chosenUpgrade)
|
||||||
|
appPkg.ExposeAllEnv(stackName, compose, app.Env, newRecipeWithUpgradeVersion)
|
||||||
|
|
||||||
|
appPkg.SetRecipeLabel(compose, stackName, app.Recipe.Name)
|
||||||
|
appPkg.SetChaosLabel(compose, stackName, internal.Chaos)
|
||||||
|
if internal.Chaos {
|
||||||
|
appPkg.SetChaosVersionLabel(compose, stackName, chosenUpgrade)
|
||||||
|
}
|
||||||
|
|
||||||
|
envVars, err := appPkg.CheckEnv(app)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, envVar := range envVars {
|
||||||
|
if !envVar.Present {
|
||||||
|
upgradeWarnMessages = append(upgradeWarnMessages,
|
||||||
|
i18n.G("%s missing from %s.env", envVar.Name, app.Domain),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather secrets
|
||||||
|
secretInfo, err := deploy.GatherSecretsForDeploy(cl, app, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather configs
|
||||||
|
configInfo, err := deploy.GatherConfigsForDeploy(cl, app, compose, app.Env, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gather images
|
||||||
|
imageInfo, err := deploy.GatherImagesForDeploy(cl, app, compose, internal.ShowUnchanged)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if showReleaseNotes {
|
||||||
|
fmt.Print(upgradeReleaseNotes)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if upgradeReleaseNotes == "" {
|
||||||
|
upgradeWarnMessages = append(
|
||||||
|
upgradeWarnMessages,
|
||||||
|
i18n.G("no release notes for upgrading from %s to %s", deployMeta.Version, chosenUpgrade),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployedVersion := deployMeta.Version
|
||||||
|
if deployMeta.IsChaos {
|
||||||
|
deployedVersion = deployMeta.ChaosVersion
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := internal.DeployOverview(
|
||||||
|
app,
|
||||||
|
deployedVersion,
|
||||||
|
chosenUpgrade,
|
||||||
|
upgradeReleaseNotes,
|
||||||
|
upgradeWarnMessages,
|
||||||
|
secretInfo,
|
||||||
|
configInfo,
|
||||||
|
imageInfo,
|
||||||
|
); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
stack.WaitTimeout, err = appPkg.GetTimeoutFromLabel(compose, stackName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
serviceNames, err := appPkg.GetAppServiceNames(app.Name)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := app.Filters(true, false, serviceNames...)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := stack.RunDeploy(
|
||||||
|
cl,
|
||||||
|
deployOpts,
|
||||||
|
compose,
|
||||||
|
stackName,
|
||||||
|
app.Server,
|
||||||
|
internal.DontWaitConverge,
|
||||||
|
internal.NoInput,
|
||||||
|
f,
|
||||||
|
); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
postDeployCmds, ok := app.Env["POST_UPGRADE_CMDS"]
|
||||||
|
if ok && !internal.DontWaitConverge {
|
||||||
|
log.Debug(i18n.G("run the following post-deploy commands: %s", postDeployCmds))
|
||||||
|
|
||||||
|
if err := internal.PostCmds(cl, app, postDeployCmds); err != nil {
|
||||||
|
log.Fatal(i18n.G("attempting to run post deploy commands, saw: %s", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := app.WriteRecipeVersion(chosenUpgrade, false); err != nil {
|
||||||
|
log.Fatal(i18n.G("writing recipe version failed: %s", err))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// chooseUpgrade prompts the user to choose an upgrade interactively.
|
||||||
|
func chooseUpgrade(
|
||||||
|
availableUpgrades []string,
|
||||||
|
deployMeta stack.DeployMeta,
|
||||||
|
chosenUpgrade *string,
|
||||||
|
) error {
|
||||||
|
msg := i18n.G("please select an upgrade (version: %s):", deployMeta.Version)
|
||||||
|
|
||||||
|
if deployMeta.IsChaos {
|
||||||
|
chaosVersion := formatter.BoldDirtyDefault(deployMeta.ChaosVersion)
|
||||||
|
|
||||||
|
msg = i18n.G(
|
||||||
|
"please select an upgrade (version: %s, chaos: %s):",
|
||||||
|
deployMeta.Version,
|
||||||
|
chaosVersion,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
prompt := &survey.Select{
|
||||||
|
Message: msg,
|
||||||
|
Options: internal.SortVersionsDesc(availableUpgrades),
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := survey.AskOne(prompt, chosenUpgrade); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getReleaseNotes(
|
||||||
|
app appPkg.App,
|
||||||
|
versions []string,
|
||||||
|
chosenUpgrade string,
|
||||||
|
deployMeta stack.DeployMeta,
|
||||||
|
upgradeReleaseNotes *string,
|
||||||
|
) error {
|
||||||
|
parsedChosenUpgrade, err := tagcmp.Parse(chosenUpgrade)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("parsing chosen upgrade version failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
parsedDeployedVersion, err := tagcmp.Parse(deployMeta.Version)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("parsing deployment version failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, version := range internal.SortVersionsDesc(versions) {
|
||||||
|
parsedVersion, err := tagcmp.Parse(version)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("parsing recipe version failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if parsedVersion.IsGreaterThan(parsedDeployedVersion) &&
|
||||||
|
parsedVersion.IsLessThan(parsedChosenUpgrade) {
|
||||||
|
note, err := app.Recipe.GetReleaseNotes(version, app.Domain)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if note != "" {
|
||||||
|
// NOTE(d1): trim any final newline on the end of the note itself before
|
||||||
|
// we manually handle newlines (for multiple release notes and
|
||||||
|
// ensuring space between the warning messages)
|
||||||
|
note = strings.TrimSuffix(note, "\n")
|
||||||
|
|
||||||
|
*upgradeReleaseNotes += fmt.Sprintf("%s\n", note)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ensureUpgradesAvailable ensures that there are available upgrades.
|
||||||
|
func ensureUpgradesAvailable(
|
||||||
|
app appPkg.App,
|
||||||
|
versions []string,
|
||||||
|
availableUpgrades *[]string,
|
||||||
|
deployMeta stack.DeployMeta,
|
||||||
|
) (bool, error) {
|
||||||
|
parsedDeployedVersion, err := tagcmp.Parse(deployMeta.Version)
|
||||||
|
if err != nil {
|
||||||
|
return false, errors.New(i18n.G("parsing deployed version failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, version := range versions {
|
||||||
|
parsedVersion, err := tagcmp.Parse(version)
|
||||||
|
if err != nil {
|
||||||
|
return false, errors.New(i18n.G("parsing recipe version failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if parsedVersion.IsGreaterThan(parsedDeployedVersion) &&
|
||||||
|
!(parsedVersion.Equals(parsedDeployedVersion)) {
|
||||||
|
*availableUpgrades = append(*availableUpgrades, version)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(*availableUpgrades) == 0 && !internal.Force {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// validateUpgradeVersionArg validates the specific version.
|
||||||
|
func validateUpgradeVersionArg(
|
||||||
|
specificVersion string,
|
||||||
|
app appPkg.App,
|
||||||
|
deployMeta stack.DeployMeta,
|
||||||
|
) error {
|
||||||
|
parsedSpecificVersion, err := tagcmp.Parse(specificVersion)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("'%s' is not a known version for %s", specificVersion, app.Recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
parsedDeployedVersion, err := tagcmp.Parse(deployMeta.Version)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("'%s' is not a known version", deployMeta.Version))
|
||||||
|
}
|
||||||
|
|
||||||
|
if parsedSpecificVersion.IsLessThan(parsedDeployedVersion) &&
|
||||||
|
!parsedSpecificVersion.Equals(parsedDeployedVersion) {
|
||||||
|
return errors.New(i18n.G("%s is not an upgrade for %s?", deployMeta.Version, specificVersion))
|
||||||
|
}
|
||||||
|
|
||||||
|
if parsedSpecificVersion.Equals(parsedDeployedVersion) && !internal.Force {
|
||||||
|
return errors.New(i18n.G("%s is not an upgrade for %s?", deployMeta.Version, specificVersion))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ensureDeployed ensures the app is deployed and if so, returns deployment
|
||||||
|
// meta info.
|
||||||
|
func ensureDeployed(cl *dockerClient.Client, app appPkg.App) (stack.DeployMeta, error) {
|
||||||
|
log.Debug(i18n.G("checking whether %s is already deployed", app.StackName()))
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
return stack.DeployMeta{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !deployMeta.IsDeployed {
|
||||||
|
return stack.DeployMeta{}, errors.New(i18n.G("%s is not deployed?", app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
return deployMeta, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var showReleaseNotes bool
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&internal.Force,
|
||||||
|
i18n.G("force"),
|
||||||
|
i18n.G("f"),
|
||||||
|
false,
|
||||||
|
i18n.G("perform action without further prompt"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&internal.NoDomainChecks,
|
||||||
|
i18n.G("no-domain-checks"),
|
||||||
|
i18n.G("D"),
|
||||||
|
false,
|
||||||
|
i18n.G("disable public DNS checks"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&internal.DontWaitConverge,
|
||||||
|
i18n.G("no-converge-checks"),
|
||||||
|
i18n.G("c"),
|
||||||
|
false,
|
||||||
|
i18n.G("disable converge logic checks"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&showReleaseNotes,
|
||||||
|
i18n.G("releasenotes"),
|
||||||
|
i18n.G("r"),
|
||||||
|
false,
|
||||||
|
i18n.G("only show release notes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
AppUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&internal.ShowUnchanged,
|
||||||
|
i18n.G("show-unchanged"),
|
||||||
|
i18n.G("U"),
|
||||||
|
false,
|
||||||
|
i18n.G("show all configs & images, including unchanged ones"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,221 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra app volume list` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appVolumeListAliases = i18n.G("ls")
|
||||||
|
|
||||||
|
var AppVolumeListCommand = &cobra.Command{
|
||||||
|
// translators: `app volume list` command
|
||||||
|
Use: i18n.G("list <domain> [flags]"),
|
||||||
|
Aliases: strings.Split(appVolumeListAliases, ","),
|
||||||
|
// translators: Short description for `app list` command
|
||||||
|
Short: i18n.G("List volumes associated with an app"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
filters, err := app.Filters(false, true)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
volumes, err := client.GetVolumes(cl, context.Background(), app.Server, filters)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{i18n.G("NAME"), i18n.G("ON SERVER")}
|
||||||
|
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Headers(headers...)
|
||||||
|
|
||||||
|
var rows [][]string
|
||||||
|
for _, volume := range volumes {
|
||||||
|
row := []string{volume.Name, volume.Mountpoint}
|
||||||
|
rows = append(rows, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Rows(rows...)
|
||||||
|
|
||||||
|
if len(rows) > 0 {
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Warn(i18n.G("no volumes created for %s", app.Name))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app volume remove` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appVolumeRemoveAliases = i18n.G("rm")
|
||||||
|
|
||||||
|
var AppVolumeRemoveCommand = &cobra.Command{
|
||||||
|
// translators: `app volume remove` command
|
||||||
|
Use: i18n.G("remove <domain> [volume] [flags]"),
|
||||||
|
// translators: Short description for `app volume remove` command
|
||||||
|
Short: i18n.G("Remove volume(s) associated with an app"),
|
||||||
|
Long: i18n.G(`Remove volumes associated with an app.
|
||||||
|
|
||||||
|
The app in question must be undeployed before you try to remove volumes. See
|
||||||
|
"abra app undeploy <domain>" for more.
|
||||||
|
|
||||||
|
The command is interactive and will show a multiple select input which allows
|
||||||
|
you to make a seclection. Use the "?" key to see more help on navigating this
|
||||||
|
interface.
|
||||||
|
|
||||||
|
Passing "--force/-f" will select all volumes for removal. Be careful.`),
|
||||||
|
Example: i18n.G(` # delete volumes interactively
|
||||||
|
abra app volume rm 1312.net
|
||||||
|
|
||||||
|
# delete specific volume
|
||||||
|
abra app volume rm 1312.net my_volume`),
|
||||||
|
Aliases: strings.Split(appVolumeRemoveAliases, ","),
|
||||||
|
Args: cobra.MinimumNArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.AppNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
app := internal.ValidateApp(args)
|
||||||
|
|
||||||
|
var volumeToDelete string
|
||||||
|
if len(args) == 2 {
|
||||||
|
volumeToDelete = args[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.New(app.Server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if deployMeta.IsDeployed {
|
||||||
|
log.Fatal(i18n.G("%s is still deployed. Run \"abra app undeploy %s\"", app.Name, app.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
filters, err := app.Filters(false, true)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
volumeList, err := client.GetVolumes(cl, context.Background(), app.Server, filters)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
volumeNames := client.GetVolumeNames(volumeList)
|
||||||
|
|
||||||
|
if volumeToDelete != "" {
|
||||||
|
var exactMatch bool
|
||||||
|
|
||||||
|
fullVolumeToDeleteName := fmt.Sprintf("%s_%s", app.StackName(), volumeToDelete)
|
||||||
|
for _, volName := range volumeNames {
|
||||||
|
if volName == fullVolumeToDeleteName {
|
||||||
|
exactMatch = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !exactMatch {
|
||||||
|
log.Fatal(i18n.G("unable to remove volume: no volume with name '%s'?", volumeToDelete))
|
||||||
|
}
|
||||||
|
|
||||||
|
err := client.RemoveVolumes(cl, context.Background(), []string{fullVolumeToDeleteName}, internal.Force, 5)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("removing volume %s failed: %s", volumeToDelete, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("volume %s removed successfully", volumeToDelete))
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var volumesToRemove []string
|
||||||
|
if !internal.Force && !internal.NoInput {
|
||||||
|
volumesPrompt := &survey.MultiSelect{
|
||||||
|
Message: i18n.G("which volumes do you want to remove?"),
|
||||||
|
Help: i18n.G("'x' indicates selected, enter / return to confirm, ctrl-c to exit, vim mode is enabled"),
|
||||||
|
VimMode: true,
|
||||||
|
Options: volumeNames,
|
||||||
|
Default: volumeNames,
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(volumesPrompt, &volumesToRemove); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.Force || internal.NoInput {
|
||||||
|
volumesToRemove = volumeNames
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(volumesToRemove) > 0 {
|
||||||
|
err := client.RemoveVolumes(cl, context.Background(), volumesToRemove, internal.Force, 5)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("removing volumes failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("%d volumes removed successfully", len(volumesToRemove)))
|
||||||
|
} else {
|
||||||
|
log.Info(i18n.G("no volumes removed"))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra app volume` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appVolumeAliases = i18n.G("vl")
|
||||||
|
|
||||||
|
var AppVolumeCommand = &cobra.Command{
|
||||||
|
// translators: `app volume` command group
|
||||||
|
Use: i18n.G("volume [cmd] [args] [flags]"),
|
||||||
|
Aliases: strings.Split(appVolumeAliases, ","),
|
||||||
|
Short: i18n.G("Manage app volumes"),
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
AppVolumeRemoveCommand.Flags().BoolVarP(
|
||||||
|
&internal.Force,
|
||||||
|
i18n.G("force"),
|
||||||
|
i18n.G("f"),
|
||||||
|
false,
|
||||||
|
i18n.G("perform action without further prompt"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,322 @@
|
|||||||
|
package catalogue
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"slices"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/catalogue"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
gitPkg "coopcloud.tech/abra/pkg/git"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/go-git/go-git/v5"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra catalogue sync` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appCatalogueSyncAliases = i18n.G("s")
|
||||||
|
|
||||||
|
var CatalogueSyncCommand = &cobra.Command{
|
||||||
|
// translators: `catalogue sync` command
|
||||||
|
Use: i18n.G("sync [flags]"),
|
||||||
|
Aliases: strings.Split(appCatalogueSyncAliases, ","),
|
||||||
|
// translators: Short description for `catalogue sync` command
|
||||||
|
Short: i18n.G("Sync recipe catalogue for latest changes"),
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
if err := catalogue.EnsureCatalogue(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := catalogue.EnsureUpToDate(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("catalogue successfully synced"))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra catalogue` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var appCatalogueAliases = i18n.G("g")
|
||||||
|
|
||||||
|
var CatalogueGenerateCommand = &cobra.Command{
|
||||||
|
// translators: `catalogue generate` command
|
||||||
|
Use: i18n.G("generate [recipe] [flags]"),
|
||||||
|
Aliases: strings.Split(appCatalogueAliases, ","),
|
||||||
|
// translators: Short description for `catalogue generate` command
|
||||||
|
Short: i18n.G("Generate the recipe catalogue"),
|
||||||
|
Long: i18n.G(`Generate a new copy of the recipe catalogue.
|
||||||
|
|
||||||
|
N.B. this command **will** wipe local unstaged changes from your local recipes
|
||||||
|
if present. "--chaos/-C" on this command refers to the catalogue repository
|
||||||
|
("$ABRA_DIR/catalogue") and not the recipes. Please take care not to lose your
|
||||||
|
changes.
|
||||||
|
|
||||||
|
It is possible to generate new metadata for a single recipe by passing
|
||||||
|
[recipe]. The existing local catalogue will be updated, not overwritten.
|
||||||
|
|
||||||
|
It is quite easy to get rate limited by Docker Hub when running this command.
|
||||||
|
If you have a Hub account you can "docker login" and Abra will automatically
|
||||||
|
use those details.
|
||||||
|
|
||||||
|
Publish your new release to git.coopcloud.tech with "--publish/-p". This
|
||||||
|
requires that you have permission to git push to these repositories and have
|
||||||
|
your SSH keys configured on your account. Enable ssh-agent and make sure to add
|
||||||
|
your private key and enter your passphrase beforehand.
|
||||||
|
|
||||||
|
eval ` + "`ssh-agent`" + `
|
||||||
|
ssh-add ~/.ssh/<my-ssh-private-key-for-git-coopcloud-tech>`),
|
||||||
|
Example: ` # publish catalogue
|
||||||
|
eval ` + "`ssh-agent`" + `
|
||||||
|
ssh-add ~/.ssh/id_ed25519
|
||||||
|
abra catalogue generate -p`,
|
||||||
|
Args: cobra.RangeArgs(0, 1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
var recipeName string
|
||||||
|
if len(args) > 0 {
|
||||||
|
recipeName = args[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
if os.Getenv("SSH_AUTH_SOCK") == "" {
|
||||||
|
log.Warn(i18n.G("ssh: SSH_AUTH_SOCK missing, --publish/-p will fail. see \"abra catalogue generate --help\""))
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeName != "" {
|
||||||
|
internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := catalogue.EnsureCatalogue(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !internal.Chaos {
|
||||||
|
if err := catalogue.EnsureIsClean(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
repos, err := recipe.ReadReposMetadata(internal.Debug)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
barLength := len(repos)
|
||||||
|
if recipeName != "" {
|
||||||
|
barLength = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
if !skipUpdates {
|
||||||
|
if err := recipe.UpdateRepositories(repos, recipeName, internal.Debug); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var warnings []string
|
||||||
|
catl := make(recipe.RecipeCatalogue)
|
||||||
|
catlBar := formatter.CreateProgressbar(barLength, i18n.G("collecting catalogue metadata"))
|
||||||
|
for _, recipeMeta := range repos {
|
||||||
|
if recipeName != "" && recipeName != recipeMeta.Name {
|
||||||
|
if !internal.Debug {
|
||||||
|
catlBar.Add(1)
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
r := recipe.Get(recipeMeta.Name)
|
||||||
|
versions, warnMsgs, err := r.GetRecipeVersions()
|
||||||
|
if err != nil {
|
||||||
|
warnings = append(warnings, err.Error())
|
||||||
|
}
|
||||||
|
if len(warnMsgs) > 0 {
|
||||||
|
warnings = append(warnings, warnMsgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
features, category, warnMsgs, err := recipe.GetRecipeFeaturesAndCategory(r)
|
||||||
|
if err != nil {
|
||||||
|
warnings = append(warnings, err.Error())
|
||||||
|
}
|
||||||
|
if len(warnMsgs) > 0 {
|
||||||
|
warnings = append(warnings, warnMsgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
catl[recipeMeta.Name] = recipe.RecipeMeta{
|
||||||
|
Name: recipeMeta.Name,
|
||||||
|
Repository: recipeMeta.CloneURL,
|
||||||
|
SSHURL: recipeMeta.SSHURL,
|
||||||
|
Icon: recipeMeta.AvatarURL,
|
||||||
|
DefaultBranch: recipeMeta.DefaultBranch,
|
||||||
|
Description: recipeMeta.Description,
|
||||||
|
Website: recipeMeta.Website,
|
||||||
|
Versions: versions,
|
||||||
|
Category: category,
|
||||||
|
Features: features,
|
||||||
|
}
|
||||||
|
|
||||||
|
if !internal.Debug {
|
||||||
|
catlBar.Add(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := catlBar.Close(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var uniqueWarnings []string
|
||||||
|
for _, w := range warnings {
|
||||||
|
if !slices.Contains(uniqueWarnings, w) {
|
||||||
|
uniqueWarnings = append(uniqueWarnings, w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, warnMsg := range uniqueWarnings {
|
||||||
|
log.Warn(warnMsg)
|
||||||
|
}
|
||||||
|
|
||||||
|
recipesJSON, err := json.MarshalIndent(catl, "", " ")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeName == "" {
|
||||||
|
if err := ioutil.WriteFile(config.RECIPES_JSON, recipesJSON, 0764); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
catlFS, err := recipe.ReadRecipeCatalogue(internal.Offline)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
catlFS[recipeName] = catl[recipeName]
|
||||||
|
|
||||||
|
updatedRecipesJSON, err := json.MarshalIndent(catlFS, "", " ")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := ioutil.WriteFile(config.RECIPES_JSON, updatedRecipesJSON, 0764); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("generated recipe catalogue: %s", config.RECIPES_JSON))
|
||||||
|
|
||||||
|
cataloguePath := path.Join(config.ABRA_DIR, "catalogue")
|
||||||
|
if publishChanges {
|
||||||
|
|
||||||
|
isClean, err := gitPkg.IsClean(cataloguePath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if isClean {
|
||||||
|
if !internal.Dry {
|
||||||
|
log.Fatal(i18n.G("no changes discovered in %s, nothing to publish?", cataloguePath))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
msg := i18n.G("chore: publish new catalogue release changes")
|
||||||
|
if err := gitPkg.Commit(cataloguePath, msg, internal.Dry); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
repo, err := git.PlainOpen(cataloguePath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
sshURL := fmt.Sprintf(config.TOOLSHED_SSH_URL_TEMPLATE, config.CATALOGUE_JSON_REPO_NAME)
|
||||||
|
if err := gitPkg.CreateRemote(repo, "origin-ssh", sshURL, internal.Dry); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := gitPkg.Push(cataloguePath, "origin-ssh", false, internal.Dry); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
repo, err := git.PlainOpen(cataloguePath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
head, err := repo.Head()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !internal.Dry && publishChanges {
|
||||||
|
url := fmt.Sprintf("%s/%s/commit/%s", config.REPOS_BASE_URL, config.CATALOGUE_JSON_REPO_NAME, head.Hash())
|
||||||
|
log.Info(i18n.G("new changes published: %s", url))
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.Dry {
|
||||||
|
log.Info(i18n.G("dry run: no changes published"))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// CatalogueCommand defines the `abra catalogue` command and sub-commands.
|
||||||
|
var CatalogueCommand = &cobra.Command{
|
||||||
|
// translators: `catalogue` command group
|
||||||
|
Use: i18n.G("catalogue [cmd] [args] [flags]"),
|
||||||
|
// translators: Short description for `catalogue` command group
|
||||||
|
Short: i18n.G("Manage the recipe catalogue"),
|
||||||
|
Aliases: []string{"c"},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
publishChanges bool
|
||||||
|
skipUpdates bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
CatalogueGenerateCommand.Flags().BoolVarP(
|
||||||
|
&publishChanges,
|
||||||
|
i18n.G("publish"),
|
||||||
|
i18n.G("p"),
|
||||||
|
false,
|
||||||
|
i18n.G("publish changes to git.coopcloud.tech"),
|
||||||
|
)
|
||||||
|
|
||||||
|
CatalogueGenerateCommand.Flags().BoolVarP(
|
||||||
|
&internal.Dry,
|
||||||
|
i18n.G("dry-run"),
|
||||||
|
i18n.G("r"),
|
||||||
|
false,
|
||||||
|
i18n.G("report changes that would be made"),
|
||||||
|
)
|
||||||
|
|
||||||
|
CatalogueGenerateCommand.Flags().BoolVarP(
|
||||||
|
&skipUpdates,
|
||||||
|
i18n.G("skip-updates"),
|
||||||
|
i18n.G("s"),
|
||||||
|
false,
|
||||||
|
i18n.G("skip updating recipe repositories"),
|
||||||
|
)
|
||||||
|
|
||||||
|
CatalogueGenerateCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,71 @@
|
|||||||
|
package cli
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra autocomplete` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var autocompleteAliases = i18n.G("ac")
|
||||||
|
|
||||||
|
var AutocompleteCommand = &cobra.Command{
|
||||||
|
// translators: `autocomplete` command
|
||||||
|
Use: i18n.G("autocomplete [bash|zsh|fish|powershell]"),
|
||||||
|
Aliases: strings.Split(autocompleteAliases, ","),
|
||||||
|
// translators: Short description for `autocomplete` command
|
||||||
|
Short: i18n.G("Generate autocompletion script"),
|
||||||
|
Long: i18n.G(`To load completions:
|
||||||
|
|
||||||
|
Bash:
|
||||||
|
# Load autocompletion for the current Bash session
|
||||||
|
$ source <(abra autocomplete bash)
|
||||||
|
|
||||||
|
# To load autocompletion for each session, execute once:
|
||||||
|
# Linux:
|
||||||
|
$ abra autocomplete bash | sudo tee /etc/bash_completion.d/abra
|
||||||
|
# macOS:
|
||||||
|
$ abra autocomplete bash | sudo tee $(brew --prefix)/etc/bash_completion.d/abra
|
||||||
|
|
||||||
|
Zsh:
|
||||||
|
# If shell autocompletion is not already enabled in your environment,
|
||||||
|
# you will need to enable it. You can execute the following once:
|
||||||
|
|
||||||
|
$ echo "autoload -U compinit; compinit" >> ~/.zshrc
|
||||||
|
|
||||||
|
# To load autocompletions for each session, execute once:
|
||||||
|
$ abra autocomplete zsh > "${fpath[1]}/_abra"
|
||||||
|
|
||||||
|
# You will need to start a new shell for this setup to take effect.
|
||||||
|
|
||||||
|
fish:
|
||||||
|
$ abra autocomplete fish | source
|
||||||
|
|
||||||
|
# To load autocompletions for each session, execute once:
|
||||||
|
$ abra autocomplete fish > ~/.config/fish/completions/abra.fish
|
||||||
|
|
||||||
|
PowerShell:
|
||||||
|
PS> abra autocomplete powershell | Out-String | Invoke-Expression
|
||||||
|
|
||||||
|
# To load autocompletions for every new session, run:
|
||||||
|
PS> abra autocomplete powershell > abra.ps1
|
||||||
|
# and source this file from your PowerShell profile.`),
|
||||||
|
DisableFlagsInUseLine: true,
|
||||||
|
ValidArgs: []string{"bash", "zsh", "fish", "powershell"},
|
||||||
|
Args: cobra.MatchAll(cobra.ExactArgs(1), cobra.OnlyValidArgs),
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
switch args[0] {
|
||||||
|
case "bash":
|
||||||
|
cmd.Root().GenBashCompletion(os.Stdout)
|
||||||
|
case "zsh":
|
||||||
|
cmd.Root().GenZshCompletion(os.Stdout)
|
||||||
|
case "fish":
|
||||||
|
cmd.Root().GenFishCompletion(os.Stdout, true)
|
||||||
|
case "powershell":
|
||||||
|
cmd.Root().GenPowerShellCompletionWithDesc(os.Stdout)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
@@ -0,0 +1,76 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
containerPkg "coopcloud.tech/abra/pkg/container"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/service"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/container"
|
||||||
|
"github.com/docker/cli/cli/command"
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
containertypes "github.com/docker/docker/api/types/container"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
)
|
||||||
|
|
||||||
|
// RetrieveBackupBotContainer gets the deployed backupbot container.
|
||||||
|
func RetrieveBackupBotContainer(cl *dockerClient.Client) (types.Container, error) {
|
||||||
|
ctx := context.Background()
|
||||||
|
chosenService, err := service.GetServiceByLabel(ctx, cl, config.BackupbotLabel, NoInput)
|
||||||
|
if err != nil {
|
||||||
|
return types.Container{}, errors.New(i18n.G("no backupbot discovered, is it deployed?"))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("retrieved %s as backup enabled service", chosenService.Spec.Name))
|
||||||
|
|
||||||
|
filters := filters.NewArgs()
|
||||||
|
filters.Add("name", chosenService.Spec.Name)
|
||||||
|
targetContainer, err := containerPkg.GetContainer(
|
||||||
|
ctx,
|
||||||
|
cl,
|
||||||
|
filters,
|
||||||
|
NoInput,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return types.Container{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return targetContainer, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RunBackupCmdRemote runs a backup related command on a remote backupbot container.
|
||||||
|
func RunBackupCmdRemote(
|
||||||
|
cl *dockerClient.Client,
|
||||||
|
backupCmd string,
|
||||||
|
containerID string,
|
||||||
|
execEnv []string) (io.Writer, error) {
|
||||||
|
execBackupListOpts := containertypes.ExecOptions{
|
||||||
|
AttachStderr: true,
|
||||||
|
AttachStdin: true,
|
||||||
|
AttachStdout: true,
|
||||||
|
Cmd: []string{"/usr/bin/backup", "--", backupCmd},
|
||||||
|
Detach: false,
|
||||||
|
Env: execEnv,
|
||||||
|
Tty: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("running backup %s on %s with exec config %v", backupCmd, containerID, execBackupListOpts))
|
||||||
|
|
||||||
|
// FIXME: avoid instantiating a new CLI
|
||||||
|
dcli, err := command.NewDockerCli()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
out, err := container.RunExec(dcli, cl, containerID, &execBackupListOpts)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return out, nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
var (
|
||||||
|
// NOTE(d1): global
|
||||||
|
Debug bool
|
||||||
|
NoInput bool
|
||||||
|
Offline bool
|
||||||
|
Help bool
|
||||||
|
Version bool
|
||||||
|
|
||||||
|
// NOTE(d1): sub-command specific
|
||||||
|
Chaos bool
|
||||||
|
DeployLatest bool
|
||||||
|
DontWaitConverge bool
|
||||||
|
Dry bool
|
||||||
|
Force bool
|
||||||
|
MachineReadable bool
|
||||||
|
Major bool
|
||||||
|
Minor bool
|
||||||
|
NoDomainChecks bool
|
||||||
|
Patch bool
|
||||||
|
ShowUnchanged bool
|
||||||
|
)
|
||||||
@@ -0,0 +1,146 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"os/exec"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
containerPkg "coopcloud.tech/abra/pkg/container"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/container"
|
||||||
|
"github.com/docker/cli/cli/command"
|
||||||
|
containertypes "github.com/docker/docker/api/types/container"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
"github.com/docker/docker/pkg/archive"
|
||||||
|
)
|
||||||
|
|
||||||
|
// RunCmdRemote executes an abra.sh command in the target service
|
||||||
|
func RunCmdRemote(
|
||||||
|
cl *dockerClient.Client,
|
||||||
|
app appPkg.App,
|
||||||
|
disableTTY bool,
|
||||||
|
abraSh, serviceName, cmdName, cmdArgs, remoteUser string) error {
|
||||||
|
filters := filters.NewArgs()
|
||||||
|
filters.Add("name", fmt.Sprintf("^%s_%s", app.StackName(), serviceName))
|
||||||
|
|
||||||
|
targetContainer, err := containerPkg.GetContainer(context.Background(), cl, filters, false)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("retrieved %s as target container on %s", formatter.ShortenID(targetContainer.ID), app.Server))
|
||||||
|
|
||||||
|
toTarOpts := &archive.TarOptions{NoOverwriteDirNonDir: true, Compression: archive.Gzip}
|
||||||
|
content, err := archive.TarWithOptions(abraSh, toTarOpts)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
copyOpts := containertypes.CopyToContainerOptions{AllowOverwriteDirWithFile: false, CopyUIDGID: false}
|
||||||
|
if err := cl.CopyToContainer(context.Background(), targetContainer.ID, "/tmp", content, copyOpts); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME: avoid instantiating a new CLI
|
||||||
|
dcli, err := command.NewDockerCli()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
shell := "/bin/bash"
|
||||||
|
findShell := []string{"test", "-e", shell}
|
||||||
|
execCreateOpts := containertypes.ExecOptions{
|
||||||
|
AttachStderr: true,
|
||||||
|
AttachStdin: true,
|
||||||
|
AttachStdout: true,
|
||||||
|
Cmd: findShell,
|
||||||
|
Detach: false,
|
||||||
|
Tty: false,
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil {
|
||||||
|
log.Info(i18n.G("%s does not exist for %s, use /bin/sh as fallback", shell, app.Name))
|
||||||
|
shell = "/bin/sh"
|
||||||
|
}
|
||||||
|
|
||||||
|
var cmd []string
|
||||||
|
if cmdArgs != "" {
|
||||||
|
cmd = []string{shell, "-c", fmt.Sprintf("TARGET=%s; APP_NAME=%s; STACK_NAME=%s; . /tmp/abra.sh; %s %s", serviceName, app.Name, app.StackName(), cmdName, cmdArgs)}
|
||||||
|
} else {
|
||||||
|
cmd = []string{shell, "-c", fmt.Sprintf("TARGET=%s; APP_NAME=%s; STACK_NAME=%s; . /tmp/abra.sh; %s", serviceName, app.Name, app.StackName(), cmdName)}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("running command: %s", strings.Join(cmd, " ")))
|
||||||
|
|
||||||
|
if remoteUser != "" {
|
||||||
|
log.Debug(i18n.G("running command with user %s", remoteUser))
|
||||||
|
execCreateOpts.User = remoteUser
|
||||||
|
}
|
||||||
|
|
||||||
|
execCreateOpts.Cmd = cmd
|
||||||
|
|
||||||
|
execCreateOpts.Tty = true
|
||||||
|
if disableTTY {
|
||||||
|
execCreateOpts.Tty = false
|
||||||
|
log.Debug(i18n.G("not requesting a remote TTY"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := container.RunExec(dcli, cl, targetContainer.ID, &execCreateOpts); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func EnsureCommand(abraSh, recipeName, execCmd string) error {
|
||||||
|
bytes, err := ioutil.ReadFile(abraSh)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !strings.Contains(string(bytes), execCmd) {
|
||||||
|
return errors.New(i18n.G("%s doesn't have a %s function", recipeName, execCmd))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RunCmd runs a shell command and streams stdout/stderr in real-time.
|
||||||
|
func RunCmd(cmd *exec.Cmd) error {
|
||||||
|
r, err := cmd.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd.Stderr = cmd.Stdout
|
||||||
|
done := make(chan struct{})
|
||||||
|
scanner := bufio.NewScanner(r)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
fmt.Println(line)
|
||||||
|
}
|
||||||
|
done <- struct{}{}
|
||||||
|
}()
|
||||||
|
|
||||||
|
if err := cmd.Start(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
<-done
|
||||||
|
|
||||||
|
if err := cmd.Wait(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,318 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/tagcmp"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/charmbracelet/lipgloss"
|
||||||
|
dockerClient "github.com/docker/docker/client"
|
||||||
|
)
|
||||||
|
|
||||||
|
var borderStyle = lipgloss.NewStyle().
|
||||||
|
BorderStyle(lipgloss.ThickBorder()).
|
||||||
|
Padding(0, 1, 0, 1).
|
||||||
|
MaxWidth(79).
|
||||||
|
BorderForeground(lipgloss.Color("63"))
|
||||||
|
|
||||||
|
var headerStyle = lipgloss.NewStyle().
|
||||||
|
Underline(true).
|
||||||
|
Bold(true).
|
||||||
|
PaddingBottom(1)
|
||||||
|
|
||||||
|
var leftStyle = lipgloss.NewStyle().
|
||||||
|
Bold(true)
|
||||||
|
|
||||||
|
var rightStyle = lipgloss.NewStyle()
|
||||||
|
|
||||||
|
// horizontal is a JoinHorizontal helper function.
|
||||||
|
func horizontal(left, mid, right string) string {
|
||||||
|
return lipgloss.JoinHorizontal(lipgloss.Left, left, mid, right)
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatComposeFiles(composeFiles string) string {
|
||||||
|
return strings.ReplaceAll(composeFiles, ":", "\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeployOverview shows a deployment overview
|
||||||
|
func DeployOverview(
|
||||||
|
app appPkg.App,
|
||||||
|
deployedVersion string,
|
||||||
|
toDeployVersion string,
|
||||||
|
releaseNotes string,
|
||||||
|
warnMessages []string,
|
||||||
|
secrets []string,
|
||||||
|
configs []string,
|
||||||
|
images []string,
|
||||||
|
) error {
|
||||||
|
deployConfig := "compose.yml"
|
||||||
|
if composeFiles, ok := app.Env["COMPOSE_FILE"]; ok {
|
||||||
|
deployConfig = formatComposeFiles(composeFiles)
|
||||||
|
}
|
||||||
|
|
||||||
|
server := app.Server
|
||||||
|
if app.Server == "default" {
|
||||||
|
server = "local"
|
||||||
|
}
|
||||||
|
|
||||||
|
domain := fmt.Sprintf("https://%s", app.Domain)
|
||||||
|
if domain == "" {
|
||||||
|
domain = config.MISSING_DEFAULT
|
||||||
|
}
|
||||||
|
|
||||||
|
envVersion := app.Recipe.EnvVersionRaw
|
||||||
|
if envVersion == "" {
|
||||||
|
envVersion = config.MISSING_DEFAULT
|
||||||
|
}
|
||||||
|
|
||||||
|
rows := [][]string{
|
||||||
|
{i18n.G("DOMAIN"), domain},
|
||||||
|
{i18n.G("RECIPE"), app.Recipe.Name},
|
||||||
|
{i18n.G("SERVER"), server},
|
||||||
|
{i18n.G("CONFIG"), deployConfig},
|
||||||
|
{"", ""},
|
||||||
|
{i18n.G("CURRENT DEPLOYMENT"), formatter.BoldDirtyDefault(deployedVersion)},
|
||||||
|
{i18n.G("ENV VERSION"), formatter.BoldDirtyDefault(envVersion)},
|
||||||
|
{i18n.G("NEW DEPLOYMENT"), formatter.BoldDirtyDefault(toDeployVersion)},
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(images) > 0 {
|
||||||
|
imageRows := [][]string{
|
||||||
|
{"", ""},
|
||||||
|
{i18n.G("IMAGES"), strings.Join(images, "\n")},
|
||||||
|
}
|
||||||
|
rows = append(rows, imageRows...)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(secrets) > 0 {
|
||||||
|
secretsRows := [][]string{
|
||||||
|
{"", ""},
|
||||||
|
{i18n.G("SECRETS"), strings.Join(secrets, "\n")},
|
||||||
|
}
|
||||||
|
rows = append(rows, secretsRows...)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(configs) > 0 {
|
||||||
|
configsRows := [][]string{
|
||||||
|
{"", ""},
|
||||||
|
{i18n.G("CONFIGS"), strings.Join(configs, "\n")},
|
||||||
|
}
|
||||||
|
rows = append(rows, configsRows...)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployType := getDeployType(deployedVersion, toDeployVersion)
|
||||||
|
overview := formatter.CreateOverview(i18n.G("%s OVERVIEW", deployType), rows)
|
||||||
|
|
||||||
|
fmt.Println(overview)
|
||||||
|
|
||||||
|
if releaseNotes != "" {
|
||||||
|
fmt.Print(releaseNotes)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, msg := range warnMessages {
|
||||||
|
log.Warn(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
if NoInput {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
response := false
|
||||||
|
prompt := &survey.Confirm{Message: i18n.G("proceed?")}
|
||||||
|
if err := survey.AskOne(prompt, &response); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !response {
|
||||||
|
log.Fatal(i18n.G("deployment cancelled"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getDeployType(currentVersion, newVersion string) string {
|
||||||
|
if newVersion == config.MISSING_DEFAULT {
|
||||||
|
return i18n.G("UNDEPLOY")
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.Contains(newVersion, "+U") {
|
||||||
|
return i18n.G("CHAOS DEPLOY")
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.Contains(currentVersion, "+U") {
|
||||||
|
return i18n.G("UNCHAOS DEPLOY")
|
||||||
|
}
|
||||||
|
|
||||||
|
if currentVersion == newVersion {
|
||||||
|
return ("REDEPLOY")
|
||||||
|
}
|
||||||
|
|
||||||
|
if currentVersion == config.MISSING_DEFAULT {
|
||||||
|
return i18n.G("NEW DEPLOY")
|
||||||
|
}
|
||||||
|
|
||||||
|
currentParsed, err := tagcmp.Parse(currentVersion)
|
||||||
|
if err != nil {
|
||||||
|
return i18n.G("DEPLOY")
|
||||||
|
}
|
||||||
|
|
||||||
|
newParsed, err := tagcmp.Parse(newVersion)
|
||||||
|
if err != nil {
|
||||||
|
return i18n.G("DEPLOY")
|
||||||
|
}
|
||||||
|
|
||||||
|
if currentParsed.IsLessThan(newParsed) {
|
||||||
|
return i18n.G("UPGRADE")
|
||||||
|
}
|
||||||
|
|
||||||
|
return i18n.G("DOWNGRADE")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MoveOverview shows a overview before moving an app to a different server
|
||||||
|
func MoveOverview(
|
||||||
|
app appPkg.App,
|
||||||
|
newServer string,
|
||||||
|
secrets []string,
|
||||||
|
volumes []string,
|
||||||
|
) {
|
||||||
|
server := app.Server
|
||||||
|
if app.Server == "default" {
|
||||||
|
server = "local"
|
||||||
|
}
|
||||||
|
|
||||||
|
domain := app.Domain
|
||||||
|
if domain == "" {
|
||||||
|
domain = config.MISSING_DEFAULT
|
||||||
|
}
|
||||||
|
|
||||||
|
secretsOverview := strings.Join(secrets, "\n")
|
||||||
|
if len(secrets) == 0 {
|
||||||
|
secretsOverview = config.MISSING_DEFAULT
|
||||||
|
}
|
||||||
|
|
||||||
|
volumesOverview := strings.Join(volumes, "\n")
|
||||||
|
if len(volumes) == 0 {
|
||||||
|
volumesOverview = config.MISSING_DEFAULT
|
||||||
|
}
|
||||||
|
|
||||||
|
rows := [][]string{
|
||||||
|
{i18n.G("DOMAIN"), domain},
|
||||||
|
{i18n.G("RECIPE"), app.Recipe.Name},
|
||||||
|
{i18n.G("OLD SERVER"), server},
|
||||||
|
{i18n.G("NEW SERVER"), newServer},
|
||||||
|
{i18n.G("SECRETS"), secretsOverview},
|
||||||
|
{i18n.G("VOLUMES"), volumesOverview},
|
||||||
|
}
|
||||||
|
|
||||||
|
overview := formatter.CreateOverview(i18n.G("MOVE OVERVIEW"), rows)
|
||||||
|
|
||||||
|
fmt.Println(overview)
|
||||||
|
}
|
||||||
|
|
||||||
|
func PromptProcced() error {
|
||||||
|
if NoInput {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if Dry {
|
||||||
|
return errors.New(i18n.G("dry run"))
|
||||||
|
}
|
||||||
|
|
||||||
|
response := false
|
||||||
|
prompt := &survey.Confirm{Message: i18n.G("proceed?")}
|
||||||
|
if err := survey.AskOne(prompt, &response); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !response {
|
||||||
|
return errors.New(i18n.G("cancelled"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// PostCmds parses a string of commands and executes them inside of the respective services
|
||||||
|
// the commands string must have the following format:
|
||||||
|
// "<service> <command> <arguments>|<service> <command> <arguments>|... "
|
||||||
|
func PostCmds(cl *dockerClient.Client, app appPkg.App, commands string) error {
|
||||||
|
if _, err := os.Stat(app.Recipe.AbraShPath); err != nil {
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
return errors.New(i18n.G("%s does not exist for %s?", app.Recipe.AbraShPath, app.Name))
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, command := range strings.Split(commands, "|") {
|
||||||
|
commandParts := strings.Split(command, " ")
|
||||||
|
if len(commandParts) < 2 {
|
||||||
|
return errors.New(i18n.G("not enough arguments: %s", command))
|
||||||
|
}
|
||||||
|
targetServiceName := commandParts[0]
|
||||||
|
cmdName := commandParts[1]
|
||||||
|
parsedCmdArgs := ""
|
||||||
|
if len(commandParts) > 2 {
|
||||||
|
parsedCmdArgs = fmt.Sprintf("%s ", strings.Join(commandParts[2:], " "))
|
||||||
|
}
|
||||||
|
log.Info(i18n.G("running post-command '%s %s' in container %s", cmdName, parsedCmdArgs, targetServiceName))
|
||||||
|
|
||||||
|
if err := EnsureCommand(app.Recipe.AbraShPath, app.Recipe.Name, cmdName); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
serviceNames, err := appPkg.GetAppServiceNames(app.Name)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
matchingServiceName := false
|
||||||
|
for _, serviceName := range serviceNames {
|
||||||
|
if serviceName == targetServiceName {
|
||||||
|
matchingServiceName = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !matchingServiceName {
|
||||||
|
return fmt.Errorf("no service %s for %s?", targetServiceName, app.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("running command %s %s within the context of %s_%s", cmdName, parsedCmdArgs, app.StackName(), targetServiceName))
|
||||||
|
|
||||||
|
requestTTY := true
|
||||||
|
if err := RunCmdRemote(
|
||||||
|
cl,
|
||||||
|
app,
|
||||||
|
requestTTY,
|
||||||
|
app.Recipe.AbraShPath, targetServiceName, cmdName, parsedCmdArgs, ""); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SortVersionsDesc sorts versions in descending order.
|
||||||
|
func SortVersionsDesc(versions []string) []string {
|
||||||
|
var tags []tagcmp.Tag
|
||||||
|
|
||||||
|
for _, v := range versions {
|
||||||
|
parsed, _ := tagcmp.Parse(v) // skips unsupported tags
|
||||||
|
tags = append(tags, parsed)
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Sort(tagcmp.ByTagDesc(tags))
|
||||||
|
|
||||||
|
var desc []string
|
||||||
|
for _, t := range tags {
|
||||||
|
desc = append(desc, t.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
return desc
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestSortVersionsDesc(t *testing.T) {
|
||||||
|
versions := SortVersionsDesc([]string{
|
||||||
|
"0.2.3+1.2.2",
|
||||||
|
"1.0.0+2.2.2",
|
||||||
|
})
|
||||||
|
|
||||||
|
assert.Equal(t, "1.0.0+2.2.2", versions[0])
|
||||||
|
assert.Equal(t, "0.2.3+1.2.2", versions[1])
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import "coopcloud.tech/abra/pkg/recipe"
|
||||||
|
|
||||||
|
func GetEnsureContext() recipe.EnsureContext {
|
||||||
|
return recipe.EnsureContext{
|
||||||
|
Chaos,
|
||||||
|
Offline,
|
||||||
|
DeployLatest,
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,118 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/distribution/reference"
|
||||||
|
)
|
||||||
|
|
||||||
|
// PromptBumpType prompts for version bump type
|
||||||
|
func PromptBumpType(tagString, latestRelease, changeOverview string) error {
|
||||||
|
if (!Major && !Minor && !Patch) && tagString == "" {
|
||||||
|
fmt.Print(i18n.G(`
|
||||||
|
You need to make a decision about what kind of an update this new recipe
|
||||||
|
version is. If someone else performs this upgrade, do they have to do some
|
||||||
|
migration work or take care of some breaking changes? This can be signaled in
|
||||||
|
the version you specify on the recipe deploy label and is called a semantic
|
||||||
|
version.
|
||||||
|
|
||||||
|
The latest published version is %s.
|
||||||
|
|
||||||
|
%s
|
||||||
|
|
||||||
|
Here is a semver cheat sheet (more on https://semver.org):
|
||||||
|
|
||||||
|
major: new features/bug fixes, backwards incompatible (e.g 1.0.0 -> 2.0.0).
|
||||||
|
the upgrade won't work without some preparation work and others need
|
||||||
|
to take care when performing it. "it could go wrong".
|
||||||
|
|
||||||
|
minor: new features/bug fixes, backwards compatible (e.g. 0.1.0 -> 0.2.0).
|
||||||
|
the upgrade should Just Work and there are no breaking changes in
|
||||||
|
the app and the recipe config. "it should go fine".
|
||||||
|
|
||||||
|
patch: bug fixes, backwards compatible (e.g. 0.0.1 -> 0.0.2). this upgrade
|
||||||
|
should also Just Work and is mostly to do with minor bug fixes
|
||||||
|
and/or security patches. "nothing to worry about".
|
||||||
|
|
||||||
|
`, latestRelease, changeOverview))
|
||||||
|
|
||||||
|
var chosenBumpType string
|
||||||
|
prompt := &survey.Select{
|
||||||
|
Message: fmt.Sprintf("select recipe version increment type"),
|
||||||
|
Options: []string{i18n.G("major"), i18n.G("minor"), i18n.G("patch")},
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := survey.AskOne(prompt, &chosenBumpType); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
SetBumpType(chosenBumpType)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetBumpType figures out which bump type is specified
|
||||||
|
func GetBumpType() string {
|
||||||
|
var bumpType string
|
||||||
|
|
||||||
|
if Major {
|
||||||
|
bumpType = i18n.G("major")
|
||||||
|
} else if Minor {
|
||||||
|
bumpType = i18n.G("minor")
|
||||||
|
} else if Patch {
|
||||||
|
bumpType = i18n.G("patch")
|
||||||
|
} else {
|
||||||
|
log.Fatal(i18n.G("no version bump type specififed?"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return bumpType
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetBumpType figures out which bump type is specified
|
||||||
|
func SetBumpType(bumpType string) {
|
||||||
|
if bumpType == i18n.G("major") {
|
||||||
|
Major = true
|
||||||
|
} else if bumpType == i18n.G("minor") {
|
||||||
|
Minor = true
|
||||||
|
} else if bumpType == i18n.G("patch") {
|
||||||
|
Patch = true
|
||||||
|
} else {
|
||||||
|
log.Fatal(i18n.G("no version bump type specififed?"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetMainAppImage retrieves the main 'app' image name
|
||||||
|
func GetMainAppImage(recipe recipe.Recipe) (string, error) {
|
||||||
|
var path string
|
||||||
|
|
||||||
|
config, err := recipe.GetComposeConfig(nil)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
for _, service := range config.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
img, err := reference.ParseNormalizedNamed(service.Image)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
path = reference.Path(img)
|
||||||
|
path = formatter.StripTagMeta(path)
|
||||||
|
|
||||||
|
return path, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if path == "" {
|
||||||
|
return path, errors.New(i18n.G("%s has no main 'app' service?", recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
return path, nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,177 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ValidateRecipe ensures the recipe arg is valid.
|
||||||
|
func ValidateRecipe(args []string, cmdName string) recipe.Recipe {
|
||||||
|
var recipeName string
|
||||||
|
if len(args) > 0 {
|
||||||
|
recipeName = args[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
var recipes []string
|
||||||
|
|
||||||
|
catl, err := recipe.ReadRecipeCatalogue(Offline)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
knownRecipes := make(map[string]bool)
|
||||||
|
for name := range catl {
|
||||||
|
knownRecipes[name] = true
|
||||||
|
}
|
||||||
|
|
||||||
|
localRecipes, err := recipe.GetRecipesLocal()
|
||||||
|
if err != nil {
|
||||||
|
log.Debug(i18n.G("can't read local recipes: %s", err))
|
||||||
|
} else {
|
||||||
|
for _, recipeLocal := range localRecipes {
|
||||||
|
if _, ok := knownRecipes[recipeLocal]; !ok {
|
||||||
|
knownRecipes[recipeLocal] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for recipeName := range knownRecipes {
|
||||||
|
recipes = append(recipes, recipeName)
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeName == "" && !NoInput {
|
||||||
|
prompt := &survey.Select{
|
||||||
|
Message: i18n.G("Select recipe"),
|
||||||
|
Options: recipes,
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(prompt, &recipeName); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeName == "" {
|
||||||
|
log.Fatal(i18n.G("no recipe name provided"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, ok := knownRecipes[recipeName]; !ok {
|
||||||
|
if !strings.Contains(recipeName, "/") {
|
||||||
|
log.Fatal(i18n.G("no recipe '%s' exists?", recipeName))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
chosenRecipe := recipe.Get(recipeName)
|
||||||
|
if err := chosenRecipe.EnsureExists(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = chosenRecipe.GetComposeConfig(nil)
|
||||||
|
if err != nil {
|
||||||
|
if cmdName == i18n.G("generate") {
|
||||||
|
if strings.Contains(err.Error(), "missing a compose") {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
log.Warn(err)
|
||||||
|
} else {
|
||||||
|
if strings.Contains(err.Error(), "template_driver is not allowed") {
|
||||||
|
log.Warn(i18n.G("ensure %s recipe compose.* files include \"version: '3.8'\"", recipeName))
|
||||||
|
}
|
||||||
|
log.Fatal(i18n.G("unable to validate recipe: %s", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("validated %s as recipe argument", recipeName))
|
||||||
|
|
||||||
|
return chosenRecipe
|
||||||
|
}
|
||||||
|
|
||||||
|
// ValidateApp ensures the app name arg is valid.
|
||||||
|
func ValidateApp(args []string) app.App {
|
||||||
|
if len(args) == 0 {
|
||||||
|
log.Fatal(i18n.G("no app provided"))
|
||||||
|
}
|
||||||
|
|
||||||
|
appName := args[0]
|
||||||
|
|
||||||
|
app, err := app.Get(appName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("validated %s as app argument", appName))
|
||||||
|
|
||||||
|
return app
|
||||||
|
}
|
||||||
|
|
||||||
|
// ValidateDomain ensures the domain name arg is valid.
|
||||||
|
func ValidateDomain(args []string) string {
|
||||||
|
var domainName string
|
||||||
|
if len(args) > 0 {
|
||||||
|
domainName = args[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
if domainName == "" && !NoInput {
|
||||||
|
prompt := &survey.Input{
|
||||||
|
Message: i18n.G("Specify a domain name"),
|
||||||
|
Default: "1312.net",
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(prompt, &domainName); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if domainName == "" {
|
||||||
|
log.Fatal(i18n.G("no domain provided"))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("validated %s as domain argument", domainName))
|
||||||
|
|
||||||
|
return domainName
|
||||||
|
}
|
||||||
|
|
||||||
|
// ValidateServer ensures the server name arg is valid.
|
||||||
|
func ValidateServer(args []string) string {
|
||||||
|
var serverName string
|
||||||
|
if len(args) > 0 {
|
||||||
|
serverName = args[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
serverNames, err := config.ReadServerNames()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if serverName == "" && !NoInput {
|
||||||
|
prompt := &survey.Select{
|
||||||
|
Message: i18n.G("Specify a server name"),
|
||||||
|
Options: serverNames,
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(prompt, &serverName); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
matched := false
|
||||||
|
for _, name := range serverNames {
|
||||||
|
if name == serverName {
|
||||||
|
matched = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if serverName == "" {
|
||||||
|
log.Fatal(i18n.G("no server provided"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if !matched {
|
||||||
|
log.Fatal(i18n.G("server doesn't exist?"))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("validated %s as server argument", serverName))
|
||||||
|
|
||||||
|
return serverName
|
||||||
|
}
|
||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
gitPkg "coopcloud.tech/abra/pkg/git"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra recipe diff` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var recipeDiffAliases = i18n.G("d")
|
||||||
|
|
||||||
|
var RecipeDiffCommand = &cobra.Command{
|
||||||
|
// translators: `recipe diff` command
|
||||||
|
Use: i18n.G("diff <recipe> [flags]"),
|
||||||
|
Aliases: strings.Split(recipeDiffAliases, ","),
|
||||||
|
// translators: Short description for `recipe diff` command
|
||||||
|
Short: i18n.G("Show unstaged changes in recipe config"),
|
||||||
|
Long: i18n.G("This command requires /usr/bin/git."),
|
||||||
|
Args: cobra.MinimumNArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
r := internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
if err := gitPkg.DiffUnstaged(r.Dir); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
@@ -0,0 +1,142 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/go-git/go-git/v5"
|
||||||
|
gitCfg "github.com/go-git/go-git/v5/config"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra recipe fetch` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var recipeFetchAliases = i18n.G("f")
|
||||||
|
|
||||||
|
var RecipeFetchCommand = &cobra.Command{
|
||||||
|
// translators: `recipe fetch` command
|
||||||
|
Use: i18n.G("fetch [recipe | --all] [flags]"),
|
||||||
|
Aliases: strings.Split(recipeFetchAliases, ","),
|
||||||
|
// translators: Short description for `recipe fetch` command
|
||||||
|
Short: i18n.G("Clone recipe(s) locally"),
|
||||||
|
Long: i18n.G(`Using "--force/-f" Git syncs an existing recipe. It does not erase unstaged changes.`),
|
||||||
|
Args: cobra.RangeArgs(0, 1),
|
||||||
|
Example: i18n.G(` # fetch from recipe catalogue
|
||||||
|
abra recipe fetch gitea
|
||||||
|
|
||||||
|
# fetch from remote recipe
|
||||||
|
abra recipe fetch git.foo.org/recipes/myrecipe
|
||||||
|
|
||||||
|
# fetch with ssh remote for hacking
|
||||||
|
abra recipe fetch gitea --ssh`),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
var recipeName string
|
||||||
|
if len(args) > 0 {
|
||||||
|
recipeName = args[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeName == "" && !fetchAllRecipes {
|
||||||
|
log.Fatal(i18n.G("missing [recipe] or --all/-a"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeName != "" && fetchAllRecipes {
|
||||||
|
log.Fatal(i18n.G("cannot use [recipe] and --all/-a together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeName != "" {
|
||||||
|
r := recipe.Get(recipeName)
|
||||||
|
if _, err := os.Stat(r.Dir); !os.IsNotExist(err) {
|
||||||
|
if !force {
|
||||||
|
log.Warn(i18n.G("%s is already fetched", r.Name))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
r = internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
|
||||||
|
if sshRemote {
|
||||||
|
if r.SSHURL == "" {
|
||||||
|
log.Warn(i18n.G("unable to discover SSH remote for %s", r.Name))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
repo, err := git.PlainOpen(r.Dir)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to open %s: %s", r.Dir, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = repo.DeleteRemote("origin"); err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to remove default remote in %s: %s", r.Dir, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := repo.CreateRemote(&gitCfg.RemoteConfig{
|
||||||
|
Name: "origin",
|
||||||
|
URLs: []string{r.SSHURL},
|
||||||
|
}); err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to set SSH remote in %s: %s", r.Dir, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
catalogue, err := recipe.ReadRecipeCatalogue(internal.Offline)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
catlBar := formatter.CreateProgressbar(len(catalogue), i18n.G("fetching latest recipes..."))
|
||||||
|
ensureCtx := internal.GetEnsureContext()
|
||||||
|
for recipeName := range catalogue {
|
||||||
|
r := recipe.Get(recipeName)
|
||||||
|
if err := r.Ensure(ensureCtx); err != nil {
|
||||||
|
log.Error(err)
|
||||||
|
}
|
||||||
|
catlBar.Add(1)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
fetchAllRecipes bool
|
||||||
|
sshRemote bool
|
||||||
|
force bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
RecipeFetchCommand.Flags().BoolVarP(
|
||||||
|
&fetchAllRecipes,
|
||||||
|
i18n.G("all"),
|
||||||
|
i18n.GC("a", "recipe fetch"),
|
||||||
|
false,
|
||||||
|
i18n.G("fetch all recipes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeFetchCommand.Flags().BoolVarP(
|
||||||
|
&sshRemote,
|
||||||
|
i18n.G("ssh"),
|
||||||
|
i18n.G("s"),
|
||||||
|
false,
|
||||||
|
i18n.G("automatically set ssh remote"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeFetchCommand.Flags().BoolVarP(
|
||||||
|
&force,
|
||||||
|
i18n.G("force"),
|
||||||
|
i18n.G("f"),
|
||||||
|
false,
|
||||||
|
i18n.G("force re-fetch"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,149 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/lint"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra recipe lint` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var recipeLintAliases = i18n.G("l")
|
||||||
|
|
||||||
|
var RecipeLintCommand = &cobra.Command{
|
||||||
|
// translators: `recipe lint` command
|
||||||
|
Use: i18n.G("lint <recipe> [flags]"),
|
||||||
|
// translators: Short description for `recipe lint` command
|
||||||
|
Short: i18n.G("Lint a recipe"),
|
||||||
|
Aliases: strings.Split(recipeLintAliases, ","),
|
||||||
|
Args: cobra.MinimumNArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
recipe := internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
|
||||||
|
if err := recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{
|
||||||
|
i18n.G("ref"),
|
||||||
|
i18n.G("rule"),
|
||||||
|
i18n.G("severity"),
|
||||||
|
i18n.G("satisfied"),
|
||||||
|
i18n.G("skipped"),
|
||||||
|
i18n.G("resolve"),
|
||||||
|
}
|
||||||
|
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Headers(headers...)
|
||||||
|
|
||||||
|
hasError := false
|
||||||
|
var rows [][]string
|
||||||
|
var warnMessages []string
|
||||||
|
for level := range lint.LintRules {
|
||||||
|
for _, rule := range lint.LintRules[level] {
|
||||||
|
if onlyError && rule.Level != "error" {
|
||||||
|
log.Debug(i18n.G("skipping %s, does not have level \"error\"", rule.Ref))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
skipped := false
|
||||||
|
if rule.Skip(recipe) {
|
||||||
|
skipped = true
|
||||||
|
}
|
||||||
|
|
||||||
|
skippedOutput := "-"
|
||||||
|
if skipped {
|
||||||
|
skippedOutput = "✅"
|
||||||
|
}
|
||||||
|
|
||||||
|
satisfied := false
|
||||||
|
if !skipped {
|
||||||
|
ok, err := rule.Function(recipe)
|
||||||
|
if err != nil {
|
||||||
|
warnMessages = append(warnMessages, err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if !ok && rule.Level == i18n.G("error") {
|
||||||
|
hasError = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if ok {
|
||||||
|
satisfied = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
satisfiedOutput := "✅"
|
||||||
|
if !satisfied {
|
||||||
|
satisfiedOutput = "❌"
|
||||||
|
if skipped {
|
||||||
|
satisfiedOutput = "-"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
row := []string{
|
||||||
|
rule.Ref,
|
||||||
|
rule.Description,
|
||||||
|
rule.Level,
|
||||||
|
satisfiedOutput,
|
||||||
|
skippedOutput,
|
||||||
|
rule.HowToResolve,
|
||||||
|
}
|
||||||
|
|
||||||
|
rows = append(rows, row)
|
||||||
|
table.Row(row...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(rows) > 0 {
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, warnMsg := range warnMessages {
|
||||||
|
log.Warn(warnMsg)
|
||||||
|
}
|
||||||
|
|
||||||
|
if hasError {
|
||||||
|
log.Warn(i18n.G("critical errors present in %s config", recipe.Name))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
onlyError bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
RecipeLintCommand.Flags().BoolVarP(
|
||||||
|
&internal.Chaos,
|
||||||
|
i18n.G("chaos"),
|
||||||
|
i18n.G("C"),
|
||||||
|
false,
|
||||||
|
i18n.G("ignore uncommitted recipes changes"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeLintCommand.Flags().BoolVarP(
|
||||||
|
&onlyError,
|
||||||
|
i18n.G("error"),
|
||||||
|
i18n.G("e"),
|
||||||
|
false,
|
||||||
|
i18n.G("only show errors"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,116 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra recipe list` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var recipeListAliases = i18n.G("ls")
|
||||||
|
|
||||||
|
var RecipeListCommand = &cobra.Command{
|
||||||
|
// translators: `recipe list` command
|
||||||
|
Use: i18n.G("list"),
|
||||||
|
// translators: Short description for `recipe list` command
|
||||||
|
Short: i18n.G("List recipes"),
|
||||||
|
Aliases: strings.Split(recipeListAliases, ","),
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
catl, err := recipe.ReadRecipeCatalogue(internal.Offline)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
recipes := catl.Flatten()
|
||||||
|
sort.Sort(recipe.ByRecipeName(recipes))
|
||||||
|
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{
|
||||||
|
i18n.G("name"),
|
||||||
|
i18n.G("category"),
|
||||||
|
i18n.G("status"),
|
||||||
|
i18n.G("healthcheck"),
|
||||||
|
i18n.G("backups"),
|
||||||
|
i18n.G("email"),
|
||||||
|
i18n.G("tests"),
|
||||||
|
i18n.G("SSO"),
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Headers(headers...)
|
||||||
|
|
||||||
|
var rows [][]string
|
||||||
|
for _, recipe := range recipes {
|
||||||
|
row := []string{
|
||||||
|
recipe.Name,
|
||||||
|
recipe.Category,
|
||||||
|
strconv.Itoa(recipe.Features.Status),
|
||||||
|
recipe.Features.Healthcheck,
|
||||||
|
recipe.Features.Backups,
|
||||||
|
recipe.Features.Email,
|
||||||
|
recipe.Features.Tests,
|
||||||
|
recipe.Features.SSO,
|
||||||
|
}
|
||||||
|
|
||||||
|
if pattern != "" {
|
||||||
|
if strings.Contains(recipe.Name, pattern) {
|
||||||
|
table.Row(row...)
|
||||||
|
rows = append(rows, row)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
table.Row(row...)
|
||||||
|
rows = append(rows, row)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(rows) > 0 {
|
||||||
|
if internal.MachineReadable {
|
||||||
|
out, err := formatter.ToJSON(headers, rows)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to render to JSON: %s", err))
|
||||||
|
}
|
||||||
|
fmt.Println(out)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
pattern string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
RecipeListCommand.Flags().BoolVarP(
|
||||||
|
&internal.MachineReadable,
|
||||||
|
i18n.G("machine"),
|
||||||
|
i18n.G("m"),
|
||||||
|
false,
|
||||||
|
i18n.G("print machine-readable output"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeListCommand.Flags().StringVarP(
|
||||||
|
&pattern,
|
||||||
|
i18n.G("pattern"),
|
||||||
|
i18n.G("p"),
|
||||||
|
"",
|
||||||
|
i18n.G("filter by recipe"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,135 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"strings"
|
||||||
|
"text/template"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/git"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// recipeMetadata is the recipe metadata for the README.md
|
||||||
|
type recipeMetadata struct {
|
||||||
|
Name string
|
||||||
|
Description string
|
||||||
|
Category string
|
||||||
|
Status string
|
||||||
|
Image string
|
||||||
|
Healthcheck string
|
||||||
|
Backups string
|
||||||
|
Email string
|
||||||
|
Tests string
|
||||||
|
SSO string
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra recipe new` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var recipeNewAliases = i18n.G("n")
|
||||||
|
|
||||||
|
var RecipeNewCommand = &cobra.Command{
|
||||||
|
// translators: `recipe new` command
|
||||||
|
Use: i18n.G("new <recipe> [flags]"),
|
||||||
|
Aliases: strings.Split(recipeNewAliases, ","),
|
||||||
|
// translators: Short description for `abra recipe new` command
|
||||||
|
Short: i18n.G("Create a new recipe"),
|
||||||
|
Long: i18n.G(`A community managed recipe template is used.`),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
recipeName := args[0]
|
||||||
|
|
||||||
|
r := recipe.Get(recipeName)
|
||||||
|
if _, err := os.Stat(r.Dir); !os.IsNotExist(err) {
|
||||||
|
log.Fatal(i18n.G("%s recipe directory already exists?", r.Dir))
|
||||||
|
}
|
||||||
|
|
||||||
|
url := i18n.G("%s/example.git", config.REPOS_BASE_URL)
|
||||||
|
if err := git.Clone(r.Dir, url); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
gitRepo := path.Join(r.Dir, ".git")
|
||||||
|
if err := os.RemoveAll(gitRepo); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
log.Debug(i18n.G("removed .git repo in %s", gitRepo))
|
||||||
|
|
||||||
|
meta := newRecipeMeta(recipeName)
|
||||||
|
|
||||||
|
for _, path := range []string{r.ReadmePath, r.SampleEnvPath} {
|
||||||
|
tpl, err := template.ParseFiles(path)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var templated bytes.Buffer
|
||||||
|
if err := tpl.Execute(&templated, meta); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := os.WriteFile(path, templated.Bytes(), 0o644); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := git.Init(r.Dir, true, gitName, gitEmail); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("new recipe '%s' created: %s", recipeName, path.Join(r.Dir)))
|
||||||
|
log.Info(i18n.G("happy hacking 🎉"))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// newRecipeMeta creates a new recipeMetadata instance with defaults
|
||||||
|
func newRecipeMeta(recipeName string) recipeMetadata {
|
||||||
|
return recipeMetadata{
|
||||||
|
Name: recipeName,
|
||||||
|
Description: "> One line description of the recipe",
|
||||||
|
Category: "Apps",
|
||||||
|
Status: "0",
|
||||||
|
Image: fmt.Sprintf("[`%s`](https://hub.docker.com/r/%s), 4, upstream", recipeName, recipeName),
|
||||||
|
Healthcheck: "No",
|
||||||
|
Backups: "No",
|
||||||
|
Email: "No",
|
||||||
|
Tests: "No",
|
||||||
|
SSO: "No",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
gitName string
|
||||||
|
gitEmail string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
RecipeNewCommand.Flags().StringVarP(
|
||||||
|
&gitName,
|
||||||
|
i18n.G("git-name"),
|
||||||
|
i18n.G("N"),
|
||||||
|
"",
|
||||||
|
i18n.G("Git (user) name to do commits with"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeNewCommand.Flags().StringVarP(
|
||||||
|
&gitEmail,
|
||||||
|
i18n.G("git-email"),
|
||||||
|
i18n.G("e"),
|
||||||
|
"",
|
||||||
|
i18n.G("Git email name to do commits with"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra recipe` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var recipeAliases = i18n.G("r")
|
||||||
|
|
||||||
|
// RecipeCommand defines all recipe related sub-commands.
|
||||||
|
var RecipeCommand = &cobra.Command{
|
||||||
|
// translators: `recipe` command group
|
||||||
|
Use: i18n.G("recipe [cmd] [args] [flags]"),
|
||||||
|
Aliases: strings.Split(recipeAliases, ","),
|
||||||
|
// translators: Short description for `recipe` command group
|
||||||
|
Short: i18n.G("Manage recipes"),
|
||||||
|
Long: i18n.G(`A recipe is a blueprint for an app.
|
||||||
|
|
||||||
|
It is a bunch of config files which describe how to deploy and maintain an app.
|
||||||
|
Recipes are maintained by the Co-op Cloud community and you can use Abra to
|
||||||
|
read them, deploy them and create apps for you.
|
||||||
|
|
||||||
|
Anyone who uses a recipe can become a maintainer. Maintainers typically make
|
||||||
|
sure the recipe is in good working order and the config upgraded in a timely
|
||||||
|
manner.`),
|
||||||
|
}
|
||||||
@@ -0,0 +1,652 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
gitPkg "coopcloud.tech/abra/pkg/git"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
recipePkg "coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"coopcloud.tech/tagcmp"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/distribution/reference"
|
||||||
|
"github.com/go-git/go-git/v5"
|
||||||
|
"github.com/go-git/go-git/v5/plumbing"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Errors
|
||||||
|
var errEmptyVersionsInCatalogue = errors.New(i18n.G("catalogue versions list is unexpectedly empty"))
|
||||||
|
|
||||||
|
// translators: `abra recipe release` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var recipeReleaseAliases = i18n.G("rl")
|
||||||
|
|
||||||
|
var RecipeReleaseCommand = &cobra.Command{
|
||||||
|
// translators: `recipe release` command
|
||||||
|
Use: i18n.G("release <recipe> [version] [flags]"),
|
||||||
|
Aliases: strings.Split(recipeReleaseAliases, ","),
|
||||||
|
// translators: Short description for `recipe release` command
|
||||||
|
Short: i18n.G("Release a new recipe version"),
|
||||||
|
Long: i18n.G(`Create a new version of a recipe.
|
||||||
|
|
||||||
|
These versions are then published on the Co-op Cloud recipe catalogue. These
|
||||||
|
versions take the following form:
|
||||||
|
|
||||||
|
a.b.c+x.y.z
|
||||||
|
|
||||||
|
Where the "a.b.c" part is a semantic version determined by the maintainer. The
|
||||||
|
"x.y.z" part is the image tag of the recipe "app" service (the main container
|
||||||
|
which contains the software to be used, by naming convention).
|
||||||
|
|
||||||
|
We maintain a semantic versioning scheme ("a.b.c") alongside the recipe
|
||||||
|
versioning scheme ("x.y.z") in order to maximise the chances that the nature of
|
||||||
|
recipe updates are properly communicated. I.e. developers of an app might
|
||||||
|
publish a minor version but that might lead to changes in the recipe which are
|
||||||
|
major and therefore require intervention while doing the upgrade work.
|
||||||
|
|
||||||
|
This command will publish your new release to git.coopcloud.tech. This
|
||||||
|
requires that you have permission to git push to these repositories and have
|
||||||
|
your SSH keys configured on your account. Enable ssh-agent and make sure to add
|
||||||
|
your private key and enter your passphrase beforehand.
|
||||||
|
|
||||||
|
eval ` + "`ssh-agent`" + `
|
||||||
|
ssh-add ~/.ssh/<my-ssh-private-key-for-git-coopcloud-tech>`),
|
||||||
|
Example: ` # publish release
|
||||||
|
eval ` + "`ssh-agent`" + `
|
||||||
|
ssh-add ~/.ssh/id_ed25519
|
||||||
|
abra recipe release gitea`,
|
||||||
|
Args: cobra.RangeArgs(1, 2),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
switch l := len(args); l {
|
||||||
|
case 0:
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
case 1:
|
||||||
|
return autocomplete.RecipeVersionComplete(args[0])
|
||||||
|
default:
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
recipe := internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
|
||||||
|
imagesTmp, err := GetImageVersions(recipe)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
mainApp, err := internal.GetMainAppImage(recipe)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
mainAppVersion := imagesTmp[mainApp]
|
||||||
|
if mainAppVersion == "" {
|
||||||
|
log.Fatal(i18n.G("main app service version for %s is empty?", recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
repo, err := git.PlainOpen(recipe.Dir)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
preCommitHead, err := repo.Head()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
isClean, err := gitPkg.IsClean(recipe.Dir)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isClean {
|
||||||
|
log.Fatal(i18n.G("working directory not clean in %s, aborting", recipe.Dir))
|
||||||
|
}
|
||||||
|
|
||||||
|
tags, err := recipe.Tags()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var tagString string
|
||||||
|
if len(args) == 2 {
|
||||||
|
tagString = args[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
if (internal.Major || internal.Minor || internal.Patch) && tagString != "" {
|
||||||
|
log.Fatal(i18n.G("cannot specify tag and bump type at the same time"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(tags) == 0 && tagString == "" {
|
||||||
|
log.Warn(i18n.G("no git tags found for %s", recipe.Name))
|
||||||
|
if internal.NoInput {
|
||||||
|
log.Fatal(i18n.G("unable to continue, input required for initial version"))
|
||||||
|
}
|
||||||
|
fmt.Println(i18n.G(`
|
||||||
|
The following options are two types of initial semantic version that you can
|
||||||
|
pick for %s that will be published in the recipe catalogue. This follows the
|
||||||
|
semver convention (more on https://semver.org), here is a short cheatsheet
|
||||||
|
|
||||||
|
0.1.0: development release, still hacking. when you make a major upgrade
|
||||||
|
you increment the "y" part (i.e. 0.1.0 -> 0.2.0) and only move to
|
||||||
|
using the "x" part when things are stable.
|
||||||
|
|
||||||
|
1.0.0: public release, assumed to be working. you already have a stable
|
||||||
|
and reliable deployment of this app and feel relatively confident
|
||||||
|
about it.
|
||||||
|
|
||||||
|
If you want people to be able alpha test your current config for %s but don't
|
||||||
|
think it is quite reliable, go with 0.1.0 and people will know that things are
|
||||||
|
likely to change.
|
||||||
|
|
||||||
|
`, recipe.Name, recipe.Name))
|
||||||
|
var chosenVersion string
|
||||||
|
edPrompt := &survey.Select{
|
||||||
|
Message: i18n.G("which version do you want to begin with?"),
|
||||||
|
Options: []string{"0.1.0", "1.0.0"},
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := survey.AskOne(edPrompt, &chosenVersion); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
tagString = fmt.Sprintf("%s+%s", chosenVersion, mainAppVersion)
|
||||||
|
}
|
||||||
|
|
||||||
|
if tagString == "" && (!internal.Major && !internal.Minor && !internal.Patch) {
|
||||||
|
|
||||||
|
catl, err := recipePkg.ReadRecipeCatalogue(false)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
changesTable, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
latestRelease := tags[len(tags)-1]
|
||||||
|
latestRecipeVersion, err := getLatestVersion(recipe, catl)
|
||||||
|
if err != nil && err != errEmptyVersionsInCatalogue {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
changesTable.Headers(i18n.G("SERVICE"), latestRelease, i18n.G("PROPOSED CHANGES"))
|
||||||
|
|
||||||
|
allRecipeVersions := catl[recipe.Name].Versions
|
||||||
|
for _, recipeVersion := range allRecipeVersions {
|
||||||
|
if serviceVersions, ok := recipeVersion[latestRecipeVersion]; ok {
|
||||||
|
for serviceName := range serviceVersions {
|
||||||
|
serviceMeta := serviceVersions[serviceName]
|
||||||
|
|
||||||
|
existingImageTag := fmt.Sprintf("%s:%s", serviceMeta.Image, serviceMeta.Tag)
|
||||||
|
newImageTag := fmt.Sprintf("%s:%s", serviceMeta.Image, imagesTmp[serviceMeta.Image])
|
||||||
|
|
||||||
|
if existingImageTag == newImageTag {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
changesTable.Row([]string{serviceName, existingImageTag, newImageTag}...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
changeOverview := changesTable.Render()
|
||||||
|
|
||||||
|
if err := internal.PromptBumpType("", latestRelease, changeOverview); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if tagString == "" {
|
||||||
|
var lastGitTag tagcmp.Tag
|
||||||
|
iter, err := repo.Tags()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := iter.ForEach(func(ref *plumbing.Reference) error {
|
||||||
|
obj, err := repo.TagObject(ref.Hash())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("tag at commit %s is unannotated or otherwise broken", ref.Hash()))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
tagcmpTag, err := tagcmp.Parse(obj.Name)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lastGitTag == tagcmp.Tag{}) {
|
||||||
|
lastGitTag = tagcmpTag
|
||||||
|
} else if tagcmpTag.IsGreaterThan(lastGitTag) {
|
||||||
|
lastGitTag = tagcmpTag
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// bumpType is used to decide what part of the tag should be incremented
|
||||||
|
bumpType := btoi(internal.Major)*4 + btoi(internal.Minor)*2 + btoi(internal.Patch)
|
||||||
|
if bumpType != 0 {
|
||||||
|
// a bitwise check if the number is a power of 2
|
||||||
|
if (bumpType & (bumpType - 1)) != 0 {
|
||||||
|
log.Fatal(i18n.G("you can only use one version flag: --major, --minor or --patch"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
newTag := lastGitTag
|
||||||
|
if bumpType > 0 {
|
||||||
|
if internal.Patch {
|
||||||
|
now, err := strconv.Atoi(newTag.Patch)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
newTag.Patch = strconv.Itoa(now + 1)
|
||||||
|
} else if internal.Minor {
|
||||||
|
now, err := strconv.Atoi(newTag.Minor)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
newTag.Patch = "0"
|
||||||
|
newTag.Minor = strconv.Itoa(now + 1)
|
||||||
|
} else if internal.Major {
|
||||||
|
now, err := strconv.Atoi(newTag.Major)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
newTag.Patch = "0"
|
||||||
|
newTag.Minor = "0"
|
||||||
|
newTag.Major = strconv.Itoa(now + 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
newTag.Metadata = mainAppVersion
|
||||||
|
log.Debug(i18n.G("choosing %s as new version for %s", newTag.String(), recipe.Name))
|
||||||
|
tagString = newTag.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := tagcmp.Parse(tagString); err != nil {
|
||||||
|
log.Fatal(i18n.G("invalid version %s specified", tagString))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tag := range tags {
|
||||||
|
previousTagLeftHand := strings.Split(tag, "+")[0]
|
||||||
|
newTagStringLeftHand := strings.Split(tagString, "+")[0]
|
||||||
|
if previousTagLeftHand == newTagStringLeftHand {
|
||||||
|
log.Fatal(i18n.G("%s+... conflicts with a previous release: %s", newTagStringLeftHand, tag))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := createReleaseFromTag(recipe, tagString, mainAppVersion); err != nil {
|
||||||
|
if cleanErr := cleanTag(recipe, tagString); cleanErr != nil {
|
||||||
|
log.Fatal(i18n.G("unable to clean up tag after failed release attempt: %s", cleanErr))
|
||||||
|
}
|
||||||
|
if resetErr := resetCommit(recipe, preCommitHead); resetErr != nil {
|
||||||
|
log.Fatal(i18n.G("unable to reset commit after failed release attempt: %s", resetErr))
|
||||||
|
}
|
||||||
|
log.Error(err)
|
||||||
|
log.Fatal(i18n.G("release failed. any changes made have been reverted"))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetImageVersions retrieves image versions for a recipe
|
||||||
|
func GetImageVersions(recipe recipePkg.Recipe) (map[string]string, error) {
|
||||||
|
services := make(map[string]string)
|
||||||
|
|
||||||
|
config, err := recipe.GetComposeConfig(nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
missingTag := false
|
||||||
|
for _, service := range config.Services {
|
||||||
|
if service.Image == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
img, err := reference.ParseNormalizedNamed(service.Image)
|
||||||
|
if err != nil {
|
||||||
|
return services, err
|
||||||
|
}
|
||||||
|
|
||||||
|
path := reference.Path(img)
|
||||||
|
|
||||||
|
path = formatter.StripTagMeta(path)
|
||||||
|
|
||||||
|
var tag string
|
||||||
|
switch img.(type) {
|
||||||
|
case reference.NamedTagged:
|
||||||
|
tag = img.(reference.NamedTagged).Tag()
|
||||||
|
case reference.Named:
|
||||||
|
if service.Name == "app" {
|
||||||
|
missingTag = true
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
services[path] = tag
|
||||||
|
}
|
||||||
|
|
||||||
|
if missingTag {
|
||||||
|
return services, errors.New(i18n.G("app service is missing image tag?"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return services, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// createReleaseFromTag creates a new release based on a supplied recipe version string
|
||||||
|
func createReleaseFromTag(recipe recipePkg.Recipe, tagString, mainAppVersion string) error {
|
||||||
|
var err error
|
||||||
|
|
||||||
|
repo, err := git.PlainOpen(recipe.Dir)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
mainService := "app"
|
||||||
|
label := fmt.Sprintf("coop-cloud.${STACK_NAME}.version=%s", tagString)
|
||||||
|
if !internal.Dry {
|
||||||
|
if err := recipe.UpdateLabel("compose.y*ml", mainService, label); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.Info(i18n.G("dry run: not syncing label %s for recipe %s", tagString, recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := addReleaseNotes(recipe, tagString); err != nil {
|
||||||
|
return errors.New(i18n.G("failed to add release notes: %s", err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := commitRelease(recipe, tagString); err != nil {
|
||||||
|
return errors.New(i18n.G("failed to commit changes: %s", err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := tagRelease(tagString, repo); err != nil {
|
||||||
|
return errors.New(i18n.G("failed to tag release: %s", err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := pushRelease(recipe, tagString); err != nil {
|
||||||
|
return errors.New(i18n.G("failed to publish new release: %s", err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// btoi converts a boolean value into an integer
|
||||||
|
func btoi(b bool) int {
|
||||||
|
if b {
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// getTagCreateOptions constructs git tag create options
|
||||||
|
func getTagCreateOptions(tag string) (git.CreateTagOptions, error) {
|
||||||
|
msg := i18n.G("chore: publish %s release", tag)
|
||||||
|
return git.CreateTagOptions{Message: msg}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// addReleaseNotes checks if the release/next release note exists and moves the
|
||||||
|
// file to release/<tag>.
|
||||||
|
func addReleaseNotes(recipe recipePkg.Recipe, tag string) error {
|
||||||
|
releaseDir := path.Join(recipe.Dir, "release")
|
||||||
|
if _, err := os.Stat(releaseDir); errors.Is(err, os.ErrNotExist) {
|
||||||
|
if err := os.Mkdir(releaseDir, 0755); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tagReleaseNotePath := path.Join(releaseDir, tag)
|
||||||
|
if _, err := os.Stat(tagReleaseNotePath); err == nil {
|
||||||
|
// Release note for current tag already exist exists.
|
||||||
|
return nil
|
||||||
|
} else if !errors.Is(err, os.ErrNotExist) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
var addNextAsReleaseNotes bool
|
||||||
|
|
||||||
|
nextReleaseNotePath := path.Join(releaseDir, "next")
|
||||||
|
if _, err := os.Stat(nextReleaseNotePath); err == nil {
|
||||||
|
// release/next note exists. Move it to release/<tag>
|
||||||
|
if internal.Dry {
|
||||||
|
log.Debug(i18n.G("dry run: move release note from 'next' to %s", tag))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !internal.NoInput {
|
||||||
|
prompt := &survey.Confirm{
|
||||||
|
Message: i18n.G("use release note in release/next?"),
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := survey.AskOne(prompt, &addNextAsReleaseNotes); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !addNextAsReleaseNotes {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := os.Rename(nextReleaseNotePath, tagReleaseNotePath); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := gitPkg.Add(recipe.Dir, path.Join("release", "next"), internal.Dry); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := gitPkg.Add(recipe.Dir, path.Join("release", tag), internal.Dry); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else if !errors.Is(err, os.ErrNotExist) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE(d1): No release note exists for the current release. Or, we've
|
||||||
|
// already used release/next as the release note
|
||||||
|
if internal.NoInput || addNextAsReleaseNotes {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
prompt := &survey.Input{
|
||||||
|
Message: i18n.G("add release note? (leave empty to skip)"),
|
||||||
|
}
|
||||||
|
|
||||||
|
var releaseNote string
|
||||||
|
if err := survey.AskOne(prompt, &releaseNote); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if releaseNote == "" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := os.WriteFile(tagReleaseNotePath, []byte(releaseNote), 0o644); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := gitPkg.Add(recipe.Dir, path.Join("release", tag), internal.Dry); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func commitRelease(recipe recipePkg.Recipe, tag string) error {
|
||||||
|
if internal.Dry {
|
||||||
|
log.Debug(i18n.G("dry run: no changes committed"))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
isClean, err := gitPkg.IsClean(recipe.Dir)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if isClean {
|
||||||
|
if !internal.Dry {
|
||||||
|
return errors.New(i18n.G("no changes discovered in %s, nothing to publish?", recipe.Dir))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
msg := fmt.Sprintf("chore: publish %s release", tag)
|
||||||
|
if err := gitPkg.Commit(recipe.Dir, msg, internal.Dry); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func tagRelease(tagString string, repo *git.Repository) error {
|
||||||
|
if internal.Dry {
|
||||||
|
log.Debug(i18n.G("dry run: no git tag created (%s)", tagString))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
head, err := repo.Head()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
createTagOptions, err := getTagCreateOptions(tagString)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = repo.CreateTag(tagString, head.Hash(), &createTagOptions)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
hash := formatter.SmallSHA(head.Hash().String())
|
||||||
|
log.Debug(i18n.G("created tag %s at %s", tagString, hash))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func pushRelease(recipe recipePkg.Recipe, tagString string) error {
|
||||||
|
if internal.Dry {
|
||||||
|
log.Info(i18n.G("dry run: no changes published"))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if os.Getenv("SSH_AUTH_SOCK") == "" {
|
||||||
|
return errors.New(i18n.G("ssh-agent not found. see \"abra recipe release --help\" and try again"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := recipe.Push(internal.Dry); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
url := fmt.Sprintf("%s/src/tag/%s", recipe.GitURL, tagString)
|
||||||
|
log.Info(i18n.G("new release published: %s", url))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// resetCommit hard resets to the state before release was started.
|
||||||
|
// This will only remove changes made by the release process due to requiring
|
||||||
|
// a clean working directory.
|
||||||
|
func resetCommit(recipe recipePkg.Recipe, head *plumbing.Reference) error {
|
||||||
|
repo, err := git.PlainOpen(recipe.Dir)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("unable to open repo in %s: %s", recipe.Dir, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
worktree, err := repo.Worktree()
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("unable to open work tree in %s: %s", recipe.Dir, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := &git.ResetOptions{Commit: head.Hash(), Mode: git.HardReset}
|
||||||
|
if err := worktree.Reset(opts); err != nil {
|
||||||
|
return errors.New(i18n.G("unable to hard reset %s: %s", recipe.Dir, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("reset commit to pre-release state"))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// cleanTag removes a freshly created tag
|
||||||
|
func cleanTag(recipe recipePkg.Recipe, tag string) error {
|
||||||
|
repo, err := git.PlainOpen(recipe.Dir)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("unable to open repo in %s: %s", recipe.Dir, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := repo.DeleteTag(tag); err != nil {
|
||||||
|
if !strings.Contains(err.Error(), "not found") {
|
||||||
|
return errors.New(i18n.G("unable to delete tag %s: %s", tag, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("removed freshly created tag %s", tag))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getLatestVersion(recipe recipePkg.Recipe, catl recipePkg.RecipeCatalogue) (string, error) {
|
||||||
|
versions, err := recipePkg.GetRecipeCatalogueVersions(recipe.Name, catl)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if len(versions) > 0 {
|
||||||
|
return versions[len(versions)-1], nil
|
||||||
|
}
|
||||||
|
return "", errEmptyVersionsInCatalogue
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
RecipeReleaseCommand.Flags().BoolVarP(
|
||||||
|
&internal.Dry,
|
||||||
|
i18n.G("dry-run"),
|
||||||
|
i18n.G("r"),
|
||||||
|
false,
|
||||||
|
i18n.G("report changes that would be made"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeReleaseCommand.Flags().BoolVarP(
|
||||||
|
&internal.Major,
|
||||||
|
i18n.G("major"),
|
||||||
|
i18n.G("x"),
|
||||||
|
false,
|
||||||
|
i18n.G("increase the major part of the version"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeReleaseCommand.Flags().BoolVarP(
|
||||||
|
&internal.Minor,
|
||||||
|
i18n.G("minor"),
|
||||||
|
i18n.G("y"),
|
||||||
|
false,
|
||||||
|
i18n.G("increase the minor part of the version"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeReleaseCommand.Flags().BoolVarP(
|
||||||
|
&internal.Patch,
|
||||||
|
i18n.G("patch"),
|
||||||
|
i18n.G("z"),
|
||||||
|
false,
|
||||||
|
i18n.G("increase the patch part of the version"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
recipePkg "coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGetLatestVersionReturnsErrorWhenVersionsIsEmpty(t *testing.T) {
|
||||||
|
recipe := recipePkg.Recipe{}
|
||||||
|
catalogue := recipePkg.RecipeCatalogue{}
|
||||||
|
_, err := getLatestVersion(recipe, catalogue)
|
||||||
|
assert.Equal(t, err, errEmptyVersionsInCatalogue)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetLatestVersionReturnsLastVersion(t *testing.T) {
|
||||||
|
recipe := recipePkg.Recipe{
|
||||||
|
Name: "test",
|
||||||
|
}
|
||||||
|
versions := []map[string]map[string]recipePkg.ServiceMeta{
|
||||||
|
make(map[string]map[string]recipePkg.ServiceMeta),
|
||||||
|
make(map[string]map[string]recipePkg.ServiceMeta),
|
||||||
|
}
|
||||||
|
versions[0]["0.0.3"] = make(map[string]recipePkg.ServiceMeta)
|
||||||
|
versions[1]["0.0.2"] = make(map[string]recipePkg.ServiceMeta)
|
||||||
|
catalogue := make(recipePkg.RecipeCatalogue)
|
||||||
|
catalogue["test"] = recipePkg.RecipeMeta{
|
||||||
|
Versions: versions,
|
||||||
|
}
|
||||||
|
version, _ := getLatestVersion(recipe, catalogue)
|
||||||
|
assert.Equal(t, version, "0.0.3")
|
||||||
|
}
|
||||||
@@ -0,0 +1,55 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/go-git/go-git/v5"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra recipe reset` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var recipeResetAliases = i18n.G("rs")
|
||||||
|
|
||||||
|
var RecipeResetCommand = &cobra.Command{
|
||||||
|
// translators: `recipe reset` command
|
||||||
|
Use: i18n.G("reset <recipe> [flags]"),
|
||||||
|
Aliases: strings.Split(recipeResetAliases, ","),
|
||||||
|
// translators: Short description for `recipe reset` command
|
||||||
|
Short: i18n.G("Remove all unstaged changes from recipe config"),
|
||||||
|
Long: i18n.G("WARNING: this will delete your changes. Be Careful."),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
r := internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
|
||||||
|
repo, err := git.PlainOpen(r.Dir)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ref, err := repo.Head()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
worktree, err := repo.Worktree()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := &git.ResetOptions{Commit: ref.Hash(), Mode: git.HardReset}
|
||||||
|
if err := worktree.Reset(opts); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
@@ -0,0 +1,420 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
gitPkg "coopcloud.tech/abra/pkg/git"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
recipePkg "coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"coopcloud.tech/tagcmp"
|
||||||
|
"github.com/AlecAivazis/survey/v2"
|
||||||
|
"github.com/distribution/reference"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
type imgPin struct {
|
||||||
|
image string
|
||||||
|
version tagcmp.Tag
|
||||||
|
}
|
||||||
|
|
||||||
|
// anUpgrade represents a single service upgrade (as within a recipe), and the
|
||||||
|
// list of tags that it can be upgraded to, for serialization purposes.
|
||||||
|
type anUpgrade struct {
|
||||||
|
Service string `json:"service"`
|
||||||
|
Image string `json:"image"`
|
||||||
|
Tag string `json:"tag"`
|
||||||
|
UpgradeTags []string `json:"upgrades"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// translators: `abra recipe upgrade` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var recipeUpgradeAliases = i18n.G("u")
|
||||||
|
|
||||||
|
var RecipeUpgradeCommand = &cobra.Command{
|
||||||
|
// translators: `recipe upgrade` command
|
||||||
|
Use: i18n.G("upgrade <recipe> [flags]"),
|
||||||
|
Aliases: strings.Split(recipeUpgradeAliases, ","),
|
||||||
|
// translators: Short description for `recipe upgrade` command
|
||||||
|
Short: i18n.G("Upgrade recipe image tags"),
|
||||||
|
Long: i18n.G(`Upgrade a given <recipe> configuration.
|
||||||
|
|
||||||
|
It will update the relevant compose file tags on the local file system.
|
||||||
|
|
||||||
|
Some image tags cannot be parsed because they do not follow some sort of
|
||||||
|
semver-like convention. In this case, all possible tags will be listed and it
|
||||||
|
is up to the end-user to decide.
|
||||||
|
|
||||||
|
The command is interactive and will show a select input which allows you to
|
||||||
|
make a seclection. Use the "?" key to see more help on navigating this
|
||||||
|
interface.`),
|
||||||
|
Args: cobra.RangeArgs(0, 1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string,
|
||||||
|
) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
recipe := internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
|
||||||
|
if err := recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
bumpType := btoi(internal.Major)*4 + btoi(internal.Minor)*2 + btoi(internal.Patch)
|
||||||
|
if bumpType != 0 {
|
||||||
|
// a bitwise check if the number is a power of 2
|
||||||
|
if (bumpType & (bumpType - 1)) != 0 {
|
||||||
|
log.Fatal(i18n.G("you can only use one of: --major, --minor, --patch."))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.MachineReadable {
|
||||||
|
// -m implies -n in this case
|
||||||
|
internal.NoInput = true
|
||||||
|
}
|
||||||
|
|
||||||
|
upgradeList := make(map[string]anUpgrade)
|
||||||
|
|
||||||
|
// check for versions file and load pinned versions
|
||||||
|
versionsPresent := false
|
||||||
|
versionsPath := path.Join(recipe.Dir, "versions")
|
||||||
|
servicePins := make(map[string]imgPin)
|
||||||
|
if _, err := os.Stat(versionsPath); err == nil {
|
||||||
|
log.Debug(i18n.G("found versions file for %s", recipe.Name))
|
||||||
|
file, err := os.Open(versionsPath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
scanner := bufio.NewScanner(file)
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
splitLine := strings.Split(line, " ")
|
||||||
|
if splitLine[0] != "pin" || len(splitLine) != 3 {
|
||||||
|
log.Fatal(i18n.G("malformed version pin specification: %s", line))
|
||||||
|
}
|
||||||
|
pinSlice := strings.Split(splitLine[2], ":")
|
||||||
|
pinTag, err := tagcmp.Parse(pinSlice[1])
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
pin := imgPin{
|
||||||
|
image: pinSlice[0],
|
||||||
|
version: pinTag,
|
||||||
|
}
|
||||||
|
servicePins[splitLine[1]] = pin
|
||||||
|
}
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
log.Error(err)
|
||||||
|
}
|
||||||
|
versionsPresent = true
|
||||||
|
} else {
|
||||||
|
log.Debug(i18n.G("did not find versions file for %s", recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
config, err := recipe.GetComposeConfig(nil)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, service := range config.Services {
|
||||||
|
img, err := reference.ParseNormalizedNamed(service.Image)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
regVersions, err := client.GetRegistryTags(img)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
image := reference.Path(img)
|
||||||
|
log.Debug(i18n.G("retrieved %s from remote registry for %s", regVersions, image))
|
||||||
|
image = formatter.StripTagMeta(image)
|
||||||
|
|
||||||
|
switch img.(type) {
|
||||||
|
case reference.NamedTagged:
|
||||||
|
if !tagcmp.IsParsable(img.(reference.NamedTagged).Tag()) {
|
||||||
|
log.Debug(i18n.G("%s not considered semver-like", img.(reference.NamedTagged).Tag()))
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
log.Warn(i18n.G("unable to read tag for image %s, is it missing? skipping upgrade for %s", image, service.Name))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
tag, err := tagcmp.Parse(img.(reference.NamedTagged).Tag())
|
||||||
|
if err != nil {
|
||||||
|
log.Warn(i18n.G("unable to parse %s, error was: %s, skipping upgrade for %s", image, err.Error(), service.Name))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("parsed %s for %s", tag, service.Name))
|
||||||
|
|
||||||
|
var compatible []tagcmp.Tag
|
||||||
|
for _, regVersion := range regVersions {
|
||||||
|
other, err := tagcmp.Parse(regVersion)
|
||||||
|
if err != nil {
|
||||||
|
continue // skip tags that cannot be parsed
|
||||||
|
}
|
||||||
|
|
||||||
|
if tag.IsCompatible(other) && tag.IsLessThan(other) && !tag.Equals(other) {
|
||||||
|
compatible = append(compatible, other)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("detected potential upgradable tags %s for %s", compatible, service.Name))
|
||||||
|
|
||||||
|
sort.Sort(tagcmp.ByTagDesc(compatible))
|
||||||
|
|
||||||
|
if len(compatible) == 0 && !allTags {
|
||||||
|
log.Info(i18n.G("no new versions available for %s, assuming %s is the latest (use -a/--all-tags to see all anyway)", image, tag))
|
||||||
|
continue // skip on to the next tag and don't update any compose files
|
||||||
|
}
|
||||||
|
|
||||||
|
catlVersions, err := recipePkg.VersionsOfService(recipe.Name, service.Name, internal.Offline)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
compatibleStrings := []string{"skip"}
|
||||||
|
for _, compat := range compatible {
|
||||||
|
skip := false
|
||||||
|
for _, catlVersion := range catlVersions {
|
||||||
|
if compat.String() == catlVersion {
|
||||||
|
skip = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !skip {
|
||||||
|
compatibleStrings = append(compatibleStrings, compat.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("detected compatible upgradable tags %s for %s", compatibleStrings, service.Name))
|
||||||
|
|
||||||
|
var upgradeTag string
|
||||||
|
_, ok := servicePins[service.Name]
|
||||||
|
if versionsPresent && ok {
|
||||||
|
pinnedTag := servicePins[service.Name].version
|
||||||
|
if tag.IsLessThan(pinnedTag) {
|
||||||
|
pinnedTagString := pinnedTag.String()
|
||||||
|
contains := false
|
||||||
|
for _, v := range compatible {
|
||||||
|
if pinnedTag.IsUpgradeCompatible(v) {
|
||||||
|
contains = true
|
||||||
|
upgradeTag = v.String()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if contains {
|
||||||
|
log.Info(i18n.G("upgrading service %s from %s to %s (pinned tag: %s)", service.Name, tag.String(), upgradeTag, pinnedTagString))
|
||||||
|
} else {
|
||||||
|
log.Info(i18n.G("service %s, image %s pinned to %s, no compatible upgrade found", service.Name, servicePins[service.Name].image, pinnedTagString))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.Fatal(i18n.G("service %s is at version %s, but pinned to %s, please correct your compose.yml file manually!", service.Name, tag.String(), pinnedTag.String()))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if bumpType != 0 {
|
||||||
|
for _, upTag := range compatible {
|
||||||
|
upElement, err := tag.UpgradeDelta(upTag)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
delta := upElement.UpgradeType()
|
||||||
|
if delta <= bumpType {
|
||||||
|
upgradeTag = upTag.String()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if upgradeTag == "" {
|
||||||
|
log.Warn(i18n.G("not upgrading from %s to %s for %s, because the upgrade type is more serious than what user wants", tag.String(), compatible[0].String(), image))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
msg := i18n.G("upgrade to which tag? (service: %s, image: %s, tag: %s)", service.Name, image, tag)
|
||||||
|
if !tagcmp.IsParsable(img.(reference.NamedTagged).Tag()) || allTags {
|
||||||
|
tag := img.(reference.NamedTagged).Tag()
|
||||||
|
if !allTags {
|
||||||
|
log.Warn(i18n.G("unable to determine versioning semantics of %s, listing all tags", tag))
|
||||||
|
}
|
||||||
|
msg = i18n.G("upgrade to which tag? (service: %s, tag: %s)", service.Name, tag)
|
||||||
|
compatibleStrings = []string{"skip"}
|
||||||
|
for _, regVersion := range regVersions {
|
||||||
|
compatibleStrings = append(compatibleStrings, regVersion)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// there is always at least the item "skip" in compatibleStrings (a list of
|
||||||
|
// possible upgradable tags) and at least one other tag.
|
||||||
|
upgradableTags := compatibleStrings[1:]
|
||||||
|
upgrade := anUpgrade{
|
||||||
|
Service: service.Name,
|
||||||
|
Image: image,
|
||||||
|
Tag: tag.String(),
|
||||||
|
UpgradeTags: make([]string, len(upgradableTags)),
|
||||||
|
}
|
||||||
|
|
||||||
|
for n, s := range upgradableTags {
|
||||||
|
var sb strings.Builder
|
||||||
|
if _, err := sb.WriteString(s); err != nil {
|
||||||
|
}
|
||||||
|
upgrade.UpgradeTags[n] = sb.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
upgradeList[upgrade.Service] = upgrade
|
||||||
|
|
||||||
|
if internal.NoInput {
|
||||||
|
upgradeTag = "skip"
|
||||||
|
} else {
|
||||||
|
prompt := &survey.Select{
|
||||||
|
Message: msg,
|
||||||
|
Help: i18n.G("enter / return to confirm, choose 'skip' to not upgrade this tag, vim mode is enabled"),
|
||||||
|
VimMode: true,
|
||||||
|
Options: compatibleStrings,
|
||||||
|
}
|
||||||
|
if err := survey.AskOne(prompt, &upgradeTag); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if upgradeTag != "skip" {
|
||||||
|
ok, err := recipe.UpdateTag(image, upgradeTag)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if ok {
|
||||||
|
log.Info(i18n.G("tag upgraded from %s to %s for %s", tag.String(), upgradeTag, image))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if !internal.NoInput {
|
||||||
|
log.Warn(i18n.G("not upgrading %s, skipping as requested", image))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.NoInput {
|
||||||
|
if internal.MachineReadable {
|
||||||
|
jsonstring, err := json.Marshal(upgradeList)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println(string(jsonstring))
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, upgrade := range upgradeList {
|
||||||
|
log.Info(i18n.G("can upgrade service: %s, image: %s, tag: %s ::", upgrade.Service, upgrade.Image, upgrade.Tag))
|
||||||
|
for _, utag := range upgrade.UpgradeTags {
|
||||||
|
log.Infof(" %s", utag)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
isClean, err := gitPkg.IsClean(recipe.Dir)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
if !isClean {
|
||||||
|
log.Info(i18n.G("%s currently has these unstaged changes 👇", recipe.Name))
|
||||||
|
if err := gitPkg.DiffUnstaged(recipe.Dir); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !internal.NoInput && !createCommit {
|
||||||
|
prompt := &survey.Confirm{
|
||||||
|
Message: i18n.G("commit changes?"),
|
||||||
|
Default: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := survey.AskOne(prompt, &createCommit); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if createCommit {
|
||||||
|
msg := i18n.G("chore: update image tags")
|
||||||
|
if err := gitPkg.Commit(recipe.Dir, msg, internal.Dry); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
log.Info(i18n.G("committed changes as '%s'", msg))
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
if createCommit {
|
||||||
|
log.Warn(i18n.G("no changes, skip creating commit"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
allTags bool
|
||||||
|
createCommit bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
RecipeUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&internal.Major,
|
||||||
|
i18n.G("major"),
|
||||||
|
i18n.G("x"),
|
||||||
|
false,
|
||||||
|
i18n.G("increase the major part of the version"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&internal.Minor,
|
||||||
|
i18n.G("minor"),
|
||||||
|
i18n.G("y"),
|
||||||
|
false,
|
||||||
|
i18n.G("increase the minor part of the version"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&internal.Patch,
|
||||||
|
i18n.G("patch"),
|
||||||
|
i18n.G("z"),
|
||||||
|
false,
|
||||||
|
i18n.G("increase the patch part of the version"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&internal.MachineReadable,
|
||||||
|
i18n.G("machine"),
|
||||||
|
i18n.G("m"),
|
||||||
|
false,
|
||||||
|
i18n.G("print machine-readable output"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&allTags,
|
||||||
|
i18n.G("all-tags"),
|
||||||
|
i18n.GC("a", "recipe upgrade"),
|
||||||
|
false,
|
||||||
|
i18n.G("list all tags, not just upgrades"),
|
||||||
|
)
|
||||||
|
|
||||||
|
RecipeUpgradeCommand.Flags().BoolVarP(
|
||||||
|
&createCommit,
|
||||||
|
i18n.G("commit"),
|
||||||
|
i18n.GC("c", "recipe upgrade"),
|
||||||
|
false,
|
||||||
|
i18n.G("commit changes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,143 @@
|
|||||||
|
package recipe
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
recipePkg "coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra recipe versions` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var recipeVersionsAliases = i18n.G("v")
|
||||||
|
|
||||||
|
var RecipeVersionCommand = &cobra.Command{
|
||||||
|
// translators: `recipe versions` command
|
||||||
|
Use: i18n.G("versions <recipe> [flags]"),
|
||||||
|
Aliases: strings.Split(recipeVersionsAliases, ","),
|
||||||
|
// translators: Short description for `recipe versions` command
|
||||||
|
Short: i18n.G("List recipe versions"),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.RecipeNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
var warnMessages []string
|
||||||
|
|
||||||
|
recipe := internal.ValidateRecipe(args, cmd.Name())
|
||||||
|
|
||||||
|
catl, err := recipePkg.ReadRecipeCatalogue(internal.Offline)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
recipeMeta, ok := catl[recipe.Name]
|
||||||
|
if !ok {
|
||||||
|
warnMessages = append(warnMessages, i18n.G("retrieved versions from local recipe repository"))
|
||||||
|
|
||||||
|
recipeVersions, warnMsg, err := recipe.GetRecipeVersions()
|
||||||
|
if err != nil {
|
||||||
|
warnMessages = append(warnMessages, err.Error())
|
||||||
|
}
|
||||||
|
if len(warnMsg) > 0 {
|
||||||
|
warnMessages = append(warnMessages, warnMsg...)
|
||||||
|
}
|
||||||
|
|
||||||
|
recipeMeta = recipePkg.RecipeMeta{Versions: recipeVersions}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(recipeMeta.Versions) == 0 {
|
||||||
|
log.Fatal(i18n.G("%s has no published versions?", recipe.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := len(recipeMeta.Versions) - 1; i >= 0; i-- {
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Headers(i18n.G("SERVICE"), i18n.G("IMAGE"), i18n.G("TAG"), i18n.G("VERSION"))
|
||||||
|
|
||||||
|
for version, meta := range recipeMeta.Versions[i] {
|
||||||
|
var allRows [][]string
|
||||||
|
var rows [][]string
|
||||||
|
|
||||||
|
for service, serviceMeta := range meta {
|
||||||
|
recipeVersion := version
|
||||||
|
if service != "app" {
|
||||||
|
recipeVersion = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
rows = append(rows, []string{
|
||||||
|
service,
|
||||||
|
serviceMeta.Image,
|
||||||
|
serviceMeta.Tag,
|
||||||
|
recipeVersion,
|
||||||
|
})
|
||||||
|
|
||||||
|
allRows = append(allRows, []string{
|
||||||
|
version,
|
||||||
|
service,
|
||||||
|
serviceMeta.Image,
|
||||||
|
serviceMeta.Tag,
|
||||||
|
recipeVersion,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Slice(rows, sortServiceByName(rows))
|
||||||
|
|
||||||
|
table.Rows(rows...)
|
||||||
|
|
||||||
|
if !internal.MachineReadable {
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.MachineReadable {
|
||||||
|
sort.Slice(allRows, sortServiceByName(allRows))
|
||||||
|
headers := []string{i18n.G("VERSION"), i18n.G("SERVICE"), i18n.G("NAME"), i18n.G("TAG")}
|
||||||
|
out, err := formatter.ToJSON(headers, allRows)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to render to JSON: %s", err))
|
||||||
|
}
|
||||||
|
fmt.Println(out)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !internal.MachineReadable {
|
||||||
|
for _, warnMsg := range warnMessages {
|
||||||
|
log.Warn(warnMsg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func sortServiceByName(versions [][]string) func(i, j int) bool {
|
||||||
|
return func(i, j int) bool {
|
||||||
|
return versions[i][0] < versions[j][0]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
RecipeVersionCommand.Flags().BoolVarP(
|
||||||
|
&internal.MachineReadable,
|
||||||
|
i18n.G("machine"),
|
||||||
|
i18n.G("m"),
|
||||||
|
false,
|
||||||
|
i18n.G("print machine-readable output"),
|
||||||
|
)
|
||||||
|
}
|
||||||
+319
@@ -0,0 +1,319 @@
|
|||||||
|
package cli
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/app"
|
||||||
|
"coopcloud.tech/abra/cli/catalogue"
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/cli/recipe"
|
||||||
|
"coopcloud.tech/abra/cli/server"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
charmLog "github.com/charmbracelet/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
"github.com/spf13/cobra/doc"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// translators: `abra` usage template. please translate only words like
|
||||||
|
// "Aliases" and "Example" and nothing inside the {{ ... }}
|
||||||
|
usageTemplate = i18n.G(`Usage:{{if .Runnable}}
|
||||||
|
{{.UseLine}}{{end}}{{if .HasAvailableSubCommands}}
|
||||||
|
{{.CommandPath}} [command]{{end}}{{if gt (len .Aliases) 0}}
|
||||||
|
|
||||||
|
Aliases:
|
||||||
|
{{.NameAndAliases}}{{end}}{{if .HasExample}}
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
{{.Example}}{{end}}{{if .HasAvailableSubCommands}}
|
||||||
|
|
||||||
|
Available Commands:{{range .Commands}}{{if (or .IsAvailableCommand (eq .Name "help"))}}
|
||||||
|
{{rpad .Name .NamePadding }} {{.Short}}{{end}}{{end}}{{end}}{{if .HasAvailableLocalFlags}}
|
||||||
|
|
||||||
|
Flags:
|
||||||
|
{{.LocalFlags.FlagUsages | trimTrailingWhitespaces}}{{end}}{{if .HasAvailableInheritedFlags}}
|
||||||
|
|
||||||
|
Global Flags:
|
||||||
|
{{.InheritedFlags.FlagUsages | trimTrailingWhitespaces}}{{end}}{{if .HasHelpSubCommands}}
|
||||||
|
|
||||||
|
Additional help topics:{{range .Commands}}{{if .IsAdditionalHelpTopicCommand}}
|
||||||
|
{{rpad .CommandPath .CommandPathPadding}} {{.Short}}{{end}}{{end}}{{end}}{{if .HasAvailableSubCommands}}
|
||||||
|
|
||||||
|
Use "{{.CommandPath}} [command] --help" for more information about a command.{{end}}
|
||||||
|
`)
|
||||||
|
|
||||||
|
helpCmd = &cobra.Command{
|
||||||
|
Use: i18n.G("help [command]"),
|
||||||
|
// translators: Short description for `help` command
|
||||||
|
Short: i18n.G("Help about any command"),
|
||||||
|
Long: i18n.G(`Help provides help for any command in the application.
|
||||||
|
Simply type abra help [path to command] for full details.`),
|
||||||
|
Run: func(c *cobra.Command, args []string) {
|
||||||
|
cmd, _, e := c.Root().Find(args)
|
||||||
|
if cmd == nil || e != nil {
|
||||||
|
c.Print(i18n.G("unknown help topic %#q\n", args))
|
||||||
|
if err := c.Root().Usage(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
cmd.InitDefaultHelpFlag()
|
||||||
|
cmd.InitDefaultVersionFlag()
|
||||||
|
if err := cmd.Help(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
func Run(version, commit string) {
|
||||||
|
rootCmd := &cobra.Command{
|
||||||
|
// translators: `abra` binary name
|
||||||
|
Use: i18n.G("abra [cmd] [args] [flags]"),
|
||||||
|
// translators: Short description for `abra` binary
|
||||||
|
Short: i18n.G("The Co-op Cloud command-line utility belt 🎩🐇"),
|
||||||
|
// translators: Long description for `abra` binary. This needs to be
|
||||||
|
// translated in the same way as the Short description so that everything
|
||||||
|
// matches up
|
||||||
|
Long: i18n.G(`The Co-op Cloud command-line utility belt 🎩🐇
|
||||||
|
|
||||||
|
Config:
|
||||||
|
$ABRA_DIR: %s`, config.ABRA_DIR),
|
||||||
|
Version: fmt.Sprintf("%s-%s", version, commit[:7]),
|
||||||
|
ValidArgs: []string{
|
||||||
|
// translators: `abra app` command for autocompletion
|
||||||
|
i18n.G("app"),
|
||||||
|
// translators: `abra autocomplete` command for autocompletion
|
||||||
|
i18n.G("autocomplete"),
|
||||||
|
// translators: `abra catalogue` command for autocompletion
|
||||||
|
i18n.G("catalogue"),
|
||||||
|
// translators: `abra man` command for autocompletion
|
||||||
|
i18n.G("man"),
|
||||||
|
// translators: `abra recipe` command for autocompletion
|
||||||
|
i18n.G("recipe"),
|
||||||
|
// translators: `abra server` command for autocompletion
|
||||||
|
i18n.G("server"),
|
||||||
|
// translators: `abra upgrade` command for autocompletion
|
||||||
|
i18n.G("upgrade"),
|
||||||
|
},
|
||||||
|
PersistentPreRunE: func(cmd *cobra.Command, args []string) error {
|
||||||
|
dirs := []map[string]os.FileMode{
|
||||||
|
{config.ABRA_DIR: 0764},
|
||||||
|
{config.SERVERS_DIR: 0700},
|
||||||
|
{config.RECIPES_DIR: 0764},
|
||||||
|
{config.LOGS_DIR: 0764},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, dir := range dirs {
|
||||||
|
for path, perm := range dir {
|
||||||
|
if err := os.Mkdir(path, perm); err != nil {
|
||||||
|
if !os.IsExist(err) {
|
||||||
|
return errors.New(i18n.G("unable to create %s: %s", path, err))
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Logger.SetStyles(charmLog.DefaultStyles())
|
||||||
|
charmLog.SetDefault(log.Logger)
|
||||||
|
|
||||||
|
if internal.MachineReadable {
|
||||||
|
log.SetOutput(os.Stderr)
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.Debug {
|
||||||
|
log.SetLevel(log.DebugLevel)
|
||||||
|
log.SetOutput(os.Stderr)
|
||||||
|
log.SetReportCaller(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G(
|
||||||
|
"abra version: %s, commit: %s, lang: %s",
|
||||||
|
version, formatter.SmallSHA(commit), i18n.Locale,
|
||||||
|
))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
rootCmd.CompletionOptions.DisableDefaultCmd = true
|
||||||
|
rootCmd.SetUsageTemplate(usageTemplate)
|
||||||
|
rootCmd.SetHelpCommand(helpCmd)
|
||||||
|
|
||||||
|
// translators: `abra man` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
manAliases := i18n.G("m")
|
||||||
|
|
||||||
|
manCommand := &cobra.Command{
|
||||||
|
// translators: `man` command
|
||||||
|
Use: i18n.G("man [flags]"),
|
||||||
|
Aliases: strings.Split(manAliases, ","),
|
||||||
|
// translators: Short description for `man` command
|
||||||
|
Short: i18n.G("Generate manpage"),
|
||||||
|
Example: i18n.G(` # generate the man pages into /usr/local/share/man/man1
|
||||||
|
abra_path=$(which abra) # pass abra absolute path to sudo below
|
||||||
|
sudo $abra_path man
|
||||||
|
sudo mandb
|
||||||
|
|
||||||
|
# read the man pages
|
||||||
|
man abra
|
||||||
|
man abra-app-deploy`),
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
header := &doc.GenManHeader{
|
||||||
|
Title: "ABRA",
|
||||||
|
Section: "1",
|
||||||
|
}
|
||||||
|
|
||||||
|
manDir := "/usr/local/share/man/man1"
|
||||||
|
if _, err := os.Stat(manDir); os.IsNotExist(err) {
|
||||||
|
log.Fatal(i18n.G("unable to proceed, %s does not exist?", manDir))
|
||||||
|
}
|
||||||
|
|
||||||
|
err := doc.GenManTree(rootCmd, header, manDir)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("don't forget to run 'sudo mandb'"))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
rootCmd.PersistentFlags().BoolVarP(
|
||||||
|
&internal.Debug,
|
||||||
|
i18n.G("debug"),
|
||||||
|
i18n.G("d"),
|
||||||
|
false,
|
||||||
|
i18n.G("show debug messages"),
|
||||||
|
)
|
||||||
|
|
||||||
|
rootCmd.PersistentFlags().BoolVarP(
|
||||||
|
&internal.NoInput,
|
||||||
|
i18n.G("no-input"),
|
||||||
|
i18n.G("n"),
|
||||||
|
false,
|
||||||
|
i18n.G("toggle non-interactive mode"),
|
||||||
|
)
|
||||||
|
|
||||||
|
rootCmd.PersistentFlags().BoolVarP(
|
||||||
|
&internal.Help,
|
||||||
|
i18n.G("help"),
|
||||||
|
i18n.G("h"),
|
||||||
|
false,
|
||||||
|
i18n.G("help for abra"),
|
||||||
|
)
|
||||||
|
|
||||||
|
rootCmd.PersistentFlags().BoolVarP(
|
||||||
|
&internal.Offline,
|
||||||
|
i18n.G("offline"),
|
||||||
|
i18n.G("o"),
|
||||||
|
false,
|
||||||
|
i18n.G("prefer offline & filesystem access"),
|
||||||
|
)
|
||||||
|
|
||||||
|
rootCmd.Flags().BoolVarP(
|
||||||
|
&internal.Version,
|
||||||
|
i18n.G("version"),
|
||||||
|
i18n.G("v"),
|
||||||
|
false,
|
||||||
|
i18n.G("version for abra"),
|
||||||
|
)
|
||||||
|
|
||||||
|
catalogue.CatalogueCommand.AddCommand(
|
||||||
|
catalogue.CatalogueGenerateCommand,
|
||||||
|
catalogue.CatalogueSyncCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
server.ServerCommand.AddCommand(
|
||||||
|
server.ServerAddCommand,
|
||||||
|
server.ServerListCommand,
|
||||||
|
server.ServerPruneCommand,
|
||||||
|
server.ServerRemoveCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
recipe.RecipeCommand.AddCommand(
|
||||||
|
recipe.RecipeDiffCommand,
|
||||||
|
recipe.RecipeFetchCommand,
|
||||||
|
recipe.RecipeLintCommand,
|
||||||
|
recipe.RecipeListCommand,
|
||||||
|
recipe.RecipeNewCommand,
|
||||||
|
recipe.RecipeReleaseCommand,
|
||||||
|
recipe.RecipeResetCommand,
|
||||||
|
recipe.RecipeUpgradeCommand,
|
||||||
|
recipe.RecipeVersionCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
rootCmd.AddCommand(
|
||||||
|
UpgradeCommand,
|
||||||
|
AutocompleteCommand,
|
||||||
|
manCommand,
|
||||||
|
app.AppCommand,
|
||||||
|
catalogue.CatalogueCommand,
|
||||||
|
server.ServerCommand,
|
||||||
|
recipe.RecipeCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
app.AppCmdCommand.AddCommand(
|
||||||
|
app.AppCmdListCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
app.AppSecretCommand.AddCommand(
|
||||||
|
app.AppSecretGenerateCommand,
|
||||||
|
app.AppSecretInsertCommand,
|
||||||
|
app.AppSecretRmCommand,
|
||||||
|
app.AppSecretLsCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
app.AppVolumeCommand.AddCommand(
|
||||||
|
app.AppVolumeListCommand,
|
||||||
|
app.AppVolumeRemoveCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
app.AppBackupCommand.AddCommand(
|
||||||
|
app.AppBackupListCommand,
|
||||||
|
app.AppBackupDownloadCommand,
|
||||||
|
app.AppBackupCreateCommand,
|
||||||
|
app.AppBackupSnapshotsCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
app.AppEnvCommand.AddCommand(
|
||||||
|
app.AppEnvListCommand,
|
||||||
|
app.AppEnvPullCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
app.AppCommand.AddCommand(
|
||||||
|
app.AppBackupCommand,
|
||||||
|
app.AppCheckCommand,
|
||||||
|
app.AppCmdCommand,
|
||||||
|
app.AppConfigCommand,
|
||||||
|
app.AppCpCommand,
|
||||||
|
app.AppDeployCommand,
|
||||||
|
app.AppListCommand,
|
||||||
|
app.AppLogsCommand,
|
||||||
|
app.AppNewCommand,
|
||||||
|
app.AppPsCommand,
|
||||||
|
app.AppRemoveCommand,
|
||||||
|
app.AppRestartCommand,
|
||||||
|
app.AppRestoreCommand,
|
||||||
|
app.AppRollbackCommand,
|
||||||
|
app.AppMoveCommand,
|
||||||
|
app.AppRunCommand,
|
||||||
|
app.AppSecretCommand,
|
||||||
|
app.AppServicesCommand,
|
||||||
|
app.AppUndeployCommand,
|
||||||
|
app.AppUpgradeCommand,
|
||||||
|
app.AppVolumeCommand,
|
||||||
|
app.AppLabelsCommand,
|
||||||
|
app.AppEnvCommand,
|
||||||
|
)
|
||||||
|
|
||||||
|
if err := rootCmd.Execute(); err != nil {
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,211 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
contextPkg "coopcloud.tech/abra/pkg/context"
|
||||||
|
"coopcloud.tech/abra/pkg/dns"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"coopcloud.tech/abra/pkg/server"
|
||||||
|
sshPkg "coopcloud.tech/abra/pkg/ssh"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra server add` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var serverAddAliases = i18n.GC("a", "server add")
|
||||||
|
|
||||||
|
var ServerAddCommand = &cobra.Command{
|
||||||
|
// translators: `server add` command
|
||||||
|
Use: i18n.G("add [[server] | --local] [flags]"),
|
||||||
|
Aliases: strings.Split(serverAddAliases, ","),
|
||||||
|
// translators: Short description for `server add` command
|
||||||
|
Short: i18n.G("Add a new server"),
|
||||||
|
Long: i18n.G(`Add a new server to your configuration so that it can be managed by Abra.
|
||||||
|
|
||||||
|
Abra relies on the standard SSH command-line and ~/.ssh/config for client
|
||||||
|
connection details. You must configure an entry per-host in your ~/.ssh/config
|
||||||
|
for each server:
|
||||||
|
|
||||||
|
Host 1312.net 1312
|
||||||
|
Hostname 1312.net
|
||||||
|
User antifa
|
||||||
|
Port 12345
|
||||||
|
IdentityFile ~/.ssh/antifa@somewhere
|
||||||
|
|
||||||
|
If "--local" is passed, then Abra assumes that the current local server is
|
||||||
|
intended as the target server. This is useful when you want to have your entire
|
||||||
|
Co-op Cloud config located on the server itself, and not on your local
|
||||||
|
developer machine. The domain is then set to "default".`),
|
||||||
|
Example: i18n.G(" abra server add 1312.net"),
|
||||||
|
Args: cobra.RangeArgs(0, 1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
if !local {
|
||||||
|
return autocomplete.ServerNameComplete()
|
||||||
|
}
|
||||||
|
return nil, cobra.ShellCompDirectiveDefault
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
if len(args) > 0 && local {
|
||||||
|
log.Fatal(i18n.G("cannot use [server] and --local together"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(args) == 0 && !local {
|
||||||
|
log.Fatal(i18n.G("missing argument or --local/-l flag"))
|
||||||
|
}
|
||||||
|
|
||||||
|
name := "default"
|
||||||
|
if !local {
|
||||||
|
name = internal.ValidateDomain(args)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE(d1): reasonable 5 second timeout for connections which can't
|
||||||
|
// succeed. The connection is attempted twice, so this results in 10
|
||||||
|
// seconds.
|
||||||
|
timeout := client.WithTimeout(5)
|
||||||
|
|
||||||
|
if local {
|
||||||
|
created, err := createServerDir(name)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("attempting to create client for %s", name))
|
||||||
|
|
||||||
|
if _, err := client.New(name, timeout); err != nil {
|
||||||
|
cleanUp(name)
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if created {
|
||||||
|
log.Info(i18n.G("local server successfully added"))
|
||||||
|
} else {
|
||||||
|
log.Warn(i18n.G("local server already exists"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := createServerDir(name)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
created, err := newContext(name)
|
||||||
|
if err != nil {
|
||||||
|
cleanUp(name)
|
||||||
|
log.Fatal(i18n.G("unable to create local context: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("attempting to create client for %s", name))
|
||||||
|
|
||||||
|
if _, err := client.New(name, timeout); err != nil {
|
||||||
|
cleanUp(name)
|
||||||
|
log.Fatal(i18n.G("ssh %s error: %s", name, sshPkg.Fatal(name, err)))
|
||||||
|
}
|
||||||
|
|
||||||
|
if created {
|
||||||
|
log.Info(i18n.G("%s successfully added", name))
|
||||||
|
|
||||||
|
if _, err := dns.EnsureIPv4(name); err != nil {
|
||||||
|
log.Warn(i18n.G("unable to resolve IPv4 for %s", name))
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Warn(i18n.G("%s already exists", name))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// cleanUp cleans up the partially created context/client details for a failed
|
||||||
|
// "server add" attempt.
|
||||||
|
func cleanUp(name string) {
|
||||||
|
if name != "default" {
|
||||||
|
log.Debug(i18n.G("serverAdd: cleanUp: cleaning up context for %s", name))
|
||||||
|
if err := client.DeleteContext(name); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
serverDir := filepath.Join(config.SERVERS_DIR, name)
|
||||||
|
files, err := config.GetAllFilesInDirectory(serverDir)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("serverAdd: cleanUp: unable to list files in %s: %s", serverDir, err))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(files) > 0 {
|
||||||
|
log.Debug(i18n.G("serverAdd: cleanUp: %s is not empty, aborting cleanup", serverDir))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := os.RemoveAll(serverDir); err != nil {
|
||||||
|
log.Fatal(i18n.G("serverAdd: cleanUp: failed to remove %s: %s", serverDir, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// newContext creates a new internal Docker context for a server. This is how
|
||||||
|
// Docker manages SSH connection details. These are stored to disk in
|
||||||
|
// ~/.docker. Abra can manage this completely for the user, so it's an
|
||||||
|
// implementation detail.
|
||||||
|
func newContext(name string) (bool, error) {
|
||||||
|
store := contextPkg.NewDefaultDockerContextStore()
|
||||||
|
contexts, err := store.Store.List()
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, context := range contexts {
|
||||||
|
if context.Name == name {
|
||||||
|
log.Debug(i18n.G("context for %s already exists", name))
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debugf(i18n.G("creating context with domain %s", name))
|
||||||
|
|
||||||
|
if err := client.CreateContext(name); err != nil {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// createServerDir creates the ~/.abra/servers/... directory for a new server.
|
||||||
|
func createServerDir(name string) (bool, error) {
|
||||||
|
if err := server.CreateServerDir(name); err != nil {
|
||||||
|
if !os.IsExist(err) {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("server dir for %s already created", name))
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
local bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
ServerAddCommand.Flags().BoolVarP(
|
||||||
|
&local,
|
||||||
|
i18n.G("local"),
|
||||||
|
i18n.G("l"),
|
||||||
|
false,
|
||||||
|
i18n.G("use local server"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,110 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
contextPkg "coopcloud.tech/abra/pkg/context"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/docker/cli/cli/connhelper/ssh"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra server list` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var serverListAliases = i18n.G("ls")
|
||||||
|
|
||||||
|
var ServerListCommand = &cobra.Command{
|
||||||
|
// translators: `server list` command
|
||||||
|
Use: i18n.G("list [flags]"),
|
||||||
|
Aliases: strings.Split(serverListAliases, ","),
|
||||||
|
// translators: Short description for `server list` command
|
||||||
|
Short: i18n.G("List managed servers"),
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
dockerContextStore := contextPkg.NewDefaultDockerContextStore()
|
||||||
|
contexts, err := dockerContextStore.Store.List()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
table, err := formatter.CreateTable()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{i18n.G("NAME"), i18n.G("HOST")}
|
||||||
|
table.Headers(headers...)
|
||||||
|
|
||||||
|
serverNames, err := config.ReadServerNames()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var rows [][]string
|
||||||
|
for _, serverName := range serverNames {
|
||||||
|
var row []string
|
||||||
|
for _, dockerCtx := range contexts {
|
||||||
|
endpoint, err := contextPkg.GetContextEndpoint(dockerCtx)
|
||||||
|
if err != nil && strings.Contains(err.Error(), "does not exist") {
|
||||||
|
// No local context found, we can continue safely
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if dockerCtx.Name == serverName {
|
||||||
|
sp, err := ssh.ParseURL(endpoint)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if sp.Host == "" {
|
||||||
|
sp.Host = i18n.G("unknown")
|
||||||
|
}
|
||||||
|
|
||||||
|
row = []string{serverName, sp.Host}
|
||||||
|
rows = append(rows, row)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(row) == 0 {
|
||||||
|
if serverName == "default" {
|
||||||
|
row = []string{serverName, i18n.G("local")}
|
||||||
|
} else {
|
||||||
|
row = []string{serverName, i18n.G("unknown")}
|
||||||
|
}
|
||||||
|
rows = append(rows, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
table.Row(row...)
|
||||||
|
}
|
||||||
|
|
||||||
|
if internal.MachineReadable {
|
||||||
|
out, err := formatter.ToJSON(headers, rows)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(i18n.G("unable to render to JSON: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println(out)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := formatter.PrintTable(table); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
ServerListCommand.Flags().BoolVarP(
|
||||||
|
&internal.MachineReadable,
|
||||||
|
i18n.G("machine"),
|
||||||
|
i18n.G("m"),
|
||||||
|
false,
|
||||||
|
i18n.G("print machine-readable output"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,111 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra server prune` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var serverPruneliases = i18n.G("p")
|
||||||
|
|
||||||
|
var ServerPruneCommand = &cobra.Command{
|
||||||
|
// translators: `server prune` command
|
||||||
|
Use: i18n.G("prune <server> [flags]"),
|
||||||
|
Aliases: strings.Split(serverPruneliases, ","),
|
||||||
|
// translators: Short description for `server prune` command
|
||||||
|
Short: i18n.G("Prune resources on a server"),
|
||||||
|
Long: i18n.G(`Prunes unused containers, networks, and dangling images.
|
||||||
|
|
||||||
|
Use "--volumes/-v" to remove volumes that are not associated with a deployed
|
||||||
|
app. This can result in unwanted data loss if not used carefully.`),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.ServerNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
serverName := internal.ValidateServer(args)
|
||||||
|
|
||||||
|
cl, err := client.New(serverName)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var filterArgs filters.Args
|
||||||
|
|
||||||
|
cr, err := cl.ContainersPrune(cmd.Context(), filterArgs)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cntSpaceReclaimed := formatter.ByteCountSI(cr.SpaceReclaimed)
|
||||||
|
log.Info(i18n.G("containers pruned: %d; space reclaimed: %s", len(cr.ContainersDeleted), cntSpaceReclaimed))
|
||||||
|
|
||||||
|
nr, err := cl.NetworksPrune(cmd.Context(), filterArgs)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("networks pruned: %d", len(nr.NetworksDeleted)))
|
||||||
|
|
||||||
|
pruneFilters := filters.NewArgs()
|
||||||
|
if allFilter {
|
||||||
|
log.Debug(i18n.G("removing all images, not only dangling ones"))
|
||||||
|
pruneFilters.Add("dangling", "false")
|
||||||
|
}
|
||||||
|
|
||||||
|
ir, err := cl.ImagesPrune(cmd.Context(), pruneFilters)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
imgSpaceReclaimed := formatter.ByteCountSI(ir.SpaceReclaimed)
|
||||||
|
log.Info(i18n.G("images pruned: %d; space reclaimed: %s", len(ir.ImagesDeleted), imgSpaceReclaimed))
|
||||||
|
|
||||||
|
if volumesFilter {
|
||||||
|
vr, err := cl.VolumesPrune(cmd.Context(), filterArgs)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
volSpaceReclaimed := formatter.ByteCountSI(vr.SpaceReclaimed)
|
||||||
|
log.Info(i18n.G("volumes pruned: %d; space reclaimed: %s", len(vr.VolumesDeleted), volSpaceReclaimed))
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
allFilter bool
|
||||||
|
volumesFilter bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
ServerPruneCommand.Flags().BoolVarP(
|
||||||
|
&allFilter,
|
||||||
|
i18n.G("all"),
|
||||||
|
i18n.GC("a", "server prune"),
|
||||||
|
false,
|
||||||
|
i18n.G("remove all unused images"),
|
||||||
|
)
|
||||||
|
|
||||||
|
ServerPruneCommand.Flags().BoolVarP(
|
||||||
|
&volumesFilter,
|
||||||
|
i18n.G("volumes"),
|
||||||
|
i18n.G("v"),
|
||||||
|
false,
|
||||||
|
i18n.G("remove volumes"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/autocomplete"
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra server remove` aliases. use a comma separated list of
|
||||||
|
// aliases with no spaces in between
|
||||||
|
var serverRemoveAliases = i18n.G("rm")
|
||||||
|
|
||||||
|
var ServerRemoveCommand = &cobra.Command{
|
||||||
|
// translators: `server remove` command
|
||||||
|
Use: i18n.G("remove <server> [flags]"),
|
||||||
|
Aliases: strings.Split(serverRemoveAliases, ","),
|
||||||
|
// translators: Short description for `server remove` command
|
||||||
|
Short: i18n.G("Remove a managed server"),
|
||||||
|
Long: i18n.G(`Remove a managed server.
|
||||||
|
|
||||||
|
Abra will remove the internal bookkeeping ($ABRA_DIR/servers/...) and
|
||||||
|
underlying client connection context. This server will then be lost in time,
|
||||||
|
like tears in rain.`),
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
ValidArgsFunction: func(
|
||||||
|
cmd *cobra.Command,
|
||||||
|
args []string,
|
||||||
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
return autocomplete.ServerNameComplete()
|
||||||
|
},
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
serverName := internal.ValidateServer(args)
|
||||||
|
|
||||||
|
if err := client.DeleteContext(serverName); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := os.RemoveAll(filepath.Join(config.SERVERS_DIR, serverName)); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info(i18n.G("%s is now lost in time, like tears in rain", serverName))
|
||||||
|
|
||||||
|
return
|
||||||
|
},
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra server` aliases. use a comma separated list of aliases
|
||||||
|
// with no spaces in between
|
||||||
|
var serverAliases = i18n.G("s")
|
||||||
|
|
||||||
|
// ServerCommand defines the `abra server` command and its subcommands
|
||||||
|
var ServerCommand = &cobra.Command{
|
||||||
|
// translators: `server` command group
|
||||||
|
Use: i18n.G("server [cmd] [args] [flags]"),
|
||||||
|
Aliases: strings.Split(serverAliases, ","),
|
||||||
|
// translators: Short description for `server` command group
|
||||||
|
Short: i18n.G("Manage servers"),
|
||||||
|
}
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
// Package cli provides the interface for the command-line.
|
||||||
|
package cli
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os/exec"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/cli/internal"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// translators: `abra upgrade` aliases. use a comma separated list of aliases with
|
||||||
|
// no spaces in between
|
||||||
|
var upgradeAliases = i18n.G("u")
|
||||||
|
|
||||||
|
// UpgradeCommand upgrades abra in-place.
|
||||||
|
var UpgradeCommand = &cobra.Command{
|
||||||
|
// translators: `upgrade` command
|
||||||
|
Use: i18n.G("upgrade [flags]"),
|
||||||
|
Aliases: strings.Split(upgradeAliases, ","),
|
||||||
|
// translators: Short description for `upgrade` command
|
||||||
|
Short: i18n.G("Upgrade abra"),
|
||||||
|
Long: i18n.G(`Upgrade abra in-place with the latest stable or release candidate.
|
||||||
|
|
||||||
|
By default, the latest stable release is downloaded.
|
||||||
|
|
||||||
|
Use "--rc/-r" to install the latest release candidate. Please bear in mind that
|
||||||
|
it may contain absolutely catastrophic deal-breaker bugs. Thank you very much
|
||||||
|
for the testing efforts 💗`),
|
||||||
|
Example: i18n.G(" abra upgrade --rc"),
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
mainURL := "https://install.abra.coopcloud.tech"
|
||||||
|
c := exec.Command("bash", "-c", fmt.Sprintf("wget -q -O- %s | bash", mainURL))
|
||||||
|
|
||||||
|
if releaseCandidate {
|
||||||
|
releaseCandidateURL := "https://git.coopcloud.tech/coop-cloud/abra/raw/branch/main/scripts/installer/installer"
|
||||||
|
c = exec.Command("bash", "-c", fmt.Sprintf("wget -q -O- %s | bash -s -- --rc", releaseCandidateURL))
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debugf(i18n.G("attempting to run %s", c))
|
||||||
|
|
||||||
|
if err := internal.RunCmd(c); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
releaseCandidate bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
UpgradeCommand.Flags().BoolVarP(
|
||||||
|
&releaseCandidate,
|
||||||
|
"rc",
|
||||||
|
"r",
|
||||||
|
false,
|
||||||
|
i18n.G("install release candidate (may contain bugs)"),
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
// Package main provides the command-line entrypoint.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"coopcloud.tech/abra/cli"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Version is the current version of Abra.
|
||||||
|
var Version string
|
||||||
|
|
||||||
|
// Commit is the current git commit of Abra.
|
||||||
|
var Commit string
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if Version == "" {
|
||||||
|
Version = "dev"
|
||||||
|
}
|
||||||
|
if Commit == "" {
|
||||||
|
Commit = " "
|
||||||
|
}
|
||||||
|
|
||||||
|
cli.Run(Version, Commit)
|
||||||
|
}
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
#compdef abra
|
|
||||||
|
|
||||||
_abra () {
|
|
||||||
local context state line curcontext="$curcontext" ret=1
|
|
||||||
_arguments -n : \
|
|
||||||
{-h,--help}'[Help message]' \
|
|
||||||
'1:commands:(app server)' \
|
|
||||||
'*::arguments:->arguments' \
|
|
||||||
&& ret=0
|
|
||||||
|
|
||||||
case $state in
|
|
||||||
(arguments)
|
|
||||||
curcontext="${curcontext%:*:*}:abra-arguments-$words[1]:"
|
|
||||||
case $words[1] in
|
|
||||||
(app)
|
|
||||||
_arguments \
|
|
||||||
'1: :_abra_apps' \
|
|
||||||
&& ret=0
|
|
||||||
;;
|
|
||||||
(server)
|
|
||||||
_arguments \
|
|
||||||
'1:servers:_abra_servers' \
|
|
||||||
&& ret=0
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
return ret
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_servers() {
|
|
||||||
_path_files -/W $HOME/.abra/servers
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_apps()
|
|
||||||
{
|
|
||||||
local newapps apps=($HOME/.abra/servers/*/*.env)
|
|
||||||
typeset -a apps
|
|
||||||
newapps=()
|
|
||||||
for app in $apps; do
|
|
||||||
newapps+=($(_abra_basename "${app}"))
|
|
||||||
done
|
|
||||||
_describe -t apps 'app' newapps
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_basename()
|
|
||||||
{
|
|
||||||
printf -- "${1##*/}"
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra "$@"
|
|
||||||
@@ -1,140 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
_abra_basename()
|
|
||||||
{
|
|
||||||
echo "${1##*/}"
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_servers()
|
|
||||||
{
|
|
||||||
# FIXME 3wc: copied from abra/get_servers()
|
|
||||||
shopt -s nullglob dotglob
|
|
||||||
local SERVERS=(~/.abra/servers/*)
|
|
||||||
shopt -u nullglob dotglob
|
|
||||||
|
|
||||||
for SERVER in "${SERVERS[@]}"; do
|
|
||||||
_abra_basename "${SERVER}"
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_complete_servers()
|
|
||||||
{
|
|
||||||
mapfile -t COMPREPLY < <(compgen -W "$(_abra_servers)" -- "$1")
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_apps()
|
|
||||||
{
|
|
||||||
shopt -s nullglob dotglob
|
|
||||||
local APPS=(~/.abra/servers/*/*.env)
|
|
||||||
shopt -u nullglob dotglob
|
|
||||||
|
|
||||||
for APP in "${APPS[@]}"; do
|
|
||||||
_abra_basename "${APP%.env}"
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_complete_apps()
|
|
||||||
{
|
|
||||||
mapfile -t COMPREPLY < <(compgen -W "$(_abra_apps)" -- "$1")
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_recipes()
|
|
||||||
{
|
|
||||||
shopt -s nullglob dotglob
|
|
||||||
local RECIPES=(~/.abra/apps/*)
|
|
||||||
shopt -u nullglob dotglob
|
|
||||||
|
|
||||||
for RECIPE in "${RECIPES[@]}"; do
|
|
||||||
_abra_basename "${RECIPE%.env}"
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_complete_recipes()
|
|
||||||
{
|
|
||||||
mapfile -t COMPREPLY < <(compgen -W "$(_abra_recipes)" -- "$1")
|
|
||||||
}
|
|
||||||
|
|
||||||
_abra_complete()
|
|
||||||
{
|
|
||||||
compopt +o default +o nospace
|
|
||||||
COMPREPLY=()
|
|
||||||
|
|
||||||
local -r cmds='
|
|
||||||
app
|
|
||||||
server
|
|
||||||
recipe
|
|
||||||
'
|
|
||||||
local -r short_opts='-e -h -s -v'
|
|
||||||
local -r long_opts='--env --help --stack --version'
|
|
||||||
|
|
||||||
# Scan through the command line and find the abra command
|
|
||||||
# (if present), as well as its expected position.
|
|
||||||
local cmd
|
|
||||||
local cmd_index=1 # Expected index of the command token.
|
|
||||||
local i
|
|
||||||
for (( i = 1; i < ${#COMP_WORDS[@]}; i++ )); do
|
|
||||||
local word="${COMP_WORDS[i]}"
|
|
||||||
case "$word" in
|
|
||||||
-*)
|
|
||||||
((cmd_index++))
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
cmd="$word"
|
|
||||||
break
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
done
|
|
||||||
|
|
||||||
local cur="${COMP_WORDS[COMP_CWORD]}"
|
|
||||||
|
|
||||||
if (( COMP_CWORD < cmd_index )); then
|
|
||||||
# Offer option completions.
|
|
||||||
case "$cur" in
|
|
||||||
--*)
|
|
||||||
mapfile -t COMPREPLY < <(compgen -W "$long_opts" -- "$cur")
|
|
||||||
;;
|
|
||||||
-*)
|
|
||||||
mapfile -t COMPREPLY < <(compgen -W "$short_opts" -- "$cur")
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
# Skip completion; we should never get here.
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
elif (( COMP_CWORD == cmd_index )); then
|
|
||||||
# Offer command name completions.
|
|
||||||
mapfile -t COMPREPLY < <(compgen -W "$cmds" -- "$cur")
|
|
||||||
else
|
|
||||||
# Offer command argument completions.
|
|
||||||
case "$cmd" in
|
|
||||||
server)
|
|
||||||
# Offer exactly one server name completion.
|
|
||||||
if (( COMP_CWORD == cmd_index + 1 )); then
|
|
||||||
_abra_complete_servers "$cur"
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
app)
|
|
||||||
# Offer exactly one app completion.
|
|
||||||
if (( COMP_CWORD == cmd_index + 1 )); then
|
|
||||||
_abra_complete_apps "$cur"
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
recipe)
|
|
||||||
# Offer exactly one app completion.
|
|
||||||
if (( COMP_CWORD == cmd_index + 1 )); then
|
|
||||||
_abra_complete_recipes "$cur"
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
#help)
|
|
||||||
# # Offer exactly one command name completion.
|
|
||||||
# if (( COMP_CWORD == cmd_index + 1 )); then
|
|
||||||
# COMPREPLY=($(compgen -W "$cmds" -- "$cur"))
|
|
||||||
# fi
|
|
||||||
# ;;
|
|
||||||
*)
|
|
||||||
# Unknown command or unknowable argument.
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
complete -o default -F _abra_complete abra
|
|
||||||
@@ -1,217 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
# shellcheck disable=SC2154,SC2034
|
|
||||||
|
|
||||||
ABRA_VERSION="10.0.1"
|
|
||||||
GIT_URL="https://git.coopcloud.tech/coop-cloud/abra"
|
|
||||||
ABRA_SRC="$GIT_URL/raw/tag/$ABRA_VERSION/abra"
|
|
||||||
ABRA_DIR="${ABRA_DIR:-$HOME/.abra}"
|
|
||||||
|
|
||||||
DOC="
|
|
||||||
abra command-line installer script
|
|
||||||
|
|
||||||
Usage:
|
|
||||||
installer [options]
|
|
||||||
|
|
||||||
Options:
|
|
||||||
-h, --help Show this message and exit
|
|
||||||
-d, --dev Install bleeding edge development version
|
|
||||||
-n, --no-prompt Don't prompt for input and run non-interactively
|
|
||||||
-p, --no-deps Don't attempt to install system dependencies
|
|
||||||
"
|
|
||||||
|
|
||||||
# docopt parser below, refresh this parser with `docopt.sh installer`
|
|
||||||
# shellcheck disable=2016,1075
|
|
||||||
docopt() { parse() { if ${DOCOPT_DOC_CHECK:-true}; then local doc_hash
|
|
||||||
if doc_hash=$(printf "%s" "$DOC" | (sha256sum 2>/dev/null || shasum -a 256)); then
|
|
||||||
if [[ ${doc_hash:0:5} != "$digest" ]]; then
|
|
||||||
stderr "The current usage doc (${doc_hash:0:5}) does not match \
|
|
||||||
what the parser was generated with (${digest})
|
|
||||||
Run \`docopt.sh\` to refresh the parser."; _return 70; fi; fi; fi
|
|
||||||
local root_idx=$1; shift; argv=("$@"); parsed_params=(); parsed_values=()
|
|
||||||
left=(); testdepth=0; local arg; while [[ ${#argv[@]} -gt 0 ]]; do
|
|
||||||
if [[ ${argv[0]} = "--" ]]; then for arg in "${argv[@]}"; do
|
|
||||||
parsed_params+=('a'); parsed_values+=("$arg"); done; break
|
|
||||||
elif [[ ${argv[0]} = --* ]]; then parse_long
|
|
||||||
elif [[ ${argv[0]} = -* && ${argv[0]} != "-" ]]; then parse_shorts
|
|
||||||
elif ${DOCOPT_OPTIONS_FIRST:-false}; then for arg in "${argv[@]}"; do
|
|
||||||
parsed_params+=('a'); parsed_values+=("$arg"); done; break; else
|
|
||||||
parsed_params+=('a'); parsed_values+=("${argv[0]}"); argv=("${argv[@]:1}"); fi
|
|
||||||
done; local idx; if ${DOCOPT_ADD_HELP:-true}; then
|
|
||||||
for idx in "${parsed_params[@]}"; do [[ $idx = 'a' ]] && continue
|
|
||||||
if [[ ${shorts[$idx]} = "-h" || ${longs[$idx]} = "--help" ]]; then
|
|
||||||
stdout "$trimmed_doc"; _return 0; fi; done; fi
|
|
||||||
if [[ ${DOCOPT_PROGRAM_VERSION:-false} != 'false' ]]; then
|
|
||||||
for idx in "${parsed_params[@]}"; do [[ $idx = 'a' ]] && continue
|
|
||||||
if [[ ${longs[$idx]} = "--version" ]]; then stdout "$DOCOPT_PROGRAM_VERSION"
|
|
||||||
_return 0; fi; done; fi; local i=0; while [[ $i -lt ${#parsed_params[@]} ]]; do
|
|
||||||
left+=("$i"); ((i++)) || true; done
|
|
||||||
if ! required "$root_idx" || [ ${#left[@]} -gt 0 ]; then error; fi; return 0; }
|
|
||||||
parse_shorts() { local token=${argv[0]}; local value; argv=("${argv[@]:1}")
|
|
||||||
[[ $token = -* && $token != --* ]] || _return 88; local remaining=${token#-}
|
|
||||||
while [[ -n $remaining ]]; do local short="-${remaining:0:1}"
|
|
||||||
remaining="${remaining:1}"; local i=0; local similar=(); local match=false
|
|
||||||
for o in "${shorts[@]}"; do if [[ $o = "$short" ]]; then similar+=("$short")
|
|
||||||
[[ $match = false ]] && match=$i; fi; ((i++)) || true; done
|
|
||||||
if [[ ${#similar[@]} -gt 1 ]]; then
|
|
||||||
error "${short} is specified ambiguously ${#similar[@]} times"
|
|
||||||
elif [[ ${#similar[@]} -lt 1 ]]; then match=${#shorts[@]}; value=true
|
|
||||||
shorts+=("$short"); longs+=(''); argcounts+=(0); else value=false
|
|
||||||
if [[ ${argcounts[$match]} -ne 0 ]]; then if [[ $remaining = '' ]]; then
|
|
||||||
if [[ ${#argv[@]} -eq 0 || ${argv[0]} = '--' ]]; then
|
|
||||||
error "${short} requires argument"; fi; value=${argv[0]}; argv=("${argv[@]:1}")
|
|
||||||
else value=$remaining; remaining=''; fi; fi; if [[ $value = false ]]; then
|
|
||||||
value=true; fi; fi; parsed_params+=("$match"); parsed_values+=("$value"); done
|
|
||||||
}; parse_long() { local token=${argv[0]}; local long=${token%%=*}
|
|
||||||
local value=${token#*=}; local argcount; argv=("${argv[@]:1}")
|
|
||||||
[[ $token = --* ]] || _return 88; if [[ $token = *=* ]]; then eq='='; else eq=''
|
|
||||||
value=false; fi; local i=0; local similar=(); local match=false
|
|
||||||
for o in "${longs[@]}"; do if [[ $o = "$long" ]]; then similar+=("$long")
|
|
||||||
[[ $match = false ]] && match=$i; fi; ((i++)) || true; done
|
|
||||||
if [[ $match = false ]]; then i=0; for o in "${longs[@]}"; do
|
|
||||||
if [[ $o = $long* ]]; then similar+=("$long"); [[ $match = false ]] && match=$i
|
|
||||||
fi; ((i++)) || true; done; fi; if [[ ${#similar[@]} -gt 1 ]]; then
|
|
||||||
error "${long} is not a unique prefix: ${similar[*]}?"
|
|
||||||
elif [[ ${#similar[@]} -lt 1 ]]; then
|
|
||||||
[[ $eq = '=' ]] && argcount=1 || argcount=0; match=${#shorts[@]}
|
|
||||||
[[ $argcount -eq 0 ]] && value=true; shorts+=(''); longs+=("$long")
|
|
||||||
argcounts+=("$argcount"); else if [[ ${argcounts[$match]} -eq 0 ]]; then
|
|
||||||
if [[ $value != false ]]; then
|
|
||||||
error "${longs[$match]} must not have an argument"; fi
|
|
||||||
elif [[ $value = false ]]; then
|
|
||||||
if [[ ${#argv[@]} -eq 0 || ${argv[0]} = '--' ]]; then
|
|
||||||
error "${long} requires argument"; fi; value=${argv[0]}; argv=("${argv[@]:1}")
|
|
||||||
fi; if [[ $value = false ]]; then value=true; fi; fi; parsed_params+=("$match")
|
|
||||||
parsed_values+=("$value"); }; required() { local initial_left=("${left[@]}")
|
|
||||||
local node_idx; ((testdepth++)) || true; for node_idx in "$@"; do
|
|
||||||
if ! "node_$node_idx"; then left=("${initial_left[@]}"); ((testdepth--)) || true
|
|
||||||
return 1; fi; done; if [[ $((--testdepth)) -eq 0 ]]; then
|
|
||||||
left=("${initial_left[@]}"); for node_idx in "$@"; do "node_$node_idx"; done; fi
|
|
||||||
return 0; }; optional() { local node_idx; for node_idx in "$@"; do
|
|
||||||
"node_$node_idx"; done; return 0; }; switch() { local i
|
|
||||||
for i in "${!left[@]}"; do local l=${left[$i]}
|
|
||||||
if [[ ${parsed_params[$l]} = "$2" ]]; then
|
|
||||||
left=("${left[@]:0:$i}" "${left[@]:((i+1))}")
|
|
||||||
[[ $testdepth -gt 0 ]] && return 0; if [[ $3 = true ]]; then
|
|
||||||
eval "((var_$1++))" || true; else eval "var_$1=true"; fi; return 0; fi; done
|
|
||||||
return 1; }; stdout() { printf -- "cat <<'EOM'\n%s\nEOM\n" "$1"; }; stderr() {
|
|
||||||
printf -- "cat <<'EOM' >&2\n%s\nEOM\n" "$1"; }; error() {
|
|
||||||
[[ -n $1 ]] && stderr "$1"; stderr "$usage"; _return 1; }; _return() {
|
|
||||||
printf -- "exit %d\n" "$1"; exit "$1"; }; set -e; trimmed_doc=${DOC:1:333}
|
|
||||||
usage=${DOC:37:28}; digest=36916; shorts=(-h -d -n -p)
|
|
||||||
longs=(--help --dev --no-prompt --no-deps); argcounts=(0 0 0 0); node_0(){
|
|
||||||
switch __help 0; }; node_1(){ switch __dev 1; }; node_2(){ switch __no_prompt 2
|
|
||||||
}; node_3(){ switch __no_deps 3; }; node_4(){ optional 0 1 2 3; }; node_5(){
|
|
||||||
optional 4; }; node_6(){ required 5; }; node_7(){ required 6; }
|
|
||||||
cat <<<' docopt_exit() { [[ -n $1 ]] && printf "%s\n" "$1" >&2
|
|
||||||
printf "%s\n" "${DOC:37:28}" >&2; exit 1; }'; unset var___help var___dev \
|
|
||||||
var___no_prompt var___no_deps; parse 7 "$@"; local prefix=${DOCOPT_PREFIX:-''}
|
|
||||||
unset "${prefix}__help" "${prefix}__dev" "${prefix}__no_prompt" \
|
|
||||||
"${prefix}__no_deps"; eval "${prefix}"'__help=${var___help:-false}'
|
|
||||||
eval "${prefix}"'__dev=${var___dev:-false}'
|
|
||||||
eval "${prefix}"'__no_prompt=${var___no_prompt:-false}'
|
|
||||||
eval "${prefix}"'__no_deps=${var___no_deps:-false}'; local docopt_i=1
|
|
||||||
[[ $BASH_VERSION =~ ^4.3 ]] && docopt_i=2; for ((;docopt_i>0;docopt_i--)); do
|
|
||||||
declare -p "${prefix}__help" "${prefix}__dev" "${prefix}__no_prompt" \
|
|
||||||
"${prefix}__no_deps"; done; }
|
|
||||||
# docopt parser above, complete command for generating this parser is `docopt.sh installer`
|
|
||||||
|
|
||||||
function prompt_confirm {
|
|
||||||
if [ "$no_prompt" == "true" ]; then
|
|
||||||
return
|
|
||||||
fi
|
|
||||||
|
|
||||||
read -rp "Continue? [y/N]? " choice
|
|
||||||
|
|
||||||
case "$choice" in
|
|
||||||
y|Y ) return ;;
|
|
||||||
* ) exit;;
|
|
||||||
esac
|
|
||||||
}
|
|
||||||
|
|
||||||
function show_banner {
|
|
||||||
echo ""
|
|
||||||
echo " ____ ____ _ _ "
|
|
||||||
echo " / ___|___ ___ _ __ / ___| | ___ _ _ __| |"
|
|
||||||
echo " | | / _ \ _____ / _ \| '_ \ | | | |/ _ \| | | |/ _' |"
|
|
||||||
echo " | |__| (_) |_____| (_) | |_) | | |___| | (_) | |_| | (_| |"
|
|
||||||
echo " \____\___/ \___/| .__/ \____|_|\___/ \__,_|\__,_|"
|
|
||||||
echo " |_|"
|
|
||||||
echo ""
|
|
||||||
}
|
|
||||||
|
|
||||||
function install_docker {
|
|
||||||
sudo apt-get remove docker docker-engine docker.io containerd runc
|
|
||||||
sudo apt-get install -yq \
|
|
||||||
apt-transport-https \
|
|
||||||
ca-certificates \
|
|
||||||
gnupg \
|
|
||||||
lsb-release
|
|
||||||
curl -fsSL https://download.docker.com/linux/debian/gpg | sudo gpg --dearmor -o /usr/share/keyrings/docker-archive-keyring.gpg
|
|
||||||
echo \
|
|
||||||
"deb [arch=amd64 signed-by=/usr/share/keyrings/docker-archive-keyring.gpg] https://download.docker.com/linux/debian \
|
|
||||||
$(lsb_release -cs) stable" | sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
|
|
||||||
sudo apt-get update
|
|
||||||
sudo apt-get install -yq docker-ce docker-ce-cli containerd.io
|
|
||||||
}
|
|
||||||
|
|
||||||
function install_requirements {
|
|
||||||
if [ -f "/etc/debian_version" ]; then
|
|
||||||
echo "Detected Debian based distribution, attempting to install system requirements..."
|
|
||||||
|
|
||||||
sudo apt update && sudo apt install -y \
|
|
||||||
passwdqc \
|
|
||||||
pwgen
|
|
||||||
|
|
||||||
echo "Install Docker (https://docs.docker.com/engine/install/debian/)?"
|
|
||||||
prompt_confirm
|
|
||||||
install_docker
|
|
||||||
else
|
|
||||||
echo "Sorry, we only support Debian based distributions at the moment"
|
|
||||||
echo "You'll have to install the requirements manually for your distribution"
|
|
||||||
echo "See https://git.autonomic.zone/coop-cloud/abra#requirements for more"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
function install_abra_release {
|
|
||||||
mkdir -p "$HOME/.local/bin"
|
|
||||||
curl "$ABRA_SRC" > "$HOME/.local/bin/abra"
|
|
||||||
chmod +x "$HOME/.local/bin/abra"
|
|
||||||
echo "abra installed to $HOME/.local/bin/abra"
|
|
||||||
}
|
|
||||||
|
|
||||||
function install_abra_dev {
|
|
||||||
mkdir -p "$ABRA_DIR/"
|
|
||||||
if [[ ! -d "$ABRA_DIR/src" ]]; then
|
|
||||||
git clone "$GIT_URL" "$ABRA_DIR/src"
|
|
||||||
fi
|
|
||||||
(cd "$ABRA_DIR/src" && git pull origin main && cd - || exit)
|
|
||||||
mkdir -p "$HOME/.local/bin"
|
|
||||||
ln -sf "$ABRA_DIR/src/abra" "$HOME/.local/bin/abra"
|
|
||||||
echo "abra installed to $HOME/.local/bin/abra (development bleeding edge)"
|
|
||||||
}
|
|
||||||
|
|
||||||
function run_installation {
|
|
||||||
show_banner
|
|
||||||
|
|
||||||
DOCOPT_PREFIX=installer_
|
|
||||||
DOCOPT_ADD_HELP=false
|
|
||||||
eval "$(docopt "$@")"
|
|
||||||
|
|
||||||
dev="$installer___dev"
|
|
||||||
no_prompt="$installer___no_prompt"
|
|
||||||
no_deps="$installer___no_deps"
|
|
||||||
|
|
||||||
if [ "$no_deps" == "false" ]; then
|
|
||||||
install_requirements
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ "$dev" == "true" ]; then
|
|
||||||
install_abra_dev
|
|
||||||
else
|
|
||||||
install_abra_release
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
run_installation "$@"
|
|
||||||
exit 0
|
|
||||||
Generated
+61
@@ -0,0 +1,61 @@
|
|||||||
|
{
|
||||||
|
"nodes": {
|
||||||
|
"flake-utils": {
|
||||||
|
"inputs": {
|
||||||
|
"systems": "systems"
|
||||||
|
},
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1731533236,
|
||||||
|
"narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=",
|
||||||
|
"owner": "numtide",
|
||||||
|
"repo": "flake-utils",
|
||||||
|
"rev": "11707dc2f618dd54ca8739b309ec4fc024de578b",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "numtide",
|
||||||
|
"repo": "flake-utils",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"nixpkgs": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1777954456,
|
||||||
|
"narHash": "sha256-hGdgeU2Nk87RAuZyYjyDjFL6LK7dAZN5RE9+hrDTkDU=",
|
||||||
|
"owner": "nixos",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"rev": "549bd84d6279f9852cae6225e372cc67fb91a4c1",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "nixos",
|
||||||
|
"ref": "nixos-unstable",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"root": {
|
||||||
|
"inputs": {
|
||||||
|
"flake-utils": "flake-utils",
|
||||||
|
"nixpkgs": "nixpkgs"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"systems": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1681028828,
|
||||||
|
"narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
|
||||||
|
"owner": "nix-systems",
|
||||||
|
"repo": "default",
|
||||||
|
"rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "nix-systems",
|
||||||
|
"repo": "default",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"root": "root",
|
||||||
|
"version": 7
|
||||||
|
}
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
{
|
||||||
|
description = "The Co-op Cloud utility belt";
|
||||||
|
|
||||||
|
inputs = {
|
||||||
|
nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable";
|
||||||
|
flake-utils.url = "github:numtide/flake-utils";
|
||||||
|
};
|
||||||
|
|
||||||
|
outputs =
|
||||||
|
{
|
||||||
|
self,
|
||||||
|
nixpkgs,
|
||||||
|
flake-utils,
|
||||||
|
}:
|
||||||
|
flake-utils.lib.eachDefaultSystem (
|
||||||
|
system:
|
||||||
|
let
|
||||||
|
pkgs = nixpkgs.legacyPackages.${system};
|
||||||
|
in
|
||||||
|
{
|
||||||
|
packages = rec {
|
||||||
|
abra = pkgs.callPackage ./package.nix { };
|
||||||
|
default = abra;
|
||||||
|
};
|
||||||
|
apps = rec {
|
||||||
|
abra = flake-utils.lib.mkApp { drv = self.packages.${system}.abra; };
|
||||||
|
default = abra;
|
||||||
|
};
|
||||||
|
devShells.default = pkgs.mkShell {
|
||||||
|
packages = with pkgs; [
|
||||||
|
go_1_26
|
||||||
|
gnumake
|
||||||
|
];
|
||||||
|
};
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,163 @@
|
|||||||
|
module coopcloud.tech/abra
|
||||||
|
|
||||||
|
go 1.26.0
|
||||||
|
|
||||||
|
require (
|
||||||
|
coopcloud.tech/tagcmp v0.0.0-20250818180036-0ec1b205b5ca
|
||||||
|
git.coopcloud.tech/toolshed/godotenv v1.5.2-0.20250103171850-4d0ca41daa5c
|
||||||
|
github.com/AlecAivazis/survey/v2 v2.3.7
|
||||||
|
github.com/charmbracelet/bubbletea v1.3.10
|
||||||
|
github.com/charmbracelet/lipgloss v1.1.0
|
||||||
|
github.com/charmbracelet/log v1.0.0
|
||||||
|
github.com/distribution/reference v0.6.0
|
||||||
|
github.com/docker/cli v28.4.0+incompatible
|
||||||
|
github.com/docker/docker v28.5.2+incompatible
|
||||||
|
github.com/docker/go-units v0.5.0
|
||||||
|
github.com/go-git/go-git/v5 v5.17.2
|
||||||
|
github.com/google/go-cmp v0.7.0
|
||||||
|
github.com/leonelquinteros/gotext v1.7.2
|
||||||
|
github.com/moby/sys/signal v0.7.1
|
||||||
|
github.com/moby/term v0.5.2
|
||||||
|
github.com/pkg/errors v0.9.1
|
||||||
|
github.com/schollz/progressbar/v3 v3.19.0
|
||||||
|
golang.org/x/term v0.41.0
|
||||||
|
gopkg.in/yaml.v3 v3.0.1
|
||||||
|
gotest.tools/v3 v3.5.2
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
dario.cat/mergo v1.0.2 // indirect
|
||||||
|
github.com/Azure/go-ansiterm v0.0.0-20250102033503-faa5f7b0171c // indirect
|
||||||
|
github.com/BurntSushi/toml v1.6.0 // indirect
|
||||||
|
github.com/Microsoft/go-winio v0.6.2 // indirect
|
||||||
|
github.com/ProtonMail/go-crypto v1.4.1 // indirect
|
||||||
|
github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect
|
||||||
|
github.com/beorn7/perks v1.0.1 // indirect
|
||||||
|
github.com/cenkalti/backoff/v4 v4.3.0 // indirect
|
||||||
|
github.com/cenkalti/backoff/v5 v5.0.3 // indirect
|
||||||
|
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
||||||
|
github.com/charmbracelet/colorprofile v0.4.3 // indirect
|
||||||
|
github.com/charmbracelet/x/ansi v0.11.6 // indirect
|
||||||
|
github.com/charmbracelet/x/cellbuf v0.0.15 // indirect
|
||||||
|
github.com/charmbracelet/x/exp/golden v0.0.0-20241011142426-46044092ad91 // indirect
|
||||||
|
github.com/charmbracelet/x/term v0.2.2 // indirect
|
||||||
|
github.com/clipperhouse/displaywidth v0.11.0 // indirect
|
||||||
|
github.com/clipperhouse/uax29/v2 v2.7.0 // indirect
|
||||||
|
github.com/cloudflare/circl v1.6.3 // indirect
|
||||||
|
github.com/containerd/errdefs v1.0.0 // indirect
|
||||||
|
github.com/containerd/errdefs/pkg v0.3.0 // indirect
|
||||||
|
github.com/containerd/log v0.1.0 // indirect
|
||||||
|
github.com/containerd/platforms v0.2.1 // indirect
|
||||||
|
github.com/cpuguy83/go-md2man/v2 v2.0.7 // indirect
|
||||||
|
github.com/cyphar/filepath-securejoin v0.6.1 // indirect
|
||||||
|
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||||
|
github.com/docker/distribution v2.8.3+incompatible // indirect
|
||||||
|
github.com/docker/go-connections v0.6.0 // indirect
|
||||||
|
github.com/docker/go-metrics v0.0.1 // indirect
|
||||||
|
github.com/docker/libtrust v0.0.0-20160708172513-aabc10ec26b7 // indirect
|
||||||
|
github.com/emirpasic/gods v1.18.1 // indirect
|
||||||
|
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect
|
||||||
|
github.com/felixge/httpsnoop v1.0.4 // indirect
|
||||||
|
github.com/ghodss/yaml v1.0.0 // indirect
|
||||||
|
github.com/go-git/gcfg v1.5.1-0.20230307220236-3a3c6141e376 // indirect
|
||||||
|
github.com/go-git/go-billy/v5 v5.8.0 // indirect
|
||||||
|
github.com/go-logfmt/logfmt v0.6.1 // indirect
|
||||||
|
github.com/go-logr/logr v1.4.3 // indirect
|
||||||
|
github.com/go-logr/stdr v1.2.2 // indirect
|
||||||
|
github.com/go-viper/mapstructure/v2 v2.5.0 // indirect
|
||||||
|
github.com/golang/groupcache v0.0.0-20241129210726-2c02b8208cf8 // indirect
|
||||||
|
github.com/google/uuid v1.6.0 // indirect
|
||||||
|
github.com/grpc-ecosystem/grpc-gateway/v2 v2.28.0 // indirect
|
||||||
|
github.com/hashicorp/go-cleanhttp v0.5.2 // indirect
|
||||||
|
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
||||||
|
github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 // indirect
|
||||||
|
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 // indirect
|
||||||
|
github.com/kevinburke/ssh_config v1.6.0 // indirect
|
||||||
|
github.com/klauspost/compress v1.18.5 // indirect
|
||||||
|
github.com/klauspost/cpuid/v2 v2.3.0 // indirect
|
||||||
|
github.com/lucasb-eyer/go-colorful v1.4.0 // indirect
|
||||||
|
github.com/mattn/go-colorable v0.1.14 // indirect
|
||||||
|
github.com/mattn/go-isatty v0.0.20 // indirect
|
||||||
|
github.com/mattn/go-localereader v0.0.1 // indirect
|
||||||
|
github.com/mattn/go-runewidth v0.0.21 // indirect
|
||||||
|
github.com/mgutz/ansi v0.0.0-20200706080929-d51e80ef957d // indirect
|
||||||
|
github.com/miekg/pkcs11 v1.1.1 // indirect
|
||||||
|
github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db // indirect
|
||||||
|
github.com/moby/docker-image-spec v1.3.1 // indirect
|
||||||
|
github.com/moby/go-archive v0.1.0 // indirect
|
||||||
|
github.com/moby/sys/atomicwriter v0.1.0 // indirect
|
||||||
|
github.com/moby/sys/user v0.4.0 // indirect
|
||||||
|
github.com/moby/sys/userns v0.1.0 // indirect
|
||||||
|
github.com/morikuni/aec v1.1.0 // indirect
|
||||||
|
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 // indirect
|
||||||
|
github.com/muesli/cancelreader v0.2.2 // indirect
|
||||||
|
github.com/muesli/termenv v0.16.0 // indirect
|
||||||
|
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect
|
||||||
|
github.com/opencontainers/go-digest v1.0.0 // indirect
|
||||||
|
github.com/opencontainers/runc v1.1.13 // indirect
|
||||||
|
github.com/opencontainers/runtime-spec v1.1.0 // indirect
|
||||||
|
github.com/pjbgf/sha1cd v0.5.0 // indirect
|
||||||
|
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||||
|
github.com/prometheus/client_model v0.6.2 // indirect
|
||||||
|
github.com/prometheus/common v0.67.5 // indirect
|
||||||
|
github.com/prometheus/procfs v0.20.1 // indirect
|
||||||
|
github.com/rivo/uniseg v0.4.7 // indirect
|
||||||
|
github.com/russross/blackfriday/v2 v2.1.0 // indirect
|
||||||
|
github.com/sirupsen/logrus v1.9.4 // indirect
|
||||||
|
github.com/skeema/knownhosts v1.3.2 // indirect
|
||||||
|
github.com/spf13/pflag v1.0.10 // indirect
|
||||||
|
github.com/xanzy/ssh-agent v0.3.3 // indirect
|
||||||
|
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect
|
||||||
|
github.com/xeipuuv/gojsonschema v1.2.0 // indirect
|
||||||
|
github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect
|
||||||
|
go.opentelemetry.io/auto/sdk v1.2.1 // indirect
|
||||||
|
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.67.0 // indirect
|
||||||
|
go.opentelemetry.io/otel v1.42.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.42.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.42.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.42.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.19.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/metric v1.42.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/sdk v1.42.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/sdk/metric v1.42.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/trace v1.42.0 // indirect
|
||||||
|
go.opentelemetry.io/proto/otlp v1.10.0 // indirect
|
||||||
|
go.yaml.in/yaml/v2 v2.4.4 // indirect
|
||||||
|
go.yaml.in/yaml/v3 v3.0.4 // indirect
|
||||||
|
golang.org/x/crypto v0.49.0 // indirect
|
||||||
|
golang.org/x/exp v0.0.0-20260312153236-7ab1446f8b90 // indirect
|
||||||
|
golang.org/x/net v0.52.0 // indirect
|
||||||
|
golang.org/x/text v0.35.0 // indirect
|
||||||
|
golang.org/x/time v0.15.0 // indirect
|
||||||
|
google.golang.org/genproto/googleapis/api v0.0.0-20260401024825-9d38bb4040a9 // indirect
|
||||||
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20260401024825-9d38bb4040a9 // indirect
|
||||||
|
google.golang.org/grpc v1.80.0 // indirect
|
||||||
|
google.golang.org/protobuf v1.36.11 // indirect
|
||||||
|
gopkg.in/warnings.v0 v0.1.2 // indirect
|
||||||
|
gopkg.in/yaml.v2 v2.4.0 // indirect
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/containers/image v3.0.2+incompatible
|
||||||
|
github.com/containers/storage v1.38.2 // indirect
|
||||||
|
github.com/decentral1se/passgen v1.0.1
|
||||||
|
github.com/docker/docker-credential-helpers v0.9.5 // indirect
|
||||||
|
github.com/fvbommel/sortorder v1.1.0 // indirect
|
||||||
|
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect
|
||||||
|
github.com/gorilla/mux v1.8.1 // indirect
|
||||||
|
github.com/hashicorp/go-retryablehttp v0.7.8
|
||||||
|
github.com/moby/patternmatcher v0.6.1 // indirect
|
||||||
|
github.com/moby/sys/sequential v0.6.0 // indirect
|
||||||
|
github.com/opencontainers/image-spec v1.1.1 // indirect
|
||||||
|
github.com/prometheus/client_golang v1.23.2 // indirect
|
||||||
|
github.com/sergi/go-diff v1.4.0 // indirect
|
||||||
|
github.com/spf13/cobra v1.10.1
|
||||||
|
github.com/stretchr/testify v1.11.1
|
||||||
|
github.com/theupdateframework/notary v0.7.0 // indirect
|
||||||
|
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb // indirect
|
||||||
|
golang.org/x/sys v0.42.0
|
||||||
|
)
|
||||||
|
|
||||||
|
replace github.com/docker/cli v28.4.0+incompatible => git.coopcloud.tech/toolshed/docker-cli v28.5.3-0.20260202112816-30df2d0b3a00+incompatible
|
||||||
|
|
||||||
|
replace github.com/spf13/cobra => github.com/decentral1se/cobra v1.10.2-i18n
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
.PHONY: test shellcheck docopt release-installer build push deploy-docopt symlink
|
|
||||||
|
|
||||||
test:
|
|
||||||
@sudo DOCKER_CONTEXT=default docker run \
|
|
||||||
-v $$(pwd):/workdir \
|
|
||||||
--privileged \
|
|
||||||
-d \
|
|
||||||
--name=abra-test-dind \
|
|
||||||
-e DOCKER_TLS_CERTDIR="" \
|
|
||||||
decentral1se/docker-dind-bats-kcov \
|
|
||||||
@DOCKER_CONTEXT=default sudo docker exec \
|
|
||||||
-it \
|
|
||||||
abra-test-dind \
|
|
||||||
sh -c "cd /workdir && bats /workdir/tests"
|
|
||||||
@DOCKER_CONTEXT=default sudo docker stop abra-test-dind
|
|
||||||
@DOCKER_CONTEXT=default sudo docker rm abra-test-dind
|
|
||||||
|
|
||||||
shellcheck:
|
|
||||||
@docker run \
|
|
||||||
-it \
|
|
||||||
--rm \
|
|
||||||
-v $$(pwd):/workdir \
|
|
||||||
koalaman/shellcheck-alpine \
|
|
||||||
sh -c "shellcheck /workdir/abra && \
|
|
||||||
shellcheck /workdir/bin/*.sh && \
|
|
||||||
shellcheck /workdir/deploy/install.abra.coopcloud.tech/installer"
|
|
||||||
|
|
||||||
docopt:
|
|
||||||
@if [ ! -d ".venv" ]; then \
|
|
||||||
python3 -m venv .venv && \
|
|
||||||
.venv/bin/pip install -U pip setuptools wheel && \
|
|
||||||
.venv/bin/pip install docopt-sh; \
|
|
||||||
fi
|
|
||||||
.venv/bin/docopt.sh abra
|
|
||||||
|
|
||||||
deploy-docopt:
|
|
||||||
@if [ ! -d ".venv" ]; then \
|
|
||||||
python3 -m venv .venv && \
|
|
||||||
.venv/bin/pip install -U pip setuptools wheel && \
|
|
||||||
.venv/bin/pip install docopt-sh; \
|
|
||||||
fi
|
|
||||||
.venv/bin/docopt.sh deploy/install.abra.coopcloud.tech/installer
|
|
||||||
|
|
||||||
release-installer:
|
|
||||||
@DOCKER_CONTEXT=swarm.autonomic.zone \
|
|
||||||
docker stack rm abra-installer-script && \
|
|
||||||
cd deploy/install.abra.coopcloud.tech && \
|
|
||||||
DOCKER_CONTEXT=swarm.autonomic.zone docker stack deploy -c compose.yml abra-installer-script
|
|
||||||
|
|
||||||
build:
|
|
||||||
@docker build -t thecoopcloud/abra .
|
|
||||||
|
|
||||||
push: build
|
|
||||||
@docker push thecoopcloud/abra
|
|
||||||
|
|
||||||
symlink:
|
|
||||||
@mkdir -p ~/.abra/servers/ && \
|
|
||||||
ln -srf tests/default ~/.abra/servers && \
|
|
||||||
ln -srf tests/apps/* ~/.abra/apps
|
|
||||||
+47
@@ -0,0 +1,47 @@
|
|||||||
|
{
|
||||||
|
buildGo126Module,
|
||||||
|
fetchgit,
|
||||||
|
lib,
|
||||||
|
installShellFiles,
|
||||||
|
}:
|
||||||
|
|
||||||
|
buildGo126Module rec {
|
||||||
|
pname = "abra";
|
||||||
|
version = "0.13.0-beta";
|
||||||
|
rev = "06a57ded025a43c80f94d4e65299add8a31830dc";
|
||||||
|
|
||||||
|
src = fetchgit {
|
||||||
|
url = "https://git.coopcloud.tech/toolshed/abra.git";
|
||||||
|
tag = version;
|
||||||
|
hash = "sha256-rgoK0TY0WLSQ39lPvVM80zW/qJF40VFBSxYDOaKXZQo=";
|
||||||
|
};
|
||||||
|
|
||||||
|
vendorHash = null;
|
||||||
|
|
||||||
|
nativeBuildInputs = [
|
||||||
|
installShellFiles
|
||||||
|
];
|
||||||
|
|
||||||
|
env.CGO_ENABLED = 0;
|
||||||
|
ldflags = [
|
||||||
|
"-s -w -X 'main.Commit=${rev}' -X 'main.Version=${version}'"
|
||||||
|
];
|
||||||
|
|
||||||
|
doCheck = false;
|
||||||
|
postInstall = ''
|
||||||
|
export ABRA_DIR="$out"
|
||||||
|
$out/bin/abra autocomplete bash >abra.bash
|
||||||
|
$out/bin/abra autocomplete fish >abra.fish
|
||||||
|
$out/bin/abra autocomplete zsh >abra.zsh
|
||||||
|
installShellCompletion abra.{bash,fish,zsh}
|
||||||
|
'';
|
||||||
|
|
||||||
|
meta = with lib; {
|
||||||
|
description = "The Co-op Cloud utility belt";
|
||||||
|
homepage = "https://docs.coopcloud.tech/abra";
|
||||||
|
changelog = "https://git.coopcloud.tech/toolshed/abra/releases/tag/${version}";
|
||||||
|
mainProgram = "abra";
|
||||||
|
license = licenses.gpl3Plus;
|
||||||
|
maintainers = "devydave";
|
||||||
|
};
|
||||||
|
}
|
||||||
+701
@@ -0,0 +1,701 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"regexp"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/client"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/envfile"
|
||||||
|
"coopcloud.tech/abra/pkg/formatter"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/convert"
|
||||||
|
"coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
loader "coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
composetypes "github.com/docker/cli/cli/compose/types"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
"github.com/schollz/progressbar/v3"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Get retrieves an app
|
||||||
|
func Get(appName string) (App, error) {
|
||||||
|
files, err := LoadAppFiles("")
|
||||||
|
if err != nil {
|
||||||
|
return App{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
app, err := GetApp(files, appName)
|
||||||
|
if err != nil {
|
||||||
|
return App{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("loaded app %s: %s", appName, app))
|
||||||
|
|
||||||
|
return app, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetApp loads an apps settings, reading it from file, in preparation to use
|
||||||
|
// it. It should only be used when ready to use the env file to keep IO
|
||||||
|
// operations down.
|
||||||
|
func GetApp(apps AppFiles, name AppName) (App, error) {
|
||||||
|
appFile, exists := apps[name]
|
||||||
|
if !exists {
|
||||||
|
return App{}, errors.New(i18n.G("cannot find app with name %s", name))
|
||||||
|
}
|
||||||
|
|
||||||
|
app, err := ReadAppEnvFile(appFile, name)
|
||||||
|
if err != nil {
|
||||||
|
return App{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return app, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetApps returns a slice of Apps with their env files read from a given
|
||||||
|
// slice of AppFiles.
|
||||||
|
func GetApps(appFiles AppFiles, recipeFilter string) ([]App, error) {
|
||||||
|
var apps []App
|
||||||
|
|
||||||
|
for name := range appFiles {
|
||||||
|
app, err := GetApp(appFiles, name)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipeFilter != "" {
|
||||||
|
if app.Recipe.Name == recipeFilter {
|
||||||
|
apps = append(apps, app)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
apps = append(apps, app)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return apps, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// App reprents an app with its env file read into memory
|
||||||
|
type App struct {
|
||||||
|
Name AppName
|
||||||
|
Recipe recipe.Recipe
|
||||||
|
Domain string
|
||||||
|
Env envfile.AppEnv
|
||||||
|
Server string
|
||||||
|
Path string
|
||||||
|
}
|
||||||
|
|
||||||
|
// String outputs a human-friendly string representation.
|
||||||
|
func (a App) String() string {
|
||||||
|
out := fmt.Sprintf("{name: %s, ", a.Name)
|
||||||
|
out += fmt.Sprintf("recipe: %s, ", a.Recipe)
|
||||||
|
out += fmt.Sprintf("domain: %s, ", a.Domain)
|
||||||
|
out += fmt.Sprintf("env %s, ", a.Env)
|
||||||
|
out += fmt.Sprintf("server %s, ", a.Server)
|
||||||
|
out += fmt.Sprintf("path %s}", a.Path)
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
// Type aliases to make code hints easier to understand
|
||||||
|
|
||||||
|
// AppName is AppName
|
||||||
|
type AppName = string
|
||||||
|
|
||||||
|
// AppFile represents app env files on disk without reading the contents
|
||||||
|
type AppFile struct {
|
||||||
|
Path string
|
||||||
|
Server string
|
||||||
|
}
|
||||||
|
|
||||||
|
// AppFiles is a slice of appfiles
|
||||||
|
type AppFiles map[AppName]AppFile
|
||||||
|
|
||||||
|
// See documentation of config.StackName
|
||||||
|
func (a App) StackName() string {
|
||||||
|
if _, exists := a.Env["STACK_NAME"]; exists {
|
||||||
|
return a.Env["STACK_NAME"]
|
||||||
|
}
|
||||||
|
|
||||||
|
stackName := StackName(a.Name)
|
||||||
|
|
||||||
|
a.Env["STACK_NAME"] = stackName
|
||||||
|
|
||||||
|
return stackName
|
||||||
|
}
|
||||||
|
|
||||||
|
// StackName gets whatever the docker safe (uses the right delimiting
|
||||||
|
// character, e.g. "_") stack name is for the app. In general, you don't want
|
||||||
|
// to use this to show anything to end-users, you want use a.Name instead.
|
||||||
|
func StackName(appName string) string {
|
||||||
|
stackName := SanitiseAppName(appName)
|
||||||
|
|
||||||
|
if len(stackName) > config.MAX_SANITISED_APP_NAME_LENGTH {
|
||||||
|
log.Debug(i18n.G("trimming %s to %s to avoid runtime limits", stackName, stackName[:config.MAX_SANITISED_APP_NAME_LENGTH]))
|
||||||
|
stackName = stackName[:config.MAX_SANITISED_APP_NAME_LENGTH]
|
||||||
|
}
|
||||||
|
|
||||||
|
return stackName
|
||||||
|
}
|
||||||
|
|
||||||
|
// Filters retrieves app filters for querying the container runtime. By default
|
||||||
|
// it filters on all services in the app. It is also possible to pass an
|
||||||
|
// otional list of service names, which get filtered instead.
|
||||||
|
//
|
||||||
|
// Due to upstream issues, filtering works different depending on what you're
|
||||||
|
// querying. So, for example, secrets don't work with regex! The caller needs
|
||||||
|
// to implement their own validation that the right secrets are matched. In
|
||||||
|
// order to handle these cases, we provide the `appendServiceNames` /
|
||||||
|
// `exactMatch` modifiers.
|
||||||
|
func (a App) Filters(appendServiceNames, exactMatch bool, services ...string) (filters.Args, error) {
|
||||||
|
filters := filters.NewArgs()
|
||||||
|
if len(services) > 0 {
|
||||||
|
for _, serviceName := range services {
|
||||||
|
filters.Add("name", ServiceFilter(a.StackName(), serviceName, exactMatch))
|
||||||
|
}
|
||||||
|
return filters, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// When not appending the service name, just add one filter for the whole
|
||||||
|
// stack.
|
||||||
|
if !appendServiceNames {
|
||||||
|
f := fmt.Sprintf("%s", a.StackName())
|
||||||
|
if exactMatch {
|
||||||
|
f = fmt.Sprintf("^%s", f)
|
||||||
|
}
|
||||||
|
filters.Add("name", f)
|
||||||
|
return filters, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := a.Recipe.GetComposeFiles(a.Env)
|
||||||
|
if err != nil {
|
||||||
|
return filters, err
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := stack.Deploy{Composefiles: composeFiles}
|
||||||
|
compose, err := GetAppComposeConfig(a.Recipe.Name, opts, a.Env)
|
||||||
|
if err != nil {
|
||||||
|
return filters, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
f := ServiceFilter(a.StackName(), service.Name, exactMatch)
|
||||||
|
filters.Add("name", f)
|
||||||
|
}
|
||||||
|
|
||||||
|
return filters, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ServiceFilter creates a filter string for filtering a service in the docker
|
||||||
|
// container runtime. When exact match is true, it uses regex to match the
|
||||||
|
// string exactly.
|
||||||
|
func ServiceFilter(stack, service string, exact bool) string {
|
||||||
|
if exact {
|
||||||
|
return fmt.Sprintf("^%s_%s", stack, service)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s_%s", stack, service)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByServer sort a slice of Apps
|
||||||
|
type ByServer []App
|
||||||
|
|
||||||
|
func (a ByServer) Len() int { return len(a) }
|
||||||
|
func (a ByServer) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
||||||
|
func (a ByServer) Less(i, j int) bool {
|
||||||
|
return strings.ToLower(a[i].Server) < strings.ToLower(a[j].Server)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByServerAndRecipe sort a slice of Apps
|
||||||
|
type ByServerAndRecipe []App
|
||||||
|
|
||||||
|
func (a ByServerAndRecipe) Len() int { return len(a) }
|
||||||
|
func (a ByServerAndRecipe) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
||||||
|
func (a ByServerAndRecipe) Less(i, j int) bool {
|
||||||
|
if a[i].Server == a[j].Server {
|
||||||
|
return strings.ToLower(a[i].Recipe.Name) < strings.ToLower(a[j].Recipe.Name)
|
||||||
|
}
|
||||||
|
return strings.ToLower(a[i].Server) < strings.ToLower(a[j].Server)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByRecipe sort a slice of Apps
|
||||||
|
type ByRecipe []App
|
||||||
|
|
||||||
|
func (a ByRecipe) Len() int { return len(a) }
|
||||||
|
func (a ByRecipe) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
||||||
|
func (a ByRecipe) Less(i, j int) bool {
|
||||||
|
return strings.ToLower(a[i].Recipe.Name) < strings.ToLower(a[j].Recipe.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByName sort a slice of Apps
|
||||||
|
type ByName []App
|
||||||
|
|
||||||
|
func (a ByName) Len() int { return len(a) }
|
||||||
|
func (a ByName) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
||||||
|
func (a ByName) Less(i, j int) bool {
|
||||||
|
return strings.ToLower(a[i].Name) < strings.ToLower(a[j].Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadAppEnvFile(appFile AppFile, name AppName) (App, error) {
|
||||||
|
env, err := envfile.ReadEnv(appFile.Path)
|
||||||
|
if err != nil {
|
||||||
|
return App{}, errors.New(i18n.G("env file for %s couldn't be read: %s", name, err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
app, err := NewApp(env, name, appFile)
|
||||||
|
if err != nil {
|
||||||
|
return App{}, errors.New(i18n.G("env file for %s has issues: %s", name, err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
return app, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewApp creates new App object
|
||||||
|
func NewApp(env envfile.AppEnv, name string, appFile AppFile) (App, error) {
|
||||||
|
domain := env["DOMAIN"]
|
||||||
|
|
||||||
|
recipeName, exists := env["RECIPE"]
|
||||||
|
if !exists {
|
||||||
|
recipeName, exists = env["TYPE"]
|
||||||
|
if !exists {
|
||||||
|
return App{}, errors.New(i18n.G("%s is missing the TYPE env var?", name))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return App{
|
||||||
|
Name: name,
|
||||||
|
Domain: domain,
|
||||||
|
Recipe: recipe.Get(recipeName),
|
||||||
|
Env: env,
|
||||||
|
Server: appFile.Server,
|
||||||
|
Path: appFile.Path,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// LoadAppFiles gets all app files for a given set of servers or all servers.
|
||||||
|
func LoadAppFiles(servers ...string) (AppFiles, error) {
|
||||||
|
appFiles := make(AppFiles)
|
||||||
|
if len(servers) == 1 {
|
||||||
|
if servers[0] == "" {
|
||||||
|
// Empty servers flag, one string will always be passed
|
||||||
|
var err error
|
||||||
|
servers, err = config.GetAllFoldersInDirectory(config.SERVERS_DIR)
|
||||||
|
if err != nil {
|
||||||
|
return appFiles, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("collecting metadata from %v servers: %s", len(servers), strings.Join(servers, ", ")))
|
||||||
|
|
||||||
|
for _, server := range servers {
|
||||||
|
serverDir := path.Join(config.SERVERS_DIR, server)
|
||||||
|
files, err := config.GetAllFilesInDirectory(serverDir)
|
||||||
|
if err != nil {
|
||||||
|
return appFiles, errors.New(i18n.G("server %s doesn't exist? Run \"abra server ls\" to check", server))
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, file := range files {
|
||||||
|
appName := strings.TrimSuffix(file.Name(), ".env")
|
||||||
|
appFilePath := path.Join(config.SERVERS_DIR, server, file.Name())
|
||||||
|
appFiles[appName] = AppFile{
|
||||||
|
Path: appFilePath,
|
||||||
|
Server: server,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return appFiles, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAppServiceNames retrieves a list of app service names.
|
||||||
|
func GetAppServiceNames(appName string) ([]string, error) {
|
||||||
|
var serviceNames []string
|
||||||
|
|
||||||
|
appFiles, err := LoadAppFiles("")
|
||||||
|
if err != nil {
|
||||||
|
return serviceNames, err
|
||||||
|
}
|
||||||
|
|
||||||
|
app, err := GetApp(appFiles, appName)
|
||||||
|
if err != nil {
|
||||||
|
return serviceNames, err
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
return serviceNames, err
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := stack.Deploy{Composefiles: composeFiles}
|
||||||
|
compose, err := GetAppComposeConfig(app.Recipe.Name, opts, app.Env)
|
||||||
|
if err != nil {
|
||||||
|
return serviceNames, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
serviceNames = append(serviceNames, service.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
return serviceNames, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAppNames retrieves a list of app names.
|
||||||
|
func GetAppNames() ([]string, error) {
|
||||||
|
var appNames []string
|
||||||
|
|
||||||
|
appFiles, err := LoadAppFiles("")
|
||||||
|
if err != nil {
|
||||||
|
return appNames, err
|
||||||
|
}
|
||||||
|
|
||||||
|
apps, err := GetApps(appFiles, "")
|
||||||
|
if err != nil {
|
||||||
|
return appNames, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, app := range apps {
|
||||||
|
appNames = append(appNames, app.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
return appNames, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// TemplateAppEnvSample copies the example env file for the app into the users
|
||||||
|
// env files.
|
||||||
|
func TemplateAppEnvSample(r recipe.Recipe, appName, server, domain string) error {
|
||||||
|
envSample, err := os.ReadFile(r.SampleEnvPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
appEnvPath := path.Join(config.ABRA_DIR, "servers", server, fmt.Sprintf("%s.env", appName))
|
||||||
|
if _, err := os.Stat(appEnvPath); !os.IsNotExist(err) {
|
||||||
|
return errors.New(i18n.G("%s already exists?", appEnvPath))
|
||||||
|
}
|
||||||
|
|
||||||
|
err = os.WriteFile(appEnvPath, envSample, 0o664)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
read, err := os.ReadFile(appEnvPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
newContents := strings.Replace(
|
||||||
|
string(read),
|
||||||
|
fmt.Sprintf("%s.example.com", r.Name),
|
||||||
|
domain,
|
||||||
|
-1,
|
||||||
|
)
|
||||||
|
|
||||||
|
err = os.WriteFile(appEnvPath, []byte(newContents), 0)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("copied & templated %s to %s", r.SampleEnvPath, appEnvPath))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SanitiseAppName makes a app name usable with Docker by replacing illegal
|
||||||
|
// characters.
|
||||||
|
func SanitiseAppName(name string) string {
|
||||||
|
return strings.ReplaceAll(name, ".", "_")
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAppStatuses queries servers to check the deployment status of given apps.
|
||||||
|
func GetAppStatuses(apps []App, MachineReadable bool) (map[string]map[string]string, error) {
|
||||||
|
statuses := make(map[string]map[string]string)
|
||||||
|
|
||||||
|
servers := make(map[string]struct{})
|
||||||
|
for _, app := range apps {
|
||||||
|
if _, ok := servers[app.Server]; !ok {
|
||||||
|
servers[app.Server] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var bar *progressbar.ProgressBar
|
||||||
|
if !MachineReadable {
|
||||||
|
bar = formatter.CreateProgressbar(len(servers), i18n.G("querying remote servers..."))
|
||||||
|
}
|
||||||
|
|
||||||
|
ch := make(chan stack.StackStatus, len(servers))
|
||||||
|
for server := range servers {
|
||||||
|
cl, err := client.New(server)
|
||||||
|
if err != nil {
|
||||||
|
log.Warn(err)
|
||||||
|
ch <- stack.StackStatus{}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
go func(s string) {
|
||||||
|
ch <- stack.GetAllDeployedServices(cl, s)
|
||||||
|
if !MachineReadable {
|
||||||
|
bar.Add(1)
|
||||||
|
}
|
||||||
|
}(server)
|
||||||
|
}
|
||||||
|
|
||||||
|
for range servers {
|
||||||
|
status := <-ch
|
||||||
|
if status.Err != nil {
|
||||||
|
return statuses, status.Err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, service := range status.Services {
|
||||||
|
result := make(map[string]string)
|
||||||
|
name := service.Spec.Labels[convert.LabelNamespace]
|
||||||
|
|
||||||
|
if _, ok := statuses[name]; !ok {
|
||||||
|
result["status"] = "deployed"
|
||||||
|
}
|
||||||
|
|
||||||
|
labelKey := fmt.Sprintf("coop-cloud.%s.chaos", name)
|
||||||
|
chaos, ok := service.Spec.Labels[labelKey]
|
||||||
|
if ok {
|
||||||
|
result["chaos"] = chaos
|
||||||
|
}
|
||||||
|
|
||||||
|
labelKey = fmt.Sprintf("coop-cloud.%s.chaos-version", name)
|
||||||
|
if chaosVersion, ok := service.Spec.Labels[labelKey]; ok {
|
||||||
|
result["chaosVersion"] = chaosVersion
|
||||||
|
}
|
||||||
|
|
||||||
|
labelKey = fmt.Sprintf("coop-cloud.%s.version", name)
|
||||||
|
if version, ok := service.Spec.Labels[labelKey]; ok {
|
||||||
|
result["version"] = version
|
||||||
|
} else {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
statuses[name] = result
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("retrieved app statuses: %s", statuses))
|
||||||
|
|
||||||
|
return statuses, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAppComposeConfig retrieves a compose specification for a recipe. This
|
||||||
|
// specification is the result of a merge of all the compose.**.yml files in
|
||||||
|
// the recipe repository.
|
||||||
|
func GetAppComposeConfig(recipe string, opts stack.Deploy, appEnv envfile.AppEnv) (*composetypes.Config, error) {
|
||||||
|
compose, err := loader.LoadComposefile(opts, appEnv)
|
||||||
|
if err != nil {
|
||||||
|
return &composetypes.Config{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("retrieved %s for %s", compose.Filename, recipe))
|
||||||
|
|
||||||
|
return compose, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ExposeAllEnv exposes all env variables to the app container
|
||||||
|
func ExposeAllEnv(
|
||||||
|
stackName string,
|
||||||
|
compose *composetypes.Config,
|
||||||
|
appEnv envfile.AppEnv,
|
||||||
|
toDeployVersion string) {
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
log.Debug(i18n.G("adding env vars to %s service config", stackName))
|
||||||
|
for k, v := range appEnv {
|
||||||
|
_, exists := service.Environment[k]
|
||||||
|
if !exists {
|
||||||
|
value := v
|
||||||
|
if k == "TYPE" || k == "RECIPE" {
|
||||||
|
// NOTE(d1): don't use the wrong version from the app env
|
||||||
|
// since we are deploying a new version
|
||||||
|
value = toDeployVersion
|
||||||
|
}
|
||||||
|
service.Environment[k] = &value
|
||||||
|
log.Debug(i18n.G("%s: %s: %s", stackName, k, value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func CheckEnv(app App) ([]envfile.EnvVar, error) {
|
||||||
|
var envVars []envfile.EnvVar
|
||||||
|
|
||||||
|
envSample, err := app.Recipe.SampleEnv()
|
||||||
|
if err != nil {
|
||||||
|
return envVars, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var keys []string
|
||||||
|
for key := range envSample {
|
||||||
|
keys = append(keys, key)
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Strings(keys)
|
||||||
|
|
||||||
|
for _, key := range keys {
|
||||||
|
if _, ok := app.Env[key]; ok {
|
||||||
|
envVars = append(envVars, envfile.EnvVar{Name: key, Present: true})
|
||||||
|
} else {
|
||||||
|
envVars = append(envVars, envfile.EnvVar{Name: key, Present: false})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return envVars, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReadAbraShCmdNames reads the names of commands.
|
||||||
|
func ReadAbraShCmdNames(abraSh string) ([]string, error) {
|
||||||
|
var cmdNames []string
|
||||||
|
|
||||||
|
file, err := os.Open(abraSh)
|
||||||
|
if err != nil {
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
return cmdNames, nil
|
||||||
|
}
|
||||||
|
return cmdNames, err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
cmdNameRegex, err := regexp.Compile(`(\w+)(\(\).*\{)`)
|
||||||
|
if err != nil {
|
||||||
|
return cmdNames, err
|
||||||
|
}
|
||||||
|
|
||||||
|
scanner := bufio.NewScanner(file)
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
matches := cmdNameRegex.FindStringSubmatch(line)
|
||||||
|
if len(matches) > 0 {
|
||||||
|
cmdNames = append(cmdNames, matches[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(cmdNames) > 0 {
|
||||||
|
log.Debug(i18n.G("read %s from %s", strings.Join(cmdNames, " "), abraSh))
|
||||||
|
} else {
|
||||||
|
log.Debug(i18n.G("read 0 command names from %s", abraSh))
|
||||||
|
}
|
||||||
|
|
||||||
|
return cmdNames, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wipe removes the version from the app .env file.
|
||||||
|
func (a App) WipeRecipeVersion() error {
|
||||||
|
file, err := os.Open(a.Path)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
var (
|
||||||
|
lines []string
|
||||||
|
scanner = bufio.NewScanner(file)
|
||||||
|
)
|
||||||
|
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
if !strings.HasPrefix(line, "RECIPE=") && !strings.HasPrefix(line, "TYPE=") {
|
||||||
|
lines = append(lines, line)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.HasPrefix(line, "#") {
|
||||||
|
lines = append(lines, line)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
splitted := strings.Split(line, ":")
|
||||||
|
lines = append(lines, splitted[0])
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := os.WriteFile(a.Path, []byte(strings.Join(lines, "\n")), os.ModePerm); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("version wiped from %s.env", a.Domain))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// WriteRecipeVersion writes the recipe version to the app .env file.
|
||||||
|
func (a App) WriteRecipeVersion(version string, dryRun bool) error {
|
||||||
|
if version == config.UNKNOWN_DEFAULT {
|
||||||
|
log.Debug(i18n.G("version is unknown, skipping env write"))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Open(a.Path)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
var (
|
||||||
|
dirtyVersion string
|
||||||
|
skipped bool
|
||||||
|
lines []string
|
||||||
|
scanner = bufio.NewScanner(file)
|
||||||
|
)
|
||||||
|
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
if !strings.HasPrefix(line, "RECIPE=") && !strings.HasPrefix(line, "TYPE=") {
|
||||||
|
lines = append(lines, line)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.HasPrefix(line, "#") {
|
||||||
|
lines = append(lines, line)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.Contains(line, version) && !a.Recipe.Dirty && !strings.HasSuffix(line, config.DIRTY_DEFAULT) {
|
||||||
|
skipped = true
|
||||||
|
lines = append(lines, line)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
splitted := strings.Split(line, ":")
|
||||||
|
|
||||||
|
line = fmt.Sprintf("%s:%s", splitted[0], version)
|
||||||
|
lines = append(lines, line)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if a.Recipe.Dirty && dirtyVersion != "" {
|
||||||
|
version = dirtyVersion
|
||||||
|
}
|
||||||
|
|
||||||
|
if !dryRun {
|
||||||
|
if err := os.WriteFile(a.Path, []byte(strings.Join(lines, "\n")), os.ModePerm); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.Debug(i18n.G("skipping writing version %s because dry run", version))
|
||||||
|
}
|
||||||
|
|
||||||
|
if !skipped {
|
||||||
|
log.Debug(i18n.G("version %s saved to %s.env", version, a.Domain))
|
||||||
|
} else {
|
||||||
|
log.Debug(i18n.G("skipping version %s write as already exists in %s.env", version, a.Domain))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,288 @@
|
|||||||
|
package app_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
"coopcloud.tech/abra/pkg/envfile"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
recipePkg "coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"coopcloud.tech/abra/pkg/test"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
"github.com/google/go-cmp/cmp"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
expectedAppEnv = envfile.AppEnv{
|
||||||
|
"DOMAIN": test.AppName,
|
||||||
|
"RECIPE": test.RecipeName,
|
||||||
|
}
|
||||||
|
|
||||||
|
expectedApp = appPkg.App{
|
||||||
|
Name: test.AppName,
|
||||||
|
Recipe: recipePkg.Get(expectedAppEnv["RECIPE"]),
|
||||||
|
Domain: expectedAppEnv["DOMAIN"],
|
||||||
|
Env: expectedAppEnv,
|
||||||
|
Path: expectedAppFile.Path,
|
||||||
|
Server: expectedAppFile.Server,
|
||||||
|
}
|
||||||
|
|
||||||
|
expectedAppFile = appPkg.AppFile{
|
||||||
|
Path: test.AppEnvPath,
|
||||||
|
Server: test.ServerName,
|
||||||
|
}
|
||||||
|
|
||||||
|
expectedAppFiles = map[string]appPkg.AppFile{
|
||||||
|
test.AppName: expectedAppFile,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNewApp(t *testing.T) {
|
||||||
|
app, err := appPkg.NewApp(expectedAppEnv, test.AppName, expectedAppFile)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !reflect.DeepEqual(app, expectedApp) {
|
||||||
|
t.Fatalf("did not get expected app type. Expected: %s; Got: %s", app, expectedApp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadAppEnvFile(t *testing.T) {
|
||||||
|
test.Setup()
|
||||||
|
t.Cleanup(func() { test.Teardown() })
|
||||||
|
|
||||||
|
app, err := appPkg.ReadAppEnvFile(expectedAppFile, test.AppName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !reflect.DeepEqual(app, expectedApp) {
|
||||||
|
t.Fatalf("did not get expected app type. Expected: %s; Got: %s", app, expectedApp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetApp(t *testing.T) {
|
||||||
|
test.Setup()
|
||||||
|
t.Cleanup(func() { test.Teardown() })
|
||||||
|
|
||||||
|
app, err := appPkg.GetApp(expectedAppFiles, test.AppName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !reflect.DeepEqual(app, expectedApp) {
|
||||||
|
t.Fatalf("did not get expected app type. Expected: %s; Got: %s", app, expectedApp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetComposeFiles(t *testing.T) {
|
||||||
|
test.Setup()
|
||||||
|
t.Cleanup(func() { test.Teardown() })
|
||||||
|
|
||||||
|
r := recipe.Get(test.AbraTestRecipe)
|
||||||
|
if err := r.EnsureExists(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
appEnv map[string]string
|
||||||
|
composeFiles []string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
map[string]string{},
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("%s/compose.yml", r.Dir),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
map[string]string{"COMPOSE_FILE": "compose.yml"},
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("%s/compose.yml", r.Dir),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
map[string]string{"COMPOSE_FILE": "compose.extra_secret.yml"},
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("%s/compose.extra_secret.yml", r.Dir),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
map[string]string{"COMPOSE_FILE": "compose.yml:compose.extra_secret.yml"},
|
||||||
|
[]string{
|
||||||
|
fmt.Sprintf("%s/compose.yml", r.Dir),
|
||||||
|
fmt.Sprintf("%s/compose.extra_secret.yml", r.Dir),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, test := range tests {
|
||||||
|
composeFiles, err := r.GetComposeFiles(test.appEnv)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
assert.Equal(t, composeFiles, test.composeFiles)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetComposeFilesError(t *testing.T) {
|
||||||
|
test.Setup()
|
||||||
|
t.Cleanup(func() { test.Teardown() })
|
||||||
|
|
||||||
|
r := recipe.Get(test.AbraTestRecipe)
|
||||||
|
err := r.EnsureExists()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct{ appEnv map[string]string }{
|
||||||
|
{map[string]string{"COMPOSE_FILE": "compose.yml::compose.foo.yml"}},
|
||||||
|
{map[string]string{"COMPOSE_FILE": "doesnt.exist.yml"}},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, test := range tests {
|
||||||
|
_, err := r.GetComposeFiles(test.appEnv)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatalf("should have failed: %v", test.appEnv)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFilters(t *testing.T) {
|
||||||
|
oldDir := config.RECIPES_DIR
|
||||||
|
config.RECIPES_DIR = "./testdata"
|
||||||
|
defer func() {
|
||||||
|
config.RECIPES_DIR = oldDir
|
||||||
|
}()
|
||||||
|
|
||||||
|
app, err := appPkg.NewApp(envfile.AppEnv{
|
||||||
|
"DOMAIN": "test.example.com",
|
||||||
|
"RECIPE": "test-recipe",
|
||||||
|
}, "test_example_com", appPkg.AppFile{
|
||||||
|
Path: "./testdata/filtertest.end",
|
||||||
|
Server: "local",
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := app.Filters(false, false)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
compareFilter(t, f, map[string]map[string]bool{
|
||||||
|
"name": {
|
||||||
|
"test_example_com": true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
f2, err := app.Filters(false, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
compareFilter(t, f2, map[string]map[string]bool{
|
||||||
|
"name": {
|
||||||
|
"^test_example_com": true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
f3, err := app.Filters(true, false)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
compareFilter(t, f3, map[string]map[string]bool{
|
||||||
|
"name": {
|
||||||
|
"test_example_com_bar": true,
|
||||||
|
"test_example_com_foo": true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
f4, err := app.Filters(true, true)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
compareFilter(t, f4, map[string]map[string]bool{
|
||||||
|
"name": {
|
||||||
|
"^test_example_com_bar": true,
|
||||||
|
"^test_example_com_foo": true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
f5, err := app.Filters(false, false, "foo")
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
compareFilter(t, f5, map[string]map[string]bool{
|
||||||
|
"name": {
|
||||||
|
"test_example_com_foo": true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func compareFilter(t *testing.T, f1 filters.Args, f2 map[string]map[string]bool) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
j1, err := f1.MarshalJSON()
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
j2, err := json.Marshal(f2)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if diff := cmp.Diff(string(j2), string(j1)); diff != "" {
|
||||||
|
t.Errorf("filters mismatch (-want +got):\n%s", diff)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestWriteRecipeVersionOverwrite(t *testing.T) {
|
||||||
|
test.Setup()
|
||||||
|
t.Cleanup(func() { test.Teardown() })
|
||||||
|
|
||||||
|
app, err := appPkg.GetApp(expectedAppFiles, test.AppName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Cleanup(func() {
|
||||||
|
if err := app.WipeRecipeVersion(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
assert.Equal(t, "", app.Recipe.EnvVersion)
|
||||||
|
|
||||||
|
if err := app.WriteRecipeVersion("foo", false); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
app, err = appPkg.GetApp(expectedAppFiles, test.AppName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.Equal(t, "foo", app.Recipe.EnvVersion)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestWriteRecipeVersionUnknown(t *testing.T) {
|
||||||
|
test.Setup()
|
||||||
|
t.Cleanup(func() { test.Teardown() })
|
||||||
|
|
||||||
|
app, err := appPkg.GetApp(expectedAppFiles, test.AppName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := app.WriteRecipeVersion(config.UNKNOWN_DEFAULT, false); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.NotEqual(t, config.UNKNOWN_DEFAULT, app.Recipe.EnvVersion)
|
||||||
|
}
|
||||||
@@ -0,0 +1,90 @@
|
|||||||
|
package app
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
composetypes "github.com/docker/cli/cli/compose/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SetRecipeLabel adds the label 'coop-cloud.${STACK_NAME}.recipe=${RECIPE}' to the app container
|
||||||
|
// to signal which recipe is connected to the deployed app
|
||||||
|
func SetRecipeLabel(compose *composetypes.Config, stackName string, recipe string) {
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
log.Debug(i18n.G("set recipe label 'coop-cloud.%s.recipe' to %s for %s", stackName, recipe, stackName))
|
||||||
|
labelKey := fmt.Sprintf("coop-cloud.%s.recipe", stackName)
|
||||||
|
service.Deploy.Labels[labelKey] = recipe
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetChaosLabel adds the label 'coop-cloud.${STACK_NAME}.chaos=true/false' to the app container
|
||||||
|
// to signal if the app is deployed in chaos mode
|
||||||
|
func SetChaosLabel(compose *composetypes.Config, stackName string, chaos bool) {
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
log.Debug(i18n.G("set label 'coop-cloud.%s.chaos' to %v for %s", stackName, chaos, stackName))
|
||||||
|
labelKey := fmt.Sprintf("coop-cloud.%s.chaos", stackName)
|
||||||
|
service.Deploy.Labels[labelKey] = strconv.FormatBool(chaos)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetChaosVersionLabel adds the label 'coop-cloud.${STACK_NAME}.chaos-version=$(GIT_COMMIT)' to the app container
|
||||||
|
func SetChaosVersionLabel(compose *composetypes.Config, stackName string, chaosVersion string) {
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
log.Debug(i18n.G("set label 'coop-cloud.%s.chaos-version' to %v for %s", stackName, chaosVersion, stackName))
|
||||||
|
labelKey := fmt.Sprintf("coop-cloud.%s.chaos-version", stackName)
|
||||||
|
service.Deploy.Labels[labelKey] = chaosVersion
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func SetVersionLabel(compose *composetypes.Config, stackName string, version string) {
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
log.Debug(i18n.G("set label 'coop-cloud.%s.version' to %v for %s", stackName, version, stackName))
|
||||||
|
labelKey := fmt.Sprintf("coop-cloud.%s.version", stackName)
|
||||||
|
service.Deploy.Labels[labelKey] = version
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetLabel reads docker labels in the format of "coop-cloud.${STACK_NAME}.${LABEL}" from the local compose files
|
||||||
|
func GetLabel(compose *composetypes.Config, stackName string, label string) string {
|
||||||
|
for _, service := range compose.Services {
|
||||||
|
if service.Name == "app" {
|
||||||
|
labelKey := fmt.Sprintf("coop-cloud.%s.%s", stackName, label)
|
||||||
|
log.Debug(i18n.G("get label '%s'", labelKey))
|
||||||
|
if labelValue, ok := service.Deploy.Labels[labelKey]; ok {
|
||||||
|
return labelValue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.Debug(i18n.G("no %s label found for %s", label, stackName))
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetTimeoutFromLabel reads the timeout value from docker label
|
||||||
|
// `coop-cloud.${STACK_NAME}.timeout=...` if present. A value is present if the
|
||||||
|
// operator uses a `TIMEOUT=...` in their app env.
|
||||||
|
func GetTimeoutFromLabel(compose *composetypes.Config, stackName string) (int, error) {
|
||||||
|
var timeout int
|
||||||
|
|
||||||
|
if timeoutLabel := GetLabel(compose, stackName, "timeout"); timeoutLabel != "" {
|
||||||
|
log.Debug(i18n.G("timeout label: %s", timeoutLabel))
|
||||||
|
|
||||||
|
var err error
|
||||||
|
timeout, err = strconv.Atoi(timeoutLabel)
|
||||||
|
if err != nil {
|
||||||
|
return timeout, errors.New(i18n.G("unable to convert timeout label %s to int: %s", timeoutLabel, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return timeout, nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,63 @@
|
|||||||
|
package app_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/test"
|
||||||
|
testPkg "coopcloud.tech/abra/pkg/test"
|
||||||
|
stack "coopcloud.tech/abra/pkg/upstream/stack"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGetTimeoutFromLabel(t *testing.T) {
|
||||||
|
test.Setup()
|
||||||
|
t.Cleanup(func() { test.Teardown() })
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
configuredTimeout string
|
||||||
|
expectedTimeout int
|
||||||
|
}{
|
||||||
|
{"0", 0},
|
||||||
|
{"DOESNTEXIST", 0}, // NOTE(d1): test when missing from .env
|
||||||
|
{"80", 80},
|
||||||
|
{"120", 120},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, test := range tests {
|
||||||
|
app, err := appPkg.GetApp(expectedAppFiles, testPkg.AppName)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if test.configuredTimeout != "DOESNTEXIST" {
|
||||||
|
app.Env["TIMEOUT"] = test.configuredTimeout
|
||||||
|
}
|
||||||
|
|
||||||
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deployOpts := stack.Deploy{
|
||||||
|
Composefiles: composeFiles,
|
||||||
|
Namespace: app.StackName(),
|
||||||
|
Prune: false,
|
||||||
|
ResolveImage: stack.ResolveImageAlways,
|
||||||
|
Detach: false,
|
||||||
|
}
|
||||||
|
|
||||||
|
compose, err := appPkg.GetAppComposeConfig(app.Name, deployOpts, app.Env)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
timeout, err := appPkg.GetTimeoutFromLabel(compose, app.StackName())
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.Equal(t, timeout, test.expectedTimeout)
|
||||||
|
}
|
||||||
|
}
|
||||||
Vendored
+2
@@ -0,0 +1,2 @@
|
|||||||
|
RECIPE=test-recipe
|
||||||
|
DOMAIN=test.example.com
|
||||||
+6
@@ -0,0 +1,6 @@
|
|||||||
|
version: "3.8"
|
||||||
|
services:
|
||||||
|
foo:
|
||||||
|
image: debian
|
||||||
|
bar:
|
||||||
|
image: debian
|
||||||
@@ -0,0 +1,135 @@
|
|||||||
|
package autocomplete
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/app"
|
||||||
|
appPkg "coopcloud.tech/abra/pkg/app"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/recipe"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// AppNameComplete copletes app names.
|
||||||
|
func AppNameComplete() ([]string, cobra.ShellCompDirective) {
|
||||||
|
appFiles, err := app.LoadAppFiles("")
|
||||||
|
if err != nil {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
var appNames []string
|
||||||
|
for appName := range appFiles {
|
||||||
|
appNames = append(appNames, appName)
|
||||||
|
}
|
||||||
|
|
||||||
|
return appNames, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
|
||||||
|
func ServiceNameComplete(appName string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
serviceNames, err := app.GetAppServiceNames(appName)
|
||||||
|
if err != nil {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
return serviceNames, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
|
||||||
|
// RecipeNameComplete completes recipe names.
|
||||||
|
func RecipeNameComplete() ([]string, cobra.ShellCompDirective) {
|
||||||
|
catl, err := recipe.ReadRecipeCatalogue(true)
|
||||||
|
if err != nil {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
localRecipes, err := recipe.GetRecipesLocal()
|
||||||
|
if err != nil && !strings.Contains(err.Error(), "empty") {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
var recipeNames []string
|
||||||
|
for name := range catl {
|
||||||
|
recipeNames = append(recipeNames, name)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, recipeLocal := range localRecipes {
|
||||||
|
recipeNames = append(recipeNames, recipeLocal)
|
||||||
|
}
|
||||||
|
|
||||||
|
return recipeNames, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
|
||||||
|
// RecipeVersionComplete completes versions for the recipe.
|
||||||
|
func RecipeVersionComplete(recipeName string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
catl, err := recipe.ReadRecipeCatalogue(true)
|
||||||
|
if err != nil {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
var recipeVersions []string
|
||||||
|
for _, v := range catl[recipeName].Versions {
|
||||||
|
for v2 := range v {
|
||||||
|
recipeVersions = append(recipeVersions, v2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return recipeVersions, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
|
||||||
|
// ServerNameComplete completes server names.
|
||||||
|
func ServerNameComplete() ([]string, cobra.ShellCompDirective) {
|
||||||
|
files, err := app.LoadAppFiles("")
|
||||||
|
if err != nil {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
var serverNames []string
|
||||||
|
for _, appFile := range files {
|
||||||
|
serverNames = append(serverNames, appFile.Server)
|
||||||
|
}
|
||||||
|
|
||||||
|
return serverNames, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
|
||||||
|
// CommandNameComplete completes recipe commands.
|
||||||
|
func CommandNameComplete(appName string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
app, err := app.Get(appName)
|
||||||
|
if err != nil {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
cmdNames, err := appPkg.ReadAbraShCmdNames(app.Recipe.AbraShPath)
|
||||||
|
if err != nil {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Strings(cmdNames)
|
||||||
|
|
||||||
|
return cmdNames, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
|
|
||||||
|
// SecretsComplete completes recipe secrets.
|
||||||
|
func SecretComplete(recipeName string) ([]string, cobra.ShellCompDirective) {
|
||||||
|
r := recipe.Get(recipeName)
|
||||||
|
|
||||||
|
config, err := r.GetComposeConfig(nil)
|
||||||
|
if err != nil {
|
||||||
|
err := i18n.G("autocomplete failed: %s", err)
|
||||||
|
return []string{err}, cobra.ShellCompDirectiveError
|
||||||
|
}
|
||||||
|
|
||||||
|
var secretNames []string
|
||||||
|
for name := range config.Secrets {
|
||||||
|
secretNames = append(secretNames, name)
|
||||||
|
}
|
||||||
|
|
||||||
|
return secretNames, cobra.ShellCompDirectiveDefault
|
||||||
|
}
|
||||||
@@ -0,0 +1,87 @@
|
|||||||
|
package catalogue
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
gitPkg "coopcloud.tech/abra/pkg/git"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/go-git/go-git/v5"
|
||||||
|
)
|
||||||
|
|
||||||
|
// EnsureCatalogue ensures that the catalogue is cloned locally & present.
|
||||||
|
func EnsureCatalogue() error {
|
||||||
|
catalogueDir := path.Join(config.ABRA_DIR, "catalogue")
|
||||||
|
if _, err := os.Stat(catalogueDir); err != nil && os.IsNotExist(err) {
|
||||||
|
log.Debug(i18n.G("catalogue is missing, retrieving now"))
|
||||||
|
|
||||||
|
url := fmt.Sprintf("%s/%s.git", config.REPOS_BASE_URL, config.CATALOGUE_JSON_REPO_NAME)
|
||||||
|
if err := gitPkg.Clone(catalogueDir, url); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EnsureIsClean makes sure that the catalogue has no unstaged changes.
|
||||||
|
func EnsureIsClean() error {
|
||||||
|
isClean, err := gitPkg.IsClean(config.CATALOGUE_DIR)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isClean {
|
||||||
|
return errors.New(i18n.G("%s has locally unstaged changes? please commit/remove your changes before proceeding", config.CATALOGUE_DIR))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EnsureUpToDate ensures that the local catalogue is up to date.
|
||||||
|
func EnsureUpToDate() error {
|
||||||
|
repo, err := git.PlainOpen(config.CATALOGUE_DIR)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
remotes, err := repo.Remotes()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(remotes) == 0 {
|
||||||
|
log.Debug(i18n.G("cannot ensure %s is up-to-date, no git remotes configured", config.CATALOGUE_DIR))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
worktree, err := repo.Worktree()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
branch, err := gitPkg.CheckoutDefaultBranch(repo, config.CATALOGUE_DIR)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := &git.PullOptions{
|
||||||
|
Force: true,
|
||||||
|
ReferenceName: branch,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := worktree.Pull(opts); err != nil {
|
||||||
|
if !strings.Contains(err.Error(), "already up-to-date") {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("fetched latest git changes for %s", config.CATALOGUE_DIR))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,125 @@
|
|||||||
|
// Package client provides Docker client initiatialisation functions.
|
||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/config"
|
||||||
|
contextPkg "coopcloud.tech/abra/pkg/context"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
sshPkg "coopcloud.tech/abra/pkg/ssh"
|
||||||
|
commandconnPkg "coopcloud.tech/abra/pkg/upstream/commandconn"
|
||||||
|
"github.com/docker/docker/client"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Conf is a Docker client configuration.
|
||||||
|
type Conf struct {
|
||||||
|
Timeout int
|
||||||
|
}
|
||||||
|
|
||||||
|
// Opt is a Docker client option.
|
||||||
|
type Opt func(c *Conf)
|
||||||
|
|
||||||
|
// WithTimeout specifies a timeout for a Docker client.
|
||||||
|
func WithTimeout(timeout int) Opt {
|
||||||
|
return func(c *Conf) {
|
||||||
|
c.Timeout = timeout
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// New initiates a new Docker client. New client connections are validated so
|
||||||
|
// that we ensure connections via SSH to the daemon can succeed. It takes into
|
||||||
|
// account that you may only want the local client and not communicate via SSH.
|
||||||
|
// For this use-case, please pass "default" as the serverName.
|
||||||
|
func New(serverName string, opts ...Opt) (*client.Client, error) {
|
||||||
|
var clientOpts []client.Opt
|
||||||
|
|
||||||
|
ctx, err := GetContext(serverName)
|
||||||
|
if err != nil {
|
||||||
|
serverDir := path.Join(config.SERVERS_DIR, serverName)
|
||||||
|
if _, err := os.Stat(serverDir); err != nil {
|
||||||
|
return nil, errors.New(i18n.G("server missing, run \"abra server add %s\"?", serverName))
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE(p4u1): when the docker context does not exist but the server folder
|
||||||
|
// is there, let's create a new docker context.
|
||||||
|
if err = CreateContext(serverName); err != nil {
|
||||||
|
return nil, errors.New(i18n.G("server missing context, context creation failed: %s", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx, err = GetContext(serverName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.New(i18n.G("server missing context, run \"abra server add %s\"?", serverName))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ctxEndpoint, err := contextPkg.GetContextEndpoint(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var isUnix bool
|
||||||
|
if strings.Contains(ctxEndpoint, "unix://") {
|
||||||
|
isUnix = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if serverName != "default" && !isUnix {
|
||||||
|
conf := &Conf{}
|
||||||
|
|
||||||
|
for _, opt := range opts {
|
||||||
|
opt(conf)
|
||||||
|
}
|
||||||
|
|
||||||
|
helper, err := commandconnPkg.NewConnectionHelper(ctxEndpoint, conf.Timeout)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
httpClient := &http.Client{
|
||||||
|
Transport: &http.Transport{
|
||||||
|
DialContext: helper.Dialer,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
clientOpts = append(clientOpts,
|
||||||
|
client.WithHTTPClient(httpClient),
|
||||||
|
client.WithHost(helper.Host),
|
||||||
|
client.WithDialContext(helper.Dialer),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
version := os.Getenv("DOCKER_API_VERSION")
|
||||||
|
if version != "" {
|
||||||
|
clientOpts = append(clientOpts, client.WithVersion(version))
|
||||||
|
} else {
|
||||||
|
clientOpts = append(clientOpts, client.WithAPIVersionNegotiation())
|
||||||
|
}
|
||||||
|
|
||||||
|
cl, err := client.NewClientWithOpts(clientOpts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("created client for %s", serverName))
|
||||||
|
|
||||||
|
info, err := cl.Info(context.Background())
|
||||||
|
if err != nil {
|
||||||
|
return cl, sshPkg.Fatal(serverName, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if info.Swarm.LocalNodeState == "inactive" {
|
||||||
|
if serverName != "default" && !isUnix {
|
||||||
|
return cl, errors.New(i18n.G("swarm mode not enabled on %s?", serverName))
|
||||||
|
}
|
||||||
|
|
||||||
|
return cl, errors.New(i18n.G("swarm mode not enabled on local server?"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return cl, nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
"github.com/docker/docker/api/types/swarm"
|
||||||
|
"github.com/docker/docker/client"
|
||||||
|
)
|
||||||
|
|
||||||
|
func GetConfigs(cl *client.Client, ctx context.Context, server string, fs filters.Args) ([]swarm.Config, error) {
|
||||||
|
configList, err := cl.ConfigList(ctx, swarm.ConfigListOptions{Filters: fs})
|
||||||
|
if err != nil {
|
||||||
|
return configList, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return configList, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetConfigNames(configs []swarm.Config) []string {
|
||||||
|
var confNames []string
|
||||||
|
|
||||||
|
for _, conf := range configs {
|
||||||
|
confNames = append(confNames, conf.Spec.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
return confNames
|
||||||
|
}
|
||||||
|
|
||||||
|
func RemoveConfigs(cl *client.Client, ctx context.Context, configNames []string, force bool) error {
|
||||||
|
for _, confName := range configNames {
|
||||||
|
if err := cl.ConfigRemove(context.Background(), confName); err != nil {
|
||||||
|
return errors.New(i18n.G("conf %s: %s", confName, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,89 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/context"
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
commandconnPkg "coopcloud.tech/abra/pkg/upstream/commandconn"
|
||||||
|
dConfig "github.com/docker/cli/cli/config"
|
||||||
|
"github.com/docker/cli/cli/context/docker"
|
||||||
|
contextStore "github.com/docker/cli/cli/context/store"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Context = contextStore.Metadata
|
||||||
|
|
||||||
|
// CreateContext creates a new Docker context.
|
||||||
|
func CreateContext(contextName string) error {
|
||||||
|
host := fmt.Sprintf("ssh://%s", contextName)
|
||||||
|
|
||||||
|
if err := createContext(contextName, host); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debug(i18n.G("created the %s context", contextName))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// createContext interacts with Docker Context to create a Docker context config
|
||||||
|
func createContext(name string, host string) error {
|
||||||
|
s := context.NewDefaultDockerContextStore()
|
||||||
|
contextMetadata := contextStore.Metadata{
|
||||||
|
Endpoints: make(map[string]interface{}),
|
||||||
|
Name: name,
|
||||||
|
}
|
||||||
|
|
||||||
|
contextTLSData := contextStore.ContextTLSData{
|
||||||
|
Endpoints: make(map[string]contextStore.EndpointTLSData),
|
||||||
|
}
|
||||||
|
|
||||||
|
dockerEP, dockerTLS, err := commandconnPkg.GetDockerEndpointMetadataAndTLS(host)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
contextMetadata.Endpoints[docker.DockerEndpoint] = dockerEP
|
||||||
|
if dockerTLS != nil {
|
||||||
|
contextTLSData.Endpoints[docker.DockerEndpoint] = *dockerTLS
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := s.CreateOrUpdate(contextMetadata); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := s.ResetTLSMaterial(name, &contextTLSData); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func DeleteContext(name string) error {
|
||||||
|
if name == "default" {
|
||||||
|
return errors.New(i18n.G("context 'default' cannot be removed"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := GetContext(name); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := dConfig.LoadDefaultConfigFile(nil)
|
||||||
|
cfg.CurrentContext = ""
|
||||||
|
if err := cfg.Save(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return context.NewDefaultDockerContextStore().Remove(name)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetContext(contextName string) (contextStore.Metadata, error) {
|
||||||
|
ctx, err := context.NewDefaultDockerContextStore().GetMetadata(contextName)
|
||||||
|
if err != nil {
|
||||||
|
return contextStore.Metadata{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return ctx, nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"github.com/containers/image/docker"
|
||||||
|
"github.com/containers/image/types"
|
||||||
|
"github.com/distribution/reference"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GetRegistryTags retrieves all tags of an image from a container registry.
|
||||||
|
func GetRegistryTags(img reference.Named) ([]string, error) {
|
||||||
|
var tags []string
|
||||||
|
|
||||||
|
ref, err := docker.ParseReference(fmt.Sprintf("//%s", img))
|
||||||
|
if err != nil {
|
||||||
|
return tags, errors.New(i18n.G("failed to parse image %s, saw: %s", img, err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
tags, err = docker.GetRepositoryTags(ctx, &types.SystemContext{}, ref)
|
||||||
|
if err != nil {
|
||||||
|
return tags, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return tags, nil
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"github.com/docker/docker/api/types/swarm"
|
||||||
|
"github.com/docker/docker/client"
|
||||||
|
)
|
||||||
|
|
||||||
|
func StoreSecret(cl *client.Client, secretName, secretValue string) error {
|
||||||
|
ann := swarm.Annotations{Name: secretName}
|
||||||
|
spec := swarm.SecretSpec{Annotations: ann, Data: []byte(secretValue)}
|
||||||
|
|
||||||
|
if _, err := cl.SecretCreate(context.Background(), spec); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetSecretNames(secrets []swarm.Secret) []string {
|
||||||
|
var secretNames []string
|
||||||
|
for _, secret := range secrets {
|
||||||
|
secretNames = append(secretNames, secret.Spec.Name)
|
||||||
|
}
|
||||||
|
return secretNames
|
||||||
|
}
|
||||||
@@ -0,0 +1,58 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/docker/docker/api/types/swarm"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGetSecretNames(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
secrets []swarm.Secret
|
||||||
|
expected []string
|
||||||
|
description string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "empty secrets list",
|
||||||
|
secrets: []swarm.Secret{},
|
||||||
|
expected: nil,
|
||||||
|
description: "should return nil for empty input",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "single secret",
|
||||||
|
secrets: []swarm.Secret{
|
||||||
|
{Spec: swarm.SecretSpec{Annotations: swarm.Annotations{Name: "database_password"}}},
|
||||||
|
},
|
||||||
|
expected: []string{"database_password"},
|
||||||
|
description: "should return single secret name",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "multiple secrets",
|
||||||
|
secrets: []swarm.Secret{
|
||||||
|
{Spec: swarm.SecretSpec{Annotations: swarm.Annotations{Name: "db_password"}}},
|
||||||
|
{Spec: swarm.SecretSpec{Annotations: swarm.Annotations{Name: "api_key"}}},
|
||||||
|
{Spec: swarm.SecretSpec{Annotations: swarm.Annotations{Name: "ssl_cert"}}},
|
||||||
|
},
|
||||||
|
expected: []string{"db_password", "api_key", "ssl_cert"},
|
||||||
|
description: "should return all secret names in order",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "secrets with empty names",
|
||||||
|
secrets: []swarm.Secret{
|
||||||
|
{Spec: swarm.SecretSpec{Annotations: swarm.Annotations{Name: ""}}},
|
||||||
|
{Spec: swarm.SecretSpec{Annotations: swarm.Annotations{Name: "valid_name"}}},
|
||||||
|
},
|
||||||
|
expected: []string{"", "valid_name"},
|
||||||
|
description: "should include empty names if present",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
result := GetSecretNames(tt.secrets)
|
||||||
|
assert.Equal(t, tt.expected, result, tt.description)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,63 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"github.com/docker/docker/api/types/filters"
|
||||||
|
"github.com/docker/docker/api/types/volume"
|
||||||
|
"github.com/docker/docker/client"
|
||||||
|
)
|
||||||
|
|
||||||
|
func GetVolumes(cl *client.Client, ctx context.Context, server string, fs filters.Args) ([]*volume.Volume, error) {
|
||||||
|
volumeListOKBody, err := cl.VolumeList(ctx, volume.ListOptions{Filters: fs})
|
||||||
|
volumeList := volumeListOKBody.Volumes
|
||||||
|
if err != nil {
|
||||||
|
return volumeList, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return volumeList, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetVolumeNames(volumes []*volume.Volume) []string {
|
||||||
|
var volumeNames []string
|
||||||
|
|
||||||
|
for _, vol := range volumes {
|
||||||
|
volumeNames = append(volumeNames, vol.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
return volumeNames
|
||||||
|
}
|
||||||
|
|
||||||
|
func RemoveVolumes(cl *client.Client, ctx context.Context, volumeNames []string, force bool, retries int) error {
|
||||||
|
for _, volName := range volumeNames {
|
||||||
|
err := retryFunc(5, func() error {
|
||||||
|
return cl.VolumeRemove(context.Background(), volName, force)
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return errors.New(i18n.G("volume %s: %s", volName, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// retryFunc retries the given function for the given retries. After the nth
|
||||||
|
// retry it waits (n + 1)^2 seconds before the next retry (starting with n=0).
|
||||||
|
// It returns an error if the function still failed after the last retry.
|
||||||
|
func retryFunc(retries int, fn func() error) error {
|
||||||
|
for i := 0; i < retries; i++ {
|
||||||
|
err := fn()
|
||||||
|
if err == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if i+1 < retries {
|
||||||
|
sleep := time.Duration(i+1) * time.Duration(i+1)
|
||||||
|
log.Info(i18n.G("%s: waiting %d seconds before next retry", err, sleep))
|
||||||
|
time.Sleep(sleep * time.Second)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return errors.New(i18n.G("%d retries failed", retries))
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRetryFunc(t *testing.T) {
|
||||||
|
err := retryFunc(1, func() error { return nil })
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("should not return an error: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
i := 0
|
||||||
|
fn := func() error {
|
||||||
|
i++
|
||||||
|
return fmt.Errorf("oh no, something went wrong!")
|
||||||
|
}
|
||||||
|
err = retryFunc(2, fn)
|
||||||
|
if err == nil {
|
||||||
|
t.Error("should return an error")
|
||||||
|
}
|
||||||
|
if i != 2 {
|
||||||
|
t.Errorf("The function should have been called 1 times, got %d", i)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,122 @@
|
|||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"coopcloud.tech/abra/pkg/i18n"
|
||||||
|
"coopcloud.tech/abra/pkg/log"
|
||||||
|
"gopkg.in/yaml.v3"
|
||||||
|
)
|
||||||
|
|
||||||
|
// LoadAbraConfig returns the abra configuration. It tries to find a abra
|
||||||
|
// configuration file (see findAbraConfig for lookup logic). When no
|
||||||
|
// configuration was found it returns the default config.
|
||||||
|
func LoadAbraConfig() Abra {
|
||||||
|
wd, _ := os.Getwd()
|
||||||
|
configFile := findAbraConfig(wd)
|
||||||
|
if configFile == "" {
|
||||||
|
log.Debug(i18n.G("no config file found"))
|
||||||
|
return Abra{}
|
||||||
|
}
|
||||||
|
data, err := os.ReadFile(configFile)
|
||||||
|
if err != nil {
|
||||||
|
// Do nothing, when an error occurs
|
||||||
|
log.Debug(i18n.G("error reading config file: %s", err))
|
||||||
|
return Abra{}
|
||||||
|
}
|
||||||
|
|
||||||
|
config := Abra{}
|
||||||
|
err = yaml.Unmarshal(data, &config)
|
||||||
|
if err != nil {
|
||||||
|
// Do nothing, when an error occurs
|
||||||
|
log.Debug(i18n.G("error loading config file: %s", err))
|
||||||
|
return Abra{}
|
||||||
|
}
|
||||||
|
log.Debug(i18n.G("config file loaded from: %s", configFile))
|
||||||
|
config.configPath = filepath.Dir(configFile)
|
||||||
|
return config
|
||||||
|
}
|
||||||
|
|
||||||
|
// findAbraConfig recursively looks for a abra.y(a)ml file in the given directory.
|
||||||
|
// When the file was not found it calls the function again with the parent
|
||||||
|
// directory until the home directory is hit. When no abra config was found it
|
||||||
|
// returns an empty string.
|
||||||
|
func findAbraConfig(dir string) string {
|
||||||
|
dir, err := filepath.Abs(dir)
|
||||||
|
if err != nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
if dir == os.ExpandEnv("$HOME") || dir == "/" {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
p := path.Join(dir, "abra.yaml")
|
||||||
|
if _, err := os.Stat(p); err == nil {
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
p = path.Join(dir, "abra.yml")
|
||||||
|
if _, err := os.Stat(p); err == nil {
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
return findAbraConfig(filepath.Dir(dir))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Abra defines the configuration file for abra.
|
||||||
|
type Abra struct {
|
||||||
|
configPath string
|
||||||
|
AbraDir string `yaml:"abraDir"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAbraDir returns the abra dir. It has the following logic:
|
||||||
|
// 1. check if $ABRA_DIR is set
|
||||||
|
// 2. check if abraDir was set in a config file
|
||||||
|
// 3. use $HOME/.abra when above two options failed
|
||||||
|
func (a Abra) GetAbraDir() string {
|
||||||
|
if dir, exists := os.LookupEnv("ABRA_DIR"); exists && dir != "" {
|
||||||
|
log.Debug(i18n.G("read abra dir from $ABRA_DIR"))
|
||||||
|
return dir
|
||||||
|
}
|
||||||
|
if a.AbraDir != "" {
|
||||||
|
log.Debug(i18n.G("read abra dir from config file"))
|
||||||
|
if path.IsAbs(a.AbraDir) {
|
||||||
|
return a.AbraDir
|
||||||
|
}
|
||||||
|
// Make the path absolute
|
||||||
|
return path.Join(a.configPath, a.AbraDir)
|
||||||
|
}
|
||||||
|
log.Debug(i18n.G("using default abra dir"))
|
||||||
|
return os.ExpandEnv("$HOME/.abra")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a Abra) GetServersDir() string { return path.Join(a.GetAbraDir(), "servers") }
|
||||||
|
func (a Abra) GetRecipesDir() string { return path.Join(a.GetAbraDir(), "recipes") }
|
||||||
|
func (a Abra) GetLogsDir() string { return path.Join(a.GetAbraDir(), "logs") }
|
||||||
|
func (a Abra) GetCatalogueDir() string { return path.Join(a.GetAbraDir(), "catalogue") }
|
||||||
|
|
||||||
|
var config = LoadAbraConfig()
|
||||||
|
|
||||||
|
var (
|
||||||
|
ABRA_DIR = config.GetAbraDir()
|
||||||
|
SERVERS_DIR = config.GetServersDir()
|
||||||
|
RECIPES_DIR = config.GetRecipesDir()
|
||||||
|
LOGS_DIR = config.GetLogsDir()
|
||||||
|
CATALOGUE_DIR = config.GetCatalogueDir()
|
||||||
|
RECIPES_JSON = path.Join(config.GetCatalogueDir(), "recipes.json")
|
||||||
|
REPOS_BASE_URL = "https://git.coopcloud.tech/coop-cloud"
|
||||||
|
CATALOGUE_JSON_REPO_NAME = "recipes-catalogue-json"
|
||||||
|
TOOLSHED_SSH_URL_TEMPLATE = "ssh://git@git.coopcloud.tech:2222/toolshed/%s.git"
|
||||||
|
RECIPES_SSH_URL_TEMPLATE = "ssh://git@git.coopcloud.tech:2222/coop-cloud/%s.git"
|
||||||
|
|
||||||
|
// NOTE(d1): please note, this was done purely out of laziness on our part
|
||||||
|
// AFAICR. it's easy to punt the value into the label because that is what is
|
||||||
|
// expects. it's not particularly useful in terms of UI/UX but hey, nobody
|
||||||
|
// complained yet!
|
||||||
|
CHAOS_DEFAULT = "false"
|
||||||
|
|
||||||
|
DIRTY_DEFAULT = "+U"
|
||||||
|
|
||||||
|
MISSING_DEFAULT = "-"
|
||||||
|
|
||||||
|
UNKNOWN_DEFAULT = "unknown"
|
||||||
|
)
|
||||||
@@ -0,0 +1,136 @@
|
|||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestFindAbraConfig(t *testing.T) {
|
||||||
|
wd, err := os.Getwd()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
Dir string
|
||||||
|
Config string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
Dir: "testdata/abraconfig1",
|
||||||
|
Config: filepath.Join(wd, "testdata/abraconfig1/abra.yaml"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Dir: "testdata/abraconfig1/subdir",
|
||||||
|
Config: filepath.Join(wd, "testdata/abraconfig1/abra.yaml"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Dir: "testdata/abraconfig2",
|
||||||
|
Config: filepath.Join(wd, "testdata/abraconfig2/abra.yml"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Dir: "testdata/abraconfig2/subdir",
|
||||||
|
Config: filepath.Join(wd, "testdata/abraconfig2/abra.yml"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Dir: "testdata",
|
||||||
|
Config: "",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
t.Run(tc.Dir, func(t *testing.T) {
|
||||||
|
config := findAbraConfig(tc.Dir)
|
||||||
|
if config != tc.Config {
|
||||||
|
t.Errorf("\nwant: %s\ngot: %s", tc.Config, config)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLoadAbraConfigGetAbraDir(t *testing.T) {
|
||||||
|
wd, err := os.Getwd()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Setenv("ABRA_DIR", "")
|
||||||
|
|
||||||
|
t.Run("default", func(t *testing.T) {
|
||||||
|
cfg := LoadAbraConfig()
|
||||||
|
wantAbraDir := os.ExpandEnv("$HOME/.abra")
|
||||||
|
if cfg.GetAbraDir() != wantAbraDir {
|
||||||
|
t.Errorf("\nwant: %s\ngot: %s", wantAbraDir, cfg.GetAbraDir())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("from config file", func(t *testing.T) {
|
||||||
|
t.Cleanup(func() { os.Chdir(wd) })
|
||||||
|
err = os.Chdir(filepath.Join(wd, "testdata/abraconfig1"))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := LoadAbraConfig()
|
||||||
|
wantAbraDir := filepath.Join(wd, "testdata/abraconfig1/foobar")
|
||||||
|
if cfg.GetAbraDir() != wantAbraDir {
|
||||||
|
t.Errorf("\nwant: %s\ngot: %s", wantAbraDir, cfg.GetAbraDir())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("default when config file is empty", func(t *testing.T) {
|
||||||
|
t.Cleanup(func() { os.Chdir(wd) })
|
||||||
|
err := os.Chdir(filepath.Join(wd, "testdata/abraconfig2"))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := LoadAbraConfig()
|
||||||
|
wantAbraDir := os.ExpandEnv("$HOME/.abra")
|
||||||
|
if cfg.GetAbraDir() != wantAbraDir {
|
||||||
|
t.Errorf("\nwant: %s\ngot: %s", wantAbraDir, cfg.GetAbraDir())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("from env variable", func(t *testing.T) {
|
||||||
|
t.Setenv("ABRA_DIR", "foo")
|
||||||
|
cfg := LoadAbraConfig()
|
||||||
|
wantAbraDir := "foo"
|
||||||
|
if cfg.GetAbraDir() != wantAbraDir {
|
||||||
|
t.Errorf("\nwant: %s\ngot: %s", wantAbraDir, cfg.GetAbraDir())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLoadAbraConfigServersDir(t *testing.T) {
|
||||||
|
wd, err := os.Getwd()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Setenv("ABRA_DIR", "")
|
||||||
|
|
||||||
|
t.Run("default", func(t *testing.T) {
|
||||||
|
cfg := LoadAbraConfig()
|
||||||
|
wantServersDir := os.ExpandEnv("$HOME/.abra/servers")
|
||||||
|
if cfg.GetServersDir() != wantServersDir {
|
||||||
|
t.Errorf("\nwant: %s\ngot: %s", wantServersDir, cfg.GetServersDir())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("from config file", func(t *testing.T) {
|
||||||
|
t.Cleanup(func() { os.Chdir(wd) })
|
||||||
|
err = os.Chdir(filepath.Join(wd, "testdata/abraconfig1"))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := LoadAbraConfig()
|
||||||
|
log.Println(cfg)
|
||||||
|
wantServersDir := filepath.Join(wd, "testdata/abraconfig1/foobar/servers")
|
||||||
|
if cfg.GetServersDir() != wantServersDir {
|
||||||
|
t.Errorf("\nwant: %s\ngot: %s", wantServersDir, cfg.GetServersDir())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user