Compare commits
671 Commits
v0.3.0
...
feature/no
| Author | SHA1 | Date | |
|---|---|---|---|
|
9866cd0404
|
|||
|
10d29b6fab
|
|||
|
6f8f60a9e2
|
|||
|
c1b4665706
|
|||
|
5447150d4d
|
|||
|
bf26703b2d
|
|||
|
21c6264ea9
|
|||
|
79ef9fa6d5
|
|||
|
04a9061663
|
|||
|
5283f6fce7
|
|||
|
a08a4746f7
|
|||
|
9e3652479b
|
|||
|
011386fb8d
|
|||
|
4d77f5d38c
|
|||
|
64de4deddd
|
|||
|
8f7994d82e
|
|||
|
a7d0e71ab6
|
|||
|
27d9f73c61
|
|||
|
ed3de8b16f
|
|||
|
d7b4c67953
|
|||
| 7489d4a32f | |||
|
ac77e5b7c1
|
|||
|
e544c28105
|
|||
|
4909dac5c2
|
|||
| 3cf4348695 | |||
|
af3da0a26c
|
|||
|
2d32320c7d
|
|||
|
fc2bec6246
|
|||
|
5addd25186
|
|||
|
215d178e69
|
|||
|
5474bf66e7
|
|||
|
ef2a37e2bf
|
|||
|
0e3180602c
|
|||
|
15e2f9b962
|
|||
|
4ae10c9b53
|
|||
| 45137e0cfe | |||
|
717fe93104
|
|||
|
fdac789ccb
|
|||
|
9355dab6b6
|
|||
|
f3676949d2
|
|||
|
79952b73c5
|
|||
| 17c419403e | |||
|
6d06312a5c
|
|||
|
acb399b0b7
|
|||
|
bf20b6467e
|
|||
|
b91d90d75c
|
|||
|
3284bbf6ca
|
|||
|
171b84ee81
|
|||
|
54b01dd282
|
|||
|
e08ea64f47
|
|||
|
8cc2c9554f
|
|||
| 32dff9c67f | |||
|
126b8b20e0
|
|||
|
5abf69f356
|
|||
|
210a69bd9b
|
|||
|
bbed3cd367
|
|||
|
7943da0f17
|
|||
| 620167eedf | |||
|
e077debfc2
|
|||
|
531b2c3002
|
|||
|
6d2bc729b8
|
|||
|
2630ec2af4
|
|||
| daed5c1eea | |||
| 2e9429bb32 | |||
|
37c15c7a62
|
|||
|
01ecea74ff
|
|||
|
f401a03590
|
|||
|
fff6dea100
|
|||
|
48ab96dda9
|
|||
|
7ac3130c18
|
|||
|
cbfa148051
|
|||
|
87d900b627
|
|||
|
926dc06294
|
|||
|
00b73b06d7
|
|||
|
0daac33915
|
|||
|
0e472bc311
|
|||
| 40b34d0935 | |||
|
61cb8f4941
|
|||
|
433ac4dc8e
|
|||
|
62fe0d8fac
|
|||
|
2a675fd135
|
|||
|
c2c3ebc2e1
|
|||
|
5a5c316c14
|
|||
| f0d5457ec1 | |||
|
5588e3b3e8
|
|||
|
8949d76d26
|
|||
| 8bc9bbdc33 | |||
| d6d09b57b8 | |||
|
1685d6ecf8
|
|||
|
5348a229a6
|
|||
|
bad3b7a2be
|
|||
|
b541e95bb7
|
|||
|
3f43fe8101
|
|||
|
231dfc8404
|
|||
|
eeb9b0a331
|
|||
|
08e783d185
|
|||
|
fa5dc8ca46
|
|||
|
bc34e9c5e0
|
|||
| f388bd0237 | |||
|
48041630ca
|
|||
|
2d1ff29eca
|
|||
| 46fa42e387 | |||
|
c6c5d80fb4
|
|||
|
c0f4e7925e
|
|||
|
49d24990b4
|
|||
|
619bd954b7
|
|||
|
e27c64b5f1
|
|||
|
b36baf26eb
|
|||
|
adedaa5f7b
|
|||
|
596ed7fccc
|
|||
|
5685e1b7bc
|
|||
|
c3b82fc2a9
|
|||
|
77e2fe5792
|
|||
|
bc43082839
|
|||
|
b09225543b
|
|||
|
f2507409a3
|
|||
|
46b4723999
|
|||
|
3f90a011c4
|
|||
|
3ba333e802
|
|||
| d9dff3e872 | |||
| 6ddeacb779 | |||
|
78aff3d796
|
|||
|
8f600f44bd
|
|||
|
819ecf6ad8
|
|||
|
945eaba5e1
|
|||
|
22d362e1a0
|
|||
|
d4e67a830c
|
|||
|
670b2da1ef
|
|||
|
ed5c5b3081
|
|||
| 4ee6bfddfa | |||
|
8b60890061
|
|||
|
0367450c4b
|
|||
|
e6f5623c7f
|
|||
| 367f566ccb | |||
|
80e69df75c
|
|||
|
02af69b055
|
|||
|
5d459e7e7d
|
|||
| 51a3cb60ec | |||
| 43c57c128f | |||
|
5a3adba603
|
|||
|
3715cb518b
|
|||
|
2c9ecc1fef
|
|||
|
095747e89b
|
|||
|
2130369604
|
|||
|
c996351930
|
|||
| 8b897168cc | |||
|
4217ba52e0
|
|||
|
de20931d30
|
|||
|
8de0a2e26e
|
|||
|
06521d1c34
|
|||
|
38b3d68fd5
|
|||
|
eac8fa6edb
|
|||
|
43f918a074
|
|||
| e322867d79 | |||
|
4d6fa318b7
|
|||
|
7f2df3b025
|
|||
|
da22a9d448
|
|||
|
e3b96d5cff
|
|||
| 4e8878a4b5 | |||
|
e65b890880
|
|||
|
f57edd4d3b
|
|||
|
1afd56fb80
|
|||
| 71669a4b96 | |||
|
c312e30c17
|
|||
| 51f4556ede | |||
|
c36cf5eee6
|
|||
|
54220019bb
|
|||
|
079ee8833c
|
|||
|
26d613bdca
|
|||
|
69b3afb8f7
|
|||
|
fee951c05c
|
|||
| 4fa4ae6b54 | |||
| 869ff4691b | |||
|
822a2dc018
|
|||
|
5b7fc3707b
|
|||
| 0e2dc54dc6 | |||
| 87f09c94d0 | |||
|
b33b8104a8
|
|||
| 4a4a222973 | |||
| 8c524abcf5 | |||
|
a852ab75ae
|
|||
|
de1f234c15
|
|||
| 4581900427 | |||
|
56d91083e5
|
|||
|
ba7c3795f8
|
|||
|
bbf3fb91a0
|
|||
| 1754df73cb | |||
|
9a1f9abf84
|
|||
|
2753388e1e
|
|||
|
f3159d30f1
|
|||
|
ca238be6f4
|
|||
|
8747ce4eb0
|
|||
|
fcda3b9c8c
|
|||
|
67689dcce3
|
|||
|
22ffcd54db
|
|||
|
bd1b177993
|
|||
|
3f110995a4
|
|||
|
a7410058fa
|
|||
|
411587456b
|
|||
|
84e915ece9
|
|||
|
70ac3b0a70
|
|||
|
a7cbd8ce36
|
|||
|
c9052b35f6
|
|||
|
3b96130491
|
|||
|
176b1a10c6
|
|||
|
1c54e4c0b5
|
|||
|
7796a22491
|
|||
|
7e6e917ae1
|
|||
|
28cfe4b1e7
|
|||
|
179a82d2dd
|
|||
|
420442c1c0
|
|||
|
68c5758ecc
|
|||
|
c5dd3c30a6
|
|||
|
422d5c7cd2
|
|||
|
5a23d523a8
|
|||
|
f8da034e66
|
|||
|
b0b56fcf92
|
|||
| 0cf000c1b8 | |||
| fa9a924b0a | |||
|
50f91cc7d7
|
|||
|
a628a03f84
|
|||
|
eaf41e0835
|
|||
|
243cf9c08d
|
|||
|
c32fc51aab
|
|||
|
aa9178d569
|
|||
|
281938dd64
|
|||
|
fafc5d8f6f
|
|||
|
1238359b5f
|
|||
| 84220beb1c | |||
|
1e9ec9bb76
|
|||
| 21e51a7c40 | |||
|
e3c30f7b16
|
|||
|
b4f0c60ea0
|
|||
|
1a5a2177b4
|
|||
|
7e8443c598
|
|||
|
7b71f2cf76
|
|||
|
c7b137e5eb
|
|||
|
958d18d61a
|
|||
|
3aa0c49507
|
|||
|
|
4e566a0607 | ||
|
|
aab6793b86
|
||
|
|
cfd0935bdc
|
||
|
|
c2dae105ff
|
||
|
|
2a70bf2fb9
|
||
|
|
9a9947f9ad
|
||
|
|
bdf5a18ad4
|
||
|
|
aa399b862a
|
||
|
|
713e91a720
|
||
|
|
8ec2a6d7e4
|
||
|
|
4ecf2c4246
|
||
|
|
4fdf8accd6
|
||
|
|
f451adcb53
|
||
|
|
721dccb499
|
||
|
|
27bb7d1bfe
|
||
|
|
1d44181fb5
|
||
|
|
de67f59d5c
|
||
|
|
1995e6dda2
|
||
|
|
600cfe0f78
|
||
|
|
e301ac8e2e
|
||
|
|
03a1d9f277
|
||
|
|
00049f3743
|
||
|
|
60c0a43f33
|
||
|
|
0c1b1b4afe
|
||
|
|
92310d434a
|
||
|
|
56c127ca0c
|
||
|
|
5075fef616
|
||
|
|
8e090daa9c
|
||
|
|
def87a1621
|
||
|
|
00ec7fa21c
|
||
|
|
2b8bfaaca8
|
||
|
|
3e9a08a266
|
||
|
|
fcea11f0e5
|
||
|
|
261a782963
|
||
|
|
e964e7e52c
|
||
|
|
e508407df4
|
||
|
|
bec827acb1
|
||
|
|
0a69603643
|
||
|
|
d4f71e98ed
|
||
|
|
e56c9bd0d5
|
||
|
|
e1b7e1b2ef
|
||
|
|
1056ffd08e
|
||
| be5fe00f20 | |||
|
|
e9c4929726
|
||
| 14ff0c0e16 | |||
|
|
d939f5d649
|
||
|
|
69fffb29d8
|
||
|
|
91d3b977e9
|
||
| 7a5fd46835 | |||
|
|
9c4c5c2553
|
||
|
|
8f819d12c0
|
||
|
|
b810e27480
|
||
|
|
1949f1876f
|
||
|
|
2ba0116ca6
|
||
|
|
2c2ddabdff
|
||
|
|
dfcdbec0dd
|
||
|
|
3b67a8791c
|
||
|
|
d5ab532947
|
||
|
|
50c63d5c38
|
||
|
|
64d09cfb7f
|
||
|
|
def44618ef
|
||
|
|
9e5aeaf572
|
||
|
|
86f85a90f4
|
||
| d8a35ac3fd | |||
|
|
5a5f62e98a
|
||
|
|
074f9afcbb
|
||
|
|
725fd2e5ea
|
||
|
|
8349ca5e12
|
||
|
|
46d59e3371
|
||
|
|
e8e6ee0bc4
|
||
|
|
a91ee2bd0a
|
||
|
|
fcb6923c92
|
||
|
|
0f3b9f176e
|
||
| 822ae2f945 | |||
|
|
96c669ab4e
|
||
|
|
558100c35e
|
||
|
|
6739b38f4c
|
||
| 7e1272c936 | |||
|
|
ecdeb4c122
|
||
|
|
8614e2f12b
|
||
|
|
a038a857d9
|
||
|
|
eee81d0cf1
|
||
|
|
b7fa4b012a
|
||
|
|
10bcd5c32b
|
||
|
|
f79d5d4724
|
||
|
|
866ffbe615
|
||
|
|
3c1fe3396d
|
||
|
|
e4242333d9
|
||
|
|
138f13c1a0
|
||
|
|
ad5e515200
|
||
|
|
1ea8b22a59
|
||
|
|
f49aff262c
|
||
| 852e2fea1e | |||
|
|
353b55fe1a
|
||
|
|
ba0cbba96b
|
||
|
|
5f921f1b53
|
||
|
|
a2d27bf575
|
||
|
|
fcf9a065e1
|
||
|
|
ec9bcacd46
|
||
|
|
645abac810 | ||
|
|
e11be727a1 | ||
|
|
12b24337e7 | ||
|
|
b0bfc290c4 | ||
|
|
4c6c81171b | ||
|
|
4d88a40109
|
||
|
|
d9b39b36fb
|
||
|
|
06aed8c33d
|
||
| 0a778e92d8 | |||
|
|
e5a5633e44
|
||
|
|
a68825493f
|
||
|
|
e1e83386a8
|
||
|
|
3adc1917f6
|
||
|
|
8a570ce724
|
||
|
|
c78df9e5f1
|
||
|
5c2df3df07
|
|||
| 83e3e2ecd8 | |||
| b32e2fcb7b | |||
| 96a4db5bae | |||
| c7925f132e | |||
| e4406bf6ff | |||
| ee7769c8c7 | |||
| fdf3218f88 | |||
| 652ed5f7e3 | |||
|
|
e4ed797920
|
||
|
|
93740f17ef
|
||
|
|
affb058671
|
||
|
|
6acc3f2f59
|
||
| 7987e92723 | |||
|
|
d922e7f869
|
||
|
716d4b944a
|
|||
|
42af148168
|
|||
|
|
89c67f3617
|
||
| 1b959b5643 | |||
|
|
4551a14362
|
||
|
|
bfc0969829
|
||
|
|
a1be338ba1
|
||
|
|
589e46bc63
|
||
|
|
34e4cec503
|
||
|
|
c48538a1c6
|
||
|
|
2cced696f5
|
||
|
|
beaafa5d7e
|
||
|
|
9cf309aaa8
|
||
|
|
e8bbe6c713
|
||
|
|
49de4007ab
|
||
|
|
bc4d9ff528
|
||
|
|
b03c6e9513
|
||
|
|
332ad757a5
|
||
|
|
07fe8dba71
|
||
|
|
aedaabc7ba
|
||
|
|
8eb5f093a4
|
||
| de45d070aa | |||
| c0b1112e49 | |||
|
|
2f90393eb6
|
||
|
|
8b87072485
|
||
|
|
82019f47be
|
||
|
|
259e72167b
|
||
|
|
7000908891
|
||
|
|
df0c13b400
|
||
|
|
387a2fa2e6
|
||
| 68eba80fd7 | |||
|
|
7e05530ab7
|
||
|
|
745a319b3d
|
||
|
|
f829bb3379
|
||
|
|
19bafe081f
|
||
| d130f2f68b | |||
|
|
e284996c1c
|
||
|
|
51489a83ab
|
||
|
|
05426e4ced
|
||
|
|
445cdfa024
|
||
|
|
f74227fedb
|
||
|
|
32d1992632
|
||
| 48be35f1b1 | |||
| 87720ef285 | |||
|
|
193a4c2edd
|
||
|
|
134c81460a
|
||
|
|
b1a693e7cf
|
||
|
|
75bd879f84
|
||
|
|
33a9e1eaa9
|
||
|
|
7b321577db
|
||
|
|
61f12c2741
|
||
|
|
c58358c66e
|
||
|
|
287adbd365
|
||
|
|
9048052318
|
||
| cddc1e86f6 | |||
|
|
ce7387a409
|
||
|
|
f1ae5667de
|
||
|
|
67a9fc02d7
|
||
|
|
34849b28b0
|
||
| 8ce5f9708f | |||
|
|
cb2197893c
|
||
|
|
dabd892a25
|
||
|
|
eeabbdb7df
|
||
| 7a50bd23d6 | |||
| 64c8c3cb06 | |||
|
|
a2100b23a9
|
||
| 27195f693a | |||
|
9e74c89a80
|
|||
|
0774c88918
|
|||
| ef2d2b6422 | |||
|
a47e4fc16b
|
|||
|
9b89101afc
|
|||
|
|
ad90fcd539
|
||
|
|
705bd63b42
|
||
|
|
83e418cdee
|
||
|
|
7a193d6647
|
||
|
|
bb82b6b462
|
||
|
|
4e2e13108c
|
||
|
|
ca7475dca2
|
||
|
|
43a43e1a2c
|
||
|
|
595bb03c5a
|
||
|
|
62cd0eb7d1
|
||
|
|
f19baaf22a
|
||
|
|
23821f9e65
|
||
|
|
a33410eeb4
|
||
|
|
a1b238e86b
|
||
|
|
334b47353e
|
||
|
|
6848bd739c
|
||
|
|
7f77ad5528
|
||
| 6f2160b479 | |||
|
|
f08bb56a7a
|
||
|
|
fe1dfd8ec8
|
||
| c1f275463e | |||
| 324809f77e | |||
|
|
f9b07bcb01
|
||
|
|
986eb5387c
|
||
| f76e2c2f14 | |||
|
|
22a7bbe6eb
|
||
| 18f4deb30f | |||
|
|
9f9bf6fd80
|
||
|
|
d2987da70a
|
||
|
|
6b7a80e23a
|
||
|
|
42b9b27561
|
||
|
|
c17c980b69
|
||
|
ee42d68471
|
|||
|
7acc3b2106
|
|||
|
20c014607c
|
|||
|
|
f199d5d12a
|
||
|
|
4b17afa93d
|
||
|
|
6d52af53ae
|
||
|
|
4c5ad67652
|
||
|
|
3437a756eb
|
||
| 0d9fc4aa74 | |||
| 82475161a9 | |||
|
|
fb3b9af3e5
|
||
|
|
b1a0268e6b
|
||
| e1e7d8f87d | |||
|
|
5b46f3adf5
|
||
|
|
a8a8fba14c
|
||
|
|
8a7016a30b
|
||
|
|
e2618de7c6
|
||
| 90680368fb | |||
|
|
8d90847896 | ||
|
|
8da297811b | ||
|
|
fa56d6b772 | ||
|
|
ca1221e9f3 | ||
|
|
295d486761 | ||
|
|
e00390d102 | ||
|
|
b947480190 | ||
|
|
fa07978aac | ||
|
|
e758e258a8 | ||
|
|
805733939c | ||
|
|
f050d010fd
|
||
|
|
95fac38b53
|
||
| cb80465297 | |||
|
|
c7550b4f64
|
||
| 341284aa99 | |||
|
|
b34d040ce3
|
||
| 1142a4e2d5 | |||
|
|
f2c7aa2f09
|
||
| cca44d7542 | |||
| cdad7546fb | |||
| feb7833533 | |||
|
|
dfb12b8f62
|
||
|
|
6c2a97e7e5
|
||
| c8b65de7f6 | |||
| 2861254adf | |||
| 1d2910dadb | |||
|
|
251a170f2b
|
||
|
|
cbbb4c6e47
|
||
|
|
3aad27c7bd
|
||
|
|
7cff849d79
|
||
|
|
75ffd4e2f1
|
||
| b84f9109f6 | |||
| 7fd564726f | |||
|
b2a1b8caf5
|
|||
|
52cc2a8151
|
|||
|
|
c8e405d93a
|
||
|
|
5f74212603
|
||
|
|
1c3e893b6b
|
||
|
|
eec4533fea
|
||
|
|
6d20ac9a1c
|
||
|
|
27dd4163f0
|
||
|
|
1a55e5e895
|
||
|
|
8eb487600c
|
||
|
|
678e80a25d
|
||
|
|
30fb9805e5
|
||
|
|
e675970f4c
|
||
|
|
a0727e709f
|
||
|
|
55abbcc5ad
|
||
|
|
ffed398024
|
||
|
|
1a2482434c
|
||
| b530ad2f0f | |||
|
|
3c2fe7c15d
|
||
| aa7044dea7 | |||
|
|
a3f0d0f2cf
|
||
|
|
dc63506102
|
||
|
|
b87b9c2437
|
||
|
|
e580cc9991
|
||
|
|
68ab88c481
|
||
|
|
c7fe1bc3bc
|
||
| 84337c3a7d | |||
| 654b90f9ee | |||
| aa0ba18763 | |||
|
|
7dae66959e
|
||
|
|
b67d6139ac
|
||
|
|
b9259958f4
|
||
|
|
832d1e3bd7
|
||
|
|
f3f967f9f7
|
||
|
|
9407c7a94d
|
||
|
|
df3ec9f90a
|
||
|
|
25a0723166
|
||
|
|
6e884b789a
|
||
|
|
346e36e160
|
||
|
|
b7bf957dd2
|
||
|
|
084835f06a
|
||
|
|
cd7b05e2ff
|
||
|
|
7280a4c023
|
||
|
|
164400adec
|
||
|
|
c2e0909132
|
||
|
|
c44ce61e25
|
||
|
|
e2294c4029
|
||
|
|
bdc03a7181
|
||
|
|
959449a3f4
|
||
|
|
b4c9b31ce7
|
||
|
|
43f133ebd7
|
||
|
|
d9e767298b
|
||
|
|
dd482d7f2e
|
||
|
|
09d99ce9c2
|
||
|
|
8f9e1c3e84
|
||
| 4a045bf61c | |||
| f62e49f524 | |||
|
|
b0c787bbc7
|
||
|
|
86dc44d096
|
||
|
|
a1663b9f9d
|
||
|
|
aa3c2b4fa2
|
||
|
|
4c0d8283e3
|
||
|
|
d4a3f8dadb
|
||
|
|
9e988e92d1
|
||
|
|
4232df302b
|
||
|
|
2c8b3cdacc
|
||
|
|
51952ecdc2
|
||
|
|
68e0d00f6e
|
||
|
|
99dc36f13a
|
||
|
|
ee74c4847f
|
||
|
|
15b63eee73
|
||
|
|
c756528d32
|
||
|
|
fef29b4fc0
|
||
|
|
38608e053d
|
||
|
|
5f215b8ed8
|
||
|
|
87aae35974
|
||
|
|
6ad02e69a2
|
||
|
|
94ca0f3764
|
||
|
|
0fec37e0a9
|
||
|
|
620befd7c0
|
||
|
|
aba4930696
|
||
|
|
0492b42327
|
||
|
|
445a1c80a6
|
||
|
|
cf48f76553
|
||
|
|
70fa43f5d2
|
||
|
|
b37a0c25a4
|
||
|
|
3197743a55
|
||
|
|
3f49e4a3b8
|
||
| 2e1d930e0f | |||
| d849d28f62 | |||
|
|
f2a22adf6b
|
||
|
|
e1aaa2c434
|
||
|
|
e62bf67262
|
||
|
|
6df3d5933c
|
||
|
|
a5a90c4d83
|
||
|
|
80ef75ff42
|
||
|
|
67e2e45dd8
|
||
|
|
3834e5230b
|
||
|
|
4cb7c0998f
|
||
|
|
20382f7df7
|
||
|
|
add94eee8d
|
||
|
|
067dc3b63d
|
||
|
|
1a470cf1c8
|
||
|
|
f85b7f4f62
|
||
|
|
8635413002
|
||
|
|
a3da956b48
|
||
|
|
3c40dc98ca
|
||
| 28b31e63f9 | |||
|
|
efafd38f68
|
||
|
|
537e1a4774
|
||
|
|
c3b9ff8b4a
|
||
|
|
93d56f79d5
|
||
|
|
1a30345f46
|
||
|
|
778babcc05
|
||
|
|
fa3b53d3b3
|
||
|
|
0ca85656b7
|
||
|
|
f7183f68d5
|
||
| 87027b514b | |||
|
|
16ad621365
|
||
| 33e87d6472 | |||
|
03dc6c7a9c
|
|||
|
897b5bf4ea
|
|||
|
caea2d0121
|
|||
|
e1ff5c479e
|
|||
|
9b3386de30
|
|||
|
f2287c1186
|
|||
| b29197cf4e | |||
|
5c48055ac8
|
|||
|
5ead3476b7
|
|||
| fbf163740a | |||
|
|
1fc1457e97 | ||
| 1f57bbd9c2 | |||
|
2a2793ae44
|
|||
|
8773bf5f9e
|
|||
|
d9970c126a
|
|||
|
4e0d4bf86d
|
|||
|
333bcbfe7e
|
|||
| 875af6d14c | |||
| 8f87a03060 | |||
|
7838fe5f34
|
|||
|
512798d122
|
|||
|
384c28aaaa
|
|||
|
8e5d6dabdc
|
|||
|
ade9261c2c
|
|||
|
bd2a161306
|
|||
|
78c243c985
|
|||
|
cf62bfc5c2
|
|||
|
10f179a095
|
19
.drone.yml
19
.drone.yml
@@ -12,26 +12,27 @@ steps:
|
|||||||
settings:
|
settings:
|
||||||
restore: true
|
restore: true
|
||||||
mount:
|
mount:
|
||||||
- ./vendor
|
- ./vendor/cache
|
||||||
when:
|
when:
|
||||||
branch:
|
branch:
|
||||||
- master
|
- master
|
||||||
- name: rspec
|
- name: rspec
|
||||||
image: guildeducation/rails:2.7.2-12.22.0
|
image: gitea.kosmos.org/kosmos/akkounts-ci:0.9.1
|
||||||
environment:
|
environment:
|
||||||
RAILS_ENV: test
|
RAILS_ENV: test
|
||||||
|
REDIS_URL: redis://redis:6379/0
|
||||||
|
RS_REDIS_URL: redis://redis:6379/1
|
||||||
commands:
|
commands:
|
||||||
- bundle config unset deployment
|
- bundle config unset deployment
|
||||||
- bundle config set cache_all 'true'
|
- bundle config set cache_all 'true'
|
||||||
- bundle config set cache_path 'vendor/cache'
|
- bundle config set cache_path 'vendor/cache'
|
||||||
- bundle config set with 'development test'
|
- bundle config set with 'development test'
|
||||||
- bundle install --jobs=3 --retry=3
|
- bundle install --jobs=3 --retry=3
|
||||||
|
- bundle exec rails db:create
|
||||||
|
- bundle exec rails db:migrate
|
||||||
- yarn install
|
- yarn install
|
||||||
- rake css:build
|
- rake css:build
|
||||||
- rake spec
|
- bundle exec rspec
|
||||||
when:
|
|
||||||
branch:
|
|
||||||
- master
|
|
||||||
- name: rebuild-cache
|
- name: rebuild-cache
|
||||||
image: drillster/drone-volume-cache
|
image: drillster/drone-volume-cache
|
||||||
volumes:
|
volumes:
|
||||||
@@ -40,11 +41,15 @@ steps:
|
|||||||
settings:
|
settings:
|
||||||
rebuild: true
|
rebuild: true
|
||||||
mount:
|
mount:
|
||||||
- ./vendor
|
- ./vendor/cache
|
||||||
when:
|
when:
|
||||||
branch:
|
branch:
|
||||||
- master
|
- master
|
||||||
|
|
||||||
|
services:
|
||||||
|
- name: redis
|
||||||
|
image: redis
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
- name: cache
|
- name: cache
|
||||||
host:
|
host:
|
||||||
|
|||||||
74
.env.example
74
.env.example
@@ -1,3 +1,71 @@
|
|||||||
EJABBERD_API_URL='https://xmpp.kosmos.org/api'
|
# PRIMARY_DOMAIN=kosmos.org
|
||||||
LNDHUB_API_URL='http://localhost:3023'
|
# AKKOUNTS_DOMAIN=accounts.example.com
|
||||||
LNDHUB_PUBLIC_URL='https://lndhub.kosmos.org'
|
|
||||||
|
# SMTP_SERVER=smtp.example.com
|
||||||
|
# SMTP_PORT=587
|
||||||
|
# SMTP_LOGIN=accounts
|
||||||
|
# SMTP_PASSWORD=123abc
|
||||||
|
# SMTP_FROM_ADDRESS=accounts@example.com
|
||||||
|
# SMTP_DOMAIN=example.com
|
||||||
|
# SMTP_AUTH_METHOD=plain
|
||||||
|
# SMTP_ENABLE_STARTTLS=auto
|
||||||
|
|
||||||
|
# S3_ENABLED=true
|
||||||
|
# S3_ENDPOINT=https://s3.kosmos.org
|
||||||
|
# S3_REGION=garage
|
||||||
|
# S3_BUCKET=akkounts-production
|
||||||
|
# S3_ALIAS_HOST=https://accounts.web.s3.kosmos.org
|
||||||
|
# S3_ACCESS_KEY=123456abcdefg
|
||||||
|
# S3_SECRET_KEY=123456789123456789123456789
|
||||||
|
|
||||||
|
# LDAP_HOST=localhost
|
||||||
|
# LDAP_PORT=389
|
||||||
|
# LDAP_ADMIN_PASSWORD=passthebutter
|
||||||
|
# LDAP_SUFFIX='dc=kosmos,dc=org'
|
||||||
|
|
||||||
|
# REDIS_URL='redis://localhost:6379/1'
|
||||||
|
|
||||||
|
# WEBHOOKS_ALLOWED_IPS='10.1.1.163'
|
||||||
|
|
||||||
|
#
|
||||||
|
# Service Integrations
|
||||||
|
# (sorted alphabetically by service name)
|
||||||
|
#
|
||||||
|
|
||||||
|
# BTCPAY_PUBLIC_URL='https://btcpay.example.com'
|
||||||
|
# BTCPAY_API_URL='http://localhost:23001/api/v1'
|
||||||
|
# BTCPAY_STORE_ID=''
|
||||||
|
# BTCPAY_AUTH_TOKEN=''
|
||||||
|
|
||||||
|
# DISCOURSE_PUBLIC_URL='https://community.kosmos.org'
|
||||||
|
# DISCOURSE_CONNECT_SECRET='discourse_connect_ftw'
|
||||||
|
|
||||||
|
# DRONECI_PUBLIC_URL='https://drone.kosmos.org'
|
||||||
|
|
||||||
|
# EJABBERD_ADMIN_URL='https://xmpp.kosmos.org/admin'
|
||||||
|
# EJABBERD_API_URL='https://xmpp.kosmos.org/api'
|
||||||
|
|
||||||
|
# GITEA_PUBLIC_URL='https://gitea.kosmos.org'
|
||||||
|
|
||||||
|
# LNDHUB_API_URL='http://localhost:3023'
|
||||||
|
# LNDHUB_PUBLIC_URL='https://lndhub.kosmos.org'
|
||||||
|
# LNDHUB_PUBLIC_KEY='0123d3be18617f39cf645851e3ba63f51fc13f0bb09e3bb25e6fd4de556486d946'
|
||||||
|
# LNDHUB_ADMIN_UI=true
|
||||||
|
# LNDHUB_ADMIN_TOKEN=123456789
|
||||||
|
# LNDHUB_PG_HOST=localhost
|
||||||
|
# LNDHUB_PG_PORT=5432
|
||||||
|
# LNDHUB_PG_DATABASE=lndhub
|
||||||
|
# LNDHUB_PG_USERNAME=lndhub
|
||||||
|
# LNDHUB_PG_PASSWORD=''
|
||||||
|
|
||||||
|
# MASTODON_PUBLIC_URL='https://kosmos.social'
|
||||||
|
# MASTODON_ADDRESS_DOMAIN='https://kosmos.org'
|
||||||
|
|
||||||
|
# MEDIAWIKI_PUBLIC_URL='https://wiki.kosmos.org'
|
||||||
|
|
||||||
|
# NOSTR_PRIVATE_KEY='123456abcdef...'
|
||||||
|
# NOSTR_PUBLIC_KEY='123456abcdef...'
|
||||||
|
# NOSTR_RELAY_URL='wss://nostr.kosmos.org'
|
||||||
|
|
||||||
|
# RS_STORAGE_URL='https://storage.kosmos.org'
|
||||||
|
# RS_REDIS_URL='redis://localhost:6379/2'
|
||||||
|
|||||||
@@ -1,3 +0,0 @@
|
|||||||
EJABBERD_API_URL='https://xmpp.kosmos.org:5443/api'
|
|
||||||
LNDHUB_API_URL='http://10.1.1.163:3023'
|
|
||||||
LNDHUB_PUBLIC_URL='https://lndhub.kosmos.org'
|
|
||||||
26
.env.test
26
.env.test
@@ -1,3 +1,27 @@
|
|||||||
|
PRIMARY_DOMAIN=kosmos.org
|
||||||
|
AKKOUNTS_DOMAIN=accounts.kosmos.org
|
||||||
|
|
||||||
|
REDIS_URL='redis://localhost:6379/0'
|
||||||
|
|
||||||
|
BTCPAY_PUBLIC_URL='https://btcpay.example.com'
|
||||||
|
BTCPAY_API_URL='http://btcpay.example.com/api/v1'
|
||||||
|
BTCPAY_STORE_ID='123456'
|
||||||
|
|
||||||
|
DISCOURSE_PUBLIC_URL='http://discourse.example.com'
|
||||||
|
DISCOURSE_CONNECT_SECRET='discourse_connect_ftw'
|
||||||
|
|
||||||
EJABBERD_API_URL='http://xmpp.example.com/api'
|
EJABBERD_API_URL='http://xmpp.example.com/api'
|
||||||
LNDHUB_API_URL='http://localhost:3023'
|
|
||||||
|
MASTODON_PUBLIC_URL='http://example.social'
|
||||||
|
|
||||||
|
LNDHUB_API_URL='http://localhost:3026'
|
||||||
LNDHUB_PUBLIC_URL='https://lndhub.kosmos.org'
|
LNDHUB_PUBLIC_URL='https://lndhub.kosmos.org'
|
||||||
|
LNDHUB_PUBLIC_KEY='024cd3be18617f39cf645851e3ba63f51fc13f0bb09e3bb25e6fd4de556486d946'
|
||||||
|
|
||||||
|
NOSTR_PRIVATE_KEY='7c3ef7e448505f0615137af38569d01807d3b05b5005d5ecf8aaafcd40323cea'
|
||||||
|
NOSTR_PUBLIC_KEY='bdd76ce2934b2f591f9fad2ebe9da18f20d2921de527494ba00eeaa0a0efadcf'
|
||||||
|
|
||||||
|
RS_STORAGE_URL='https://storage.kosmos.org'
|
||||||
|
RS_REDIS_URL='redis://localhost:6379/1'
|
||||||
|
|
||||||
|
WEBHOOKS_ALLOWED_IPS='10.1.1.23'
|
||||||
|
|||||||
14
.gitea/release-drafter.yml
Normal file
14
.gitea/release-drafter.yml
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
name-template: 'v$RESOLVED_VERSION'
|
||||||
|
tag-template: 'v$RESOLVED_VERSION'
|
||||||
|
version-resolver:
|
||||||
|
major:
|
||||||
|
labels:
|
||||||
|
- 'release/major'
|
||||||
|
minor:
|
||||||
|
labels:
|
||||||
|
- 'release/minor'
|
||||||
|
- 'feature'
|
||||||
|
patch:
|
||||||
|
labels:
|
||||||
|
- 'release/patch'
|
||||||
|
default: patch
|
||||||
11
.gitea/workflows/release_drafter.yml
Normal file
11
.gitea/workflows/release_drafter.yml
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
name: Release Drafter
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
types: [closed]
|
||||||
|
jobs:
|
||||||
|
release_drafter_job:
|
||||||
|
name: Update release notes draft
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Release Drafter
|
||||||
|
uses: https://github.com/raucao/gitea-release-drafter@dev
|
||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -23,6 +23,7 @@
|
|||||||
!/tmp/pids/
|
!/tmp/pids/
|
||||||
!/tmp/pids/.keep
|
!/tmp/pids/.keep
|
||||||
|
|
||||||
|
/storage
|
||||||
|
|
||||||
/public/assets
|
/public/assets
|
||||||
.byebug_history
|
.byebug_history
|
||||||
@@ -39,6 +40,7 @@ yarn-debug.log*
|
|||||||
|
|
||||||
# Ignore local dotenv config file
|
# Ignore local dotenv config file
|
||||||
.env
|
.env
|
||||||
|
.env.development
|
||||||
|
|
||||||
# Ignore redis dumps from sidekiq
|
# Ignore redis dumps from sidekiq
|
||||||
dump.rdb
|
dump.rdb
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
2.7.2
|
3.3.0
|
||||||
|
|||||||
22
Dockerfile
Normal file
22
Dockerfile
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
# syntax=docker/dockerfile:1
|
||||||
|
FROM ruby:3.3.4
|
||||||
|
|
||||||
|
SHELL ["/bin/bash", "-o", "pipefail", "-c"]
|
||||||
|
|
||||||
|
RUN apt-get update -qq && apt-get install -y --no-install-recommends curl \
|
||||||
|
ldap-utils tini libvips
|
||||||
|
|
||||||
|
RUN curl -fsSL https://deb.nodesource.com/setup_lts.x | bash -
|
||||||
|
RUN apt-get update && apt-get install -y nodejs
|
||||||
|
|
||||||
|
WORKDIR /akkounts
|
||||||
|
|
||||||
|
COPY ["Gemfile", "Gemfile.lock", "package.json", "./"]
|
||||||
|
|
||||||
|
RUN bundle install
|
||||||
|
RUN gem install foreman
|
||||||
|
RUN npm install -g yarn
|
||||||
|
RUN yarn install
|
||||||
|
|
||||||
|
ENTRYPOINT ["/usr/bin/tini", "--"]
|
||||||
|
EXPOSE 3000
|
||||||
37
Gemfile
37
Gemfile
@@ -2,7 +2,7 @@ source 'https://rubygems.org'
|
|||||||
git_source(:github) { |repo| "https://github.com/#{repo}.git" }
|
git_source(:github) { |repo| "https://github.com/#{repo}.git" }
|
||||||
|
|
||||||
# Bundle edge Rails instead: gem 'rails', github: 'rails/rails'
|
# Bundle edge Rails instead: gem 'rails', github: 'rails/rails'
|
||||||
gem 'rails', '~> 7.0.2'
|
gem 'rails', '~> 7.1'
|
||||||
# Use Puma as the app server
|
# Use Puma as the app server
|
||||||
gem 'puma', '~> 4.1'
|
gem 'puma', '~> 4.1'
|
||||||
# View components
|
# View components
|
||||||
@@ -22,7 +22,7 @@ gem 'jbuilder', '~> 2.7'
|
|||||||
# Use Redis adapter to run Action Cable in production
|
# Use Redis adapter to run Action Cable in production
|
||||||
# gem 'redis', '~> 4.0'
|
# gem 'redis', '~> 4.0'
|
||||||
# Use Active Model has_secure_password
|
# Use Active Model has_secure_password
|
||||||
# gem 'bcrypt', '~> 3.1.7'
|
gem 'bcrypt', '~> 3.1'
|
||||||
|
|
||||||
# Configuration
|
# Configuration
|
||||||
gem 'dotenv-rails'
|
gem 'dotenv-rails'
|
||||||
@@ -32,32 +32,53 @@ gem 'lockbox'
|
|||||||
|
|
||||||
# Authentication
|
# Authentication
|
||||||
gem 'warden'
|
gem 'warden'
|
||||||
gem 'devise'
|
gem 'devise', '~> 4.9.0'
|
||||||
gem 'devise_ldap_authenticatable'
|
gem 'devise_ldap_authenticatable'
|
||||||
gem 'net-ldap'
|
gem 'net-ldap'
|
||||||
|
|
||||||
# Utilities
|
# Utilities
|
||||||
|
gem "image_processing", "~> 1.12.2"
|
||||||
gem "rqrcode", "~> 2.0"
|
gem "rqrcode", "~> 2.0"
|
||||||
|
gem 'rails-settings-cached', '~> 2.8.3'
|
||||||
|
gem 'pagy', '~> 6.0', '>= 6.0.2'
|
||||||
|
gem 'flipper'
|
||||||
|
gem 'flipper-active_record'
|
||||||
|
gem 'flipper-ui'
|
||||||
|
|
||||||
# HTTP requests
|
# HTTP requests
|
||||||
gem 'faraday'
|
gem 'faraday'
|
||||||
|
gem 'down'
|
||||||
|
gem 'aws-sdk-s3', require: false
|
||||||
|
|
||||||
# Background/scheduled jobs
|
# Background/scheduled jobs
|
||||||
gem 'sidekiq'
|
gem 'sidekiq', '< 7'
|
||||||
gem 'sidekiq-scheduler'
|
gem 'sidekiq-scheduler'
|
||||||
|
|
||||||
|
# Monitoring
|
||||||
|
gem "sentry-ruby"
|
||||||
|
gem "sentry-rails"
|
||||||
|
|
||||||
|
# Services
|
||||||
|
gem 'discourse_api'
|
||||||
|
gem "lnurl"
|
||||||
|
gem 'manifique', '~> 1.1.0'
|
||||||
|
gem 'nostr', '~> 0.6.0'
|
||||||
|
|
||||||
group :development, :test do
|
group :development, :test do
|
||||||
# Use sqlite3 as the database for Active Record
|
# Use sqlite3 as the database for Active Record
|
||||||
gem 'sqlite3', '~> 1.4'
|
gem 'sqlite3', '~> 1.7.2'
|
||||||
gem 'rspec-rails'
|
gem 'rspec-rails'
|
||||||
|
gem 'rails-controller-testing'
|
||||||
end
|
end
|
||||||
|
|
||||||
group :development do
|
group :development do
|
||||||
# Access an interactive console on exception pages or by calling 'console' anywhere in the code.
|
# Access an interactive console on exception pages or by calling 'console' anywhere in the code.
|
||||||
gem 'web-console', '>= 3.3.0'
|
gem 'web-console', '~> 4.2'
|
||||||
gem 'listen', '~> 3.2'
|
gem 'listen', '~> 3.2'
|
||||||
gem 'letter_opener'
|
gem 'letter_opener'
|
||||||
gem 'letter_opener_web'
|
gem 'letter_opener_web'
|
||||||
|
gem 'faker'
|
||||||
|
gem 'solargraph'
|
||||||
end
|
end
|
||||||
|
|
||||||
group :test do
|
group :test do
|
||||||
@@ -68,8 +89,8 @@ group :test do
|
|||||||
end
|
end
|
||||||
|
|
||||||
group :production do
|
group :production do
|
||||||
# Use postgresql as the database for Active Record
|
gem 'pg', '~> 1.5'
|
||||||
gem 'pg', '~> 1.2.3'
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# Windows does not include zoneinfo files, so bundle the tzinfo-data gem
|
# Windows does not include zoneinfo files, so bundle the tzinfo-data gem
|
||||||
gem 'tzinfo-data', platforms: [:mingw, :mswin, :x64_mingw, :jruby]
|
gem 'tzinfo-data', platforms: [:mingw, :mswin, :x64_mingw, :jruby]
|
||||||
|
|||||||
590
Gemfile.lock
590
Gemfile.lock
@@ -1,100 +1,136 @@
|
|||||||
GEM
|
GEM
|
||||||
remote: https://rubygems.org/
|
remote: https://rubygems.org/
|
||||||
specs:
|
specs:
|
||||||
actioncable (7.0.2.2)
|
actioncable (7.1.3)
|
||||||
actionpack (= 7.0.2.2)
|
actionpack (= 7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activesupport (= 7.1.3)
|
||||||
nio4r (~> 2.0)
|
nio4r (~> 2.0)
|
||||||
websocket-driver (>= 0.6.1)
|
websocket-driver (>= 0.6.1)
|
||||||
actionmailbox (7.0.2.2)
|
zeitwerk (~> 2.6)
|
||||||
actionpack (= 7.0.2.2)
|
actionmailbox (7.1.3)
|
||||||
activejob (= 7.0.2.2)
|
actionpack (= 7.1.3)
|
||||||
activerecord (= 7.0.2.2)
|
activejob (= 7.1.3)
|
||||||
activestorage (= 7.0.2.2)
|
activerecord (= 7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activestorage (= 7.1.3)
|
||||||
|
activesupport (= 7.1.3)
|
||||||
mail (>= 2.7.1)
|
mail (>= 2.7.1)
|
||||||
net-imap
|
net-imap
|
||||||
net-pop
|
net-pop
|
||||||
net-smtp
|
net-smtp
|
||||||
actionmailer (7.0.2.2)
|
actionmailer (7.1.3)
|
||||||
actionpack (= 7.0.2.2)
|
actionpack (= 7.1.3)
|
||||||
actionview (= 7.0.2.2)
|
actionview (= 7.1.3)
|
||||||
activejob (= 7.0.2.2)
|
activejob (= 7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activesupport (= 7.1.3)
|
||||||
mail (~> 2.5, >= 2.5.4)
|
mail (~> 2.5, >= 2.5.4)
|
||||||
net-imap
|
net-imap
|
||||||
net-pop
|
net-pop
|
||||||
net-smtp
|
net-smtp
|
||||||
rails-dom-testing (~> 2.0)
|
rails-dom-testing (~> 2.2)
|
||||||
actionpack (7.0.2.2)
|
actionpack (7.1.3)
|
||||||
actionview (= 7.0.2.2)
|
actionview (= 7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activesupport (= 7.1.3)
|
||||||
rack (~> 2.0, >= 2.2.0)
|
nokogiri (>= 1.8.5)
|
||||||
|
racc
|
||||||
|
rack (>= 2.2.4)
|
||||||
|
rack-session (>= 1.0.1)
|
||||||
rack-test (>= 0.6.3)
|
rack-test (>= 0.6.3)
|
||||||
rails-dom-testing (~> 2.0)
|
rails-dom-testing (~> 2.2)
|
||||||
rails-html-sanitizer (~> 1.0, >= 1.2.0)
|
rails-html-sanitizer (~> 1.6)
|
||||||
actiontext (7.0.2.2)
|
actiontext (7.1.3)
|
||||||
actionpack (= 7.0.2.2)
|
actionpack (= 7.1.3)
|
||||||
activerecord (= 7.0.2.2)
|
activerecord (= 7.1.3)
|
||||||
activestorage (= 7.0.2.2)
|
activestorage (= 7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activesupport (= 7.1.3)
|
||||||
globalid (>= 0.6.0)
|
globalid (>= 0.6.0)
|
||||||
nokogiri (>= 1.8.5)
|
nokogiri (>= 1.8.5)
|
||||||
actionview (7.0.2.2)
|
actionview (7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activesupport (= 7.1.3)
|
||||||
builder (~> 3.1)
|
builder (~> 3.1)
|
||||||
erubi (~> 1.4)
|
erubi (~> 1.11)
|
||||||
rails-dom-testing (~> 2.0)
|
rails-dom-testing (~> 2.2)
|
||||||
rails-html-sanitizer (~> 1.1, >= 1.2.0)
|
rails-html-sanitizer (~> 1.6)
|
||||||
activejob (7.0.2.2)
|
activejob (7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activesupport (= 7.1.3)
|
||||||
globalid (>= 0.3.6)
|
globalid (>= 0.3.6)
|
||||||
activemodel (7.0.2.2)
|
activemodel (7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activesupport (= 7.1.3)
|
||||||
activerecord (7.0.2.2)
|
activerecord (7.1.3)
|
||||||
activemodel (= 7.0.2.2)
|
activemodel (= 7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activesupport (= 7.1.3)
|
||||||
activestorage (7.0.2.2)
|
timeout (>= 0.4.0)
|
||||||
actionpack (= 7.0.2.2)
|
activestorage (7.1.3)
|
||||||
activejob (= 7.0.2.2)
|
actionpack (= 7.1.3)
|
||||||
activerecord (= 7.0.2.2)
|
activejob (= 7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
activerecord (= 7.1.3)
|
||||||
|
activesupport (= 7.1.3)
|
||||||
marcel (~> 1.0)
|
marcel (~> 1.0)
|
||||||
mini_mime (>= 1.1.0)
|
activesupport (7.1.3)
|
||||||
activesupport (7.0.2.2)
|
base64
|
||||||
|
bigdecimal
|
||||||
concurrent-ruby (~> 1.0, >= 1.0.2)
|
concurrent-ruby (~> 1.0, >= 1.0.2)
|
||||||
|
connection_pool (>= 2.2.5)
|
||||||
|
drb
|
||||||
i18n (>= 1.6, < 2)
|
i18n (>= 1.6, < 2)
|
||||||
minitest (>= 5.1)
|
minitest (>= 5.1)
|
||||||
|
mutex_m
|
||||||
tzinfo (~> 2.0)
|
tzinfo (~> 2.0)
|
||||||
addressable (2.8.0)
|
addressable (2.8.6)
|
||||||
public_suffix (>= 2.0.2, < 5.0)
|
public_suffix (>= 2.0.2, < 6.0)
|
||||||
bcrypt (3.1.16)
|
ast (2.4.2)
|
||||||
|
aws-eventstream (1.3.0)
|
||||||
|
aws-partitions (1.886.0)
|
||||||
|
aws-sdk-core (3.191.0)
|
||||||
|
aws-eventstream (~> 1, >= 1.3.0)
|
||||||
|
aws-partitions (~> 1, >= 1.651.0)
|
||||||
|
aws-sigv4 (~> 1.8)
|
||||||
|
jmespath (~> 1, >= 1.6.1)
|
||||||
|
aws-sdk-kms (1.77.0)
|
||||||
|
aws-sdk-core (~> 3, >= 3.191.0)
|
||||||
|
aws-sigv4 (~> 1.1)
|
||||||
|
aws-sdk-s3 (1.143.0)
|
||||||
|
aws-sdk-core (~> 3, >= 3.191.0)
|
||||||
|
aws-sdk-kms (~> 1)
|
||||||
|
aws-sigv4 (~> 1.8)
|
||||||
|
aws-sigv4 (1.8.0)
|
||||||
|
aws-eventstream (~> 1, >= 1.0.2)
|
||||||
|
backport (1.2.0)
|
||||||
|
base64 (0.2.0)
|
||||||
|
bcrypt (3.1.20)
|
||||||
|
bech32 (1.4.2)
|
||||||
|
thor (>= 1.1.0)
|
||||||
|
benchmark (0.3.0)
|
||||||
|
bigdecimal (3.1.6)
|
||||||
bindex (0.8.1)
|
bindex (0.8.1)
|
||||||
|
bip-schnorr (0.7.0)
|
||||||
|
ecdsa_ext (~> 0.5.0)
|
||||||
builder (3.2.4)
|
builder (3.2.4)
|
||||||
capybara (3.36.0)
|
capybara (3.40.0)
|
||||||
addressable
|
addressable
|
||||||
matrix
|
matrix
|
||||||
mini_mime (>= 0.1.3)
|
mini_mime (>= 0.1.3)
|
||||||
nokogiri (~> 1.8)
|
nokogiri (~> 1.11)
|
||||||
rack (>= 1.6.0)
|
rack (>= 1.6.0)
|
||||||
rack-test (>= 0.6.3)
|
rack-test (>= 0.6.3)
|
||||||
regexp_parser (>= 1.5, < 3.0)
|
regexp_parser (>= 1.5, < 3.0)
|
||||||
xpath (~> 3.2)
|
xpath (~> 3.2)
|
||||||
chunky_png (1.4.0)
|
chunky_png (1.4.0)
|
||||||
concurrent-ruby (1.1.9)
|
concurrent-ruby (1.2.3)
|
||||||
connection_pool (2.2.5)
|
connection_pool (2.4.1)
|
||||||
crack (0.4.5)
|
crack (0.4.6)
|
||||||
|
bigdecimal
|
||||||
rexml
|
rexml
|
||||||
crass (1.0.6)
|
crass (1.0.6)
|
||||||
cssbundling-rails (1.0.0)
|
cssbundling-rails (1.4.0)
|
||||||
railties (>= 6.0.0)
|
railties (>= 6.0.0)
|
||||||
database_cleaner (2.0.1)
|
database_cleaner (2.0.2)
|
||||||
database_cleaner-active_record (~> 2.0.0)
|
database_cleaner-active_record (>= 2, < 3)
|
||||||
database_cleaner-active_record (2.0.1)
|
database_cleaner-active_record (2.1.0)
|
||||||
activerecord (>= 5.a)
|
activerecord (>= 5.a)
|
||||||
database_cleaner-core (~> 2.0.0)
|
database_cleaner-core (~> 2.0.0)
|
||||||
database_cleaner-core (2.0.1)
|
database_cleaner-core (2.0.1)
|
||||||
devise (4.8.1)
|
date (3.3.4)
|
||||||
|
devise (4.9.3)
|
||||||
bcrypt (~> 3.0)
|
bcrypt (~> 3.0)
|
||||||
orm_adapter (~> 0.1)
|
orm_adapter (~> 0.1)
|
||||||
railties (>= 4.1.0)
|
railties (>= 4.1.0)
|
||||||
@@ -103,241 +139,407 @@ GEM
|
|||||||
devise_ldap_authenticatable (0.8.7)
|
devise_ldap_authenticatable (0.8.7)
|
||||||
devise (>= 3.4.1)
|
devise (>= 3.4.1)
|
||||||
net-ldap (>= 0.16.0)
|
net-ldap (>= 0.16.0)
|
||||||
diff-lcs (1.5.0)
|
diff-lcs (1.5.1)
|
||||||
digest (3.1.0)
|
discourse_api (2.0.1)
|
||||||
dotenv (2.7.6)
|
faraday (~> 2.7)
|
||||||
dotenv-rails (2.7.6)
|
faraday-follow_redirects
|
||||||
dotenv (= 2.7.6)
|
faraday-multipart
|
||||||
|
rack (>= 1.6)
|
||||||
|
dotenv (2.8.1)
|
||||||
|
dotenv-rails (2.8.1)
|
||||||
|
dotenv (= 2.8.1)
|
||||||
railties (>= 3.2)
|
railties (>= 3.2)
|
||||||
|
down (5.4.1)
|
||||||
|
addressable (~> 2.8)
|
||||||
|
drb (2.2.0)
|
||||||
|
ruby2_keywords
|
||||||
e2mmap (0.1.0)
|
e2mmap (0.1.0)
|
||||||
erubi (1.10.0)
|
ecdsa (1.2.0)
|
||||||
et-orbi (1.2.6)
|
ecdsa_ext (0.5.1)
|
||||||
|
ecdsa (~> 1.2.0)
|
||||||
|
erubi (1.12.0)
|
||||||
|
et-orbi (1.2.7)
|
||||||
tzinfo
|
tzinfo
|
||||||
factory_bot (6.2.0)
|
event_emitter (0.2.6)
|
||||||
|
eventmachine (1.2.7)
|
||||||
|
factory_bot (6.4.6)
|
||||||
activesupport (>= 5.0.0)
|
activesupport (>= 5.0.0)
|
||||||
factory_bot_rails (6.2.0)
|
factory_bot_rails (6.4.3)
|
||||||
factory_bot (~> 6.2.0)
|
factory_bot (~> 6.4)
|
||||||
railties (>= 5.0.0)
|
railties (>= 5.0.0)
|
||||||
faraday (2.2.0)
|
faker (3.2.3)
|
||||||
faraday-net_http (~> 2.0)
|
i18n (>= 1.8.11, < 2)
|
||||||
ruby2_keywords (>= 0.0.4)
|
faraday (2.9.0)
|
||||||
faraday-net_http (2.0.1)
|
faraday-net_http (>= 2.0, < 3.2)
|
||||||
ffi (1.15.5)
|
faraday-follow_redirects (0.3.0)
|
||||||
fugit (1.5.2)
|
faraday (>= 1, < 3)
|
||||||
et-orbi (~> 1.1, >= 1.1.8)
|
faraday-multipart (1.0.4)
|
||||||
|
multipart-post (~> 2)
|
||||||
|
faraday-net_http (3.1.0)
|
||||||
|
net-http
|
||||||
|
faye-websocket (0.11.3)
|
||||||
|
eventmachine (>= 0.12.0)
|
||||||
|
websocket-driver (>= 0.5.1)
|
||||||
|
ffi (1.16.3)
|
||||||
|
flipper (1.2.2)
|
||||||
|
concurrent-ruby (< 2)
|
||||||
|
flipper-active_record (1.2.2)
|
||||||
|
activerecord (>= 4.2, < 8)
|
||||||
|
flipper (~> 1.2.2)
|
||||||
|
flipper-ui (1.2.2)
|
||||||
|
erubi (>= 1.0.0, < 2.0.0)
|
||||||
|
flipper (~> 1.2.2)
|
||||||
|
rack (>= 1.4, < 4)
|
||||||
|
rack-protection (>= 1.5.3, <= 4.0.0)
|
||||||
|
sanitize (< 7)
|
||||||
|
fugit (1.9.0)
|
||||||
|
et-orbi (~> 1, >= 1.2.7)
|
||||||
raabro (~> 1.4)
|
raabro (~> 1.4)
|
||||||
globalid (1.0.0)
|
globalid (1.2.1)
|
||||||
activesupport (>= 5.0)
|
activesupport (>= 6.1)
|
||||||
hashdiff (1.0.1)
|
hashdiff (1.1.0)
|
||||||
i18n (1.9.1)
|
i18n (1.14.1)
|
||||||
concurrent-ruby (~> 1.0)
|
concurrent-ruby (~> 1.0)
|
||||||
importmap-rails (1.0.2)
|
image_processing (1.12.2)
|
||||||
|
mini_magick (>= 4.9.5, < 5)
|
||||||
|
ruby-vips (>= 2.0.17, < 3)
|
||||||
|
importmap-rails (2.0.1)
|
||||||
actionpack (>= 6.0.0)
|
actionpack (>= 6.0.0)
|
||||||
|
activesupport (>= 6.0.0)
|
||||||
railties (>= 6.0.0)
|
railties (>= 6.0.0)
|
||||||
io-wait (0.2.1)
|
io-console (0.7.2)
|
||||||
|
irb (1.11.1)
|
||||||
|
rdoc
|
||||||
|
reline (>= 0.4.2)
|
||||||
|
jaro_winkler (1.5.6)
|
||||||
jbuilder (2.11.5)
|
jbuilder (2.11.5)
|
||||||
actionview (>= 5.0.0)
|
actionview (>= 5.0.0)
|
||||||
activesupport (>= 5.0.0)
|
activesupport (>= 5.0.0)
|
||||||
launchy (2.5.0)
|
jmespath (1.6.2)
|
||||||
addressable (~> 2.7)
|
json (2.7.1)
|
||||||
letter_opener (1.7.0)
|
kramdown (2.4.0)
|
||||||
launchy (~> 2.2)
|
rexml
|
||||||
|
kramdown-parser-gfm (1.1.0)
|
||||||
|
kramdown (~> 2.0)
|
||||||
|
language_server-protocol (3.17.0.3)
|
||||||
|
launchy (2.5.2)
|
||||||
|
addressable (~> 2.8)
|
||||||
|
letter_opener (1.8.1)
|
||||||
|
launchy (>= 2.2, < 3)
|
||||||
letter_opener_web (2.0.0)
|
letter_opener_web (2.0.0)
|
||||||
actionmailer (>= 5.2)
|
actionmailer (>= 5.2)
|
||||||
letter_opener (~> 1.7)
|
letter_opener (~> 1.7)
|
||||||
railties (>= 5.2)
|
railties (>= 5.2)
|
||||||
rexml
|
rexml
|
||||||
listen (3.7.1)
|
listen (3.8.0)
|
||||||
rb-fsevent (~> 0.10, >= 0.10.3)
|
rb-fsevent (~> 0.10, >= 0.10.3)
|
||||||
rb-inotify (~> 0.9, >= 0.9.10)
|
rb-inotify (~> 0.9, >= 0.9.10)
|
||||||
lockbox (0.6.8)
|
lnurl (1.1.0)
|
||||||
loofah (2.14.0)
|
bech32 (~> 1.1)
|
||||||
|
lockbox (1.3.2)
|
||||||
|
loofah (2.22.0)
|
||||||
crass (~> 1.0.2)
|
crass (~> 1.0.2)
|
||||||
nokogiri (>= 1.5.9)
|
nokogiri (>= 1.12.0)
|
||||||
mail (2.7.1)
|
mail (2.8.1)
|
||||||
mini_mime (>= 0.1.1)
|
mini_mime (>= 0.1.1)
|
||||||
|
net-imap
|
||||||
|
net-pop
|
||||||
|
net-smtp
|
||||||
|
manifique (1.1.0)
|
||||||
|
faraday (~> 2.9.0)
|
||||||
|
faraday-follow_redirects (= 0.3.0)
|
||||||
|
nokogiri (~> 1.16.0)
|
||||||
marcel (1.0.2)
|
marcel (1.0.2)
|
||||||
matrix (0.4.2)
|
matrix (0.4.2)
|
||||||
method_source (1.0.0)
|
method_source (1.0.0)
|
||||||
mini_mime (1.1.2)
|
mini_magick (4.12.0)
|
||||||
minitest (5.15.0)
|
mini_mime (1.1.5)
|
||||||
net-imap (0.2.3)
|
mini_portile2 (2.8.5)
|
||||||
digest
|
minitest (5.21.2)
|
||||||
|
multipart-post (2.3.0)
|
||||||
|
mutex_m (0.2.0)
|
||||||
|
net-http (0.4.1)
|
||||||
|
uri
|
||||||
|
net-imap (0.4.9.1)
|
||||||
|
date
|
||||||
net-protocol
|
net-protocol
|
||||||
strscan
|
net-ldap (0.19.0)
|
||||||
net-ldap (0.17.0)
|
net-pop (0.1.2)
|
||||||
net-pop (0.1.1)
|
|
||||||
digest
|
|
||||||
net-protocol
|
net-protocol
|
||||||
|
net-protocol (0.2.2)
|
||||||
timeout
|
timeout
|
||||||
net-protocol (0.1.2)
|
net-smtp (0.4.0.1)
|
||||||
io-wait
|
|
||||||
timeout
|
|
||||||
net-smtp (0.3.1)
|
|
||||||
digest
|
|
||||||
net-protocol
|
net-protocol
|
||||||
timeout
|
nio4r (2.7.0)
|
||||||
nio4r (2.5.8)
|
nokogiri (1.16.0)
|
||||||
nokogiri (1.13.1-x86_64-linux)
|
mini_portile2 (~> 2.8.2)
|
||||||
racc (~> 1.4)
|
racc (~> 1.4)
|
||||||
|
nokogiri (1.16.0-arm64-darwin)
|
||||||
|
racc (~> 1.4)
|
||||||
|
nokogiri (1.16.0-x86_64-linux)
|
||||||
|
racc (~> 1.4)
|
||||||
|
nostr (0.6.0)
|
||||||
|
bech32 (~> 1.4)
|
||||||
|
bip-schnorr (~> 0.7)
|
||||||
|
ecdsa (~> 1.2)
|
||||||
|
event_emitter (~> 0.2)
|
||||||
|
faye-websocket (~> 0.11)
|
||||||
|
json (~> 2.6)
|
||||||
orm_adapter (0.5.0)
|
orm_adapter (0.5.0)
|
||||||
pg (1.2.3)
|
pagy (6.4.3)
|
||||||
public_suffix (4.0.6)
|
parallel (1.24.0)
|
||||||
puma (4.3.11)
|
parser (3.3.0.5)
|
||||||
|
ast (~> 2.4.1)
|
||||||
|
racc
|
||||||
|
pg (1.5.4)
|
||||||
|
psych (5.1.2)
|
||||||
|
stringio
|
||||||
|
public_suffix (5.0.4)
|
||||||
|
puma (4.3.12)
|
||||||
nio4r (~> 2.0)
|
nio4r (~> 2.0)
|
||||||
raabro (1.4.0)
|
raabro (1.4.0)
|
||||||
racc (1.6.0)
|
racc (1.7.3)
|
||||||
rack (2.2.3)
|
rack (2.2.8)
|
||||||
rack-test (1.1.0)
|
rack-protection (3.2.0)
|
||||||
rack (>= 1.0, < 3)
|
base64 (>= 0.1.0)
|
||||||
rails (7.0.2.2)
|
rack (~> 2.2, >= 2.2.4)
|
||||||
actioncable (= 7.0.2.2)
|
rack-session (1.0.2)
|
||||||
actionmailbox (= 7.0.2.2)
|
rack (< 3)
|
||||||
actionmailer (= 7.0.2.2)
|
rack-test (2.1.0)
|
||||||
actionpack (= 7.0.2.2)
|
rack (>= 1.3)
|
||||||
actiontext (= 7.0.2.2)
|
rackup (1.0.0)
|
||||||
actionview (= 7.0.2.2)
|
rack (< 3)
|
||||||
activejob (= 7.0.2.2)
|
webrick
|
||||||
activemodel (= 7.0.2.2)
|
rails (7.1.3)
|
||||||
activerecord (= 7.0.2.2)
|
actioncable (= 7.1.3)
|
||||||
activestorage (= 7.0.2.2)
|
actionmailbox (= 7.1.3)
|
||||||
activesupport (= 7.0.2.2)
|
actionmailer (= 7.1.3)
|
||||||
|
actionpack (= 7.1.3)
|
||||||
|
actiontext (= 7.1.3)
|
||||||
|
actionview (= 7.1.3)
|
||||||
|
activejob (= 7.1.3)
|
||||||
|
activemodel (= 7.1.3)
|
||||||
|
activerecord (= 7.1.3)
|
||||||
|
activestorage (= 7.1.3)
|
||||||
|
activesupport (= 7.1.3)
|
||||||
bundler (>= 1.15.0)
|
bundler (>= 1.15.0)
|
||||||
railties (= 7.0.2.2)
|
railties (= 7.1.3)
|
||||||
rails-dom-testing (2.0.3)
|
rails-controller-testing (1.0.5)
|
||||||
activesupport (>= 4.2.0)
|
actionpack (>= 5.0.1.rc1)
|
||||||
|
actionview (>= 5.0.1.rc1)
|
||||||
|
activesupport (>= 5.0.1.rc1)
|
||||||
|
rails-dom-testing (2.2.0)
|
||||||
|
activesupport (>= 5.0.0)
|
||||||
|
minitest
|
||||||
nokogiri (>= 1.6)
|
nokogiri (>= 1.6)
|
||||||
rails-html-sanitizer (1.4.2)
|
rails-html-sanitizer (1.6.0)
|
||||||
loofah (~> 2.3)
|
loofah (~> 2.21)
|
||||||
railties (7.0.2.2)
|
nokogiri (~> 1.14)
|
||||||
actionpack (= 7.0.2.2)
|
rails-settings-cached (2.8.3)
|
||||||
activesupport (= 7.0.2.2)
|
activerecord (>= 5.0.0)
|
||||||
method_source
|
railties (>= 5.0.0)
|
||||||
|
railties (7.1.3)
|
||||||
|
actionpack (= 7.1.3)
|
||||||
|
activesupport (= 7.1.3)
|
||||||
|
irb
|
||||||
|
rackup (>= 1.0.0)
|
||||||
rake (>= 12.2)
|
rake (>= 12.2)
|
||||||
thor (~> 1.0)
|
thor (~> 1.0, >= 1.2.2)
|
||||||
zeitwerk (~> 2.5)
|
zeitwerk (~> 2.6)
|
||||||
rake (13.0.6)
|
rainbow (3.1.1)
|
||||||
rb-fsevent (0.11.1)
|
rake (13.1.0)
|
||||||
|
rb-fsevent (0.11.2)
|
||||||
rb-inotify (0.10.1)
|
rb-inotify (0.10.1)
|
||||||
ffi (~> 1.0)
|
ffi (~> 1.0)
|
||||||
redis (4.6.0)
|
rbs (2.8.4)
|
||||||
regexp_parser (2.2.1)
|
rdoc (6.6.2)
|
||||||
responders (3.0.1)
|
psych (>= 4.0.0)
|
||||||
actionpack (>= 5.0)
|
redis (4.8.1)
|
||||||
railties (>= 5.0)
|
regexp_parser (2.9.0)
|
||||||
rexml (3.2.5)
|
reline (0.4.2)
|
||||||
rqrcode (2.1.1)
|
io-console (~> 0.5)
|
||||||
|
responders (3.1.1)
|
||||||
|
actionpack (>= 5.2)
|
||||||
|
railties (>= 5.2)
|
||||||
|
reverse_markdown (2.1.1)
|
||||||
|
nokogiri
|
||||||
|
rexml (3.2.6)
|
||||||
|
rqrcode (2.2.0)
|
||||||
chunky_png (~> 1.0)
|
chunky_png (~> 1.0)
|
||||||
rqrcode_core (~> 1.0)
|
rqrcode_core (~> 1.0)
|
||||||
rqrcode_core (1.2.0)
|
rqrcode_core (1.2.0)
|
||||||
rspec-core (3.11.0)
|
rspec-core (3.12.2)
|
||||||
rspec-support (~> 3.11.0)
|
rspec-support (~> 3.12.0)
|
||||||
rspec-expectations (3.11.0)
|
rspec-expectations (3.12.3)
|
||||||
diff-lcs (>= 1.2.0, < 2.0)
|
diff-lcs (>= 1.2.0, < 2.0)
|
||||||
rspec-support (~> 3.11.0)
|
rspec-support (~> 3.12.0)
|
||||||
rspec-mocks (3.11.0)
|
rspec-mocks (3.12.6)
|
||||||
diff-lcs (>= 1.2.0, < 2.0)
|
diff-lcs (>= 1.2.0, < 2.0)
|
||||||
rspec-support (~> 3.11.0)
|
rspec-support (~> 3.12.0)
|
||||||
rspec-rails (5.1.0)
|
rspec-rails (6.1.1)
|
||||||
actionpack (>= 5.2)
|
actionpack (>= 6.1)
|
||||||
activesupport (>= 5.2)
|
activesupport (>= 6.1)
|
||||||
railties (>= 5.2)
|
railties (>= 6.1)
|
||||||
rspec-core (~> 3.10)
|
rspec-core (~> 3.12)
|
||||||
rspec-expectations (~> 3.10)
|
rspec-expectations (~> 3.12)
|
||||||
rspec-mocks (~> 3.10)
|
rspec-mocks (~> 3.12)
|
||||||
rspec-support (~> 3.10)
|
rspec-support (~> 3.12)
|
||||||
rspec-support (3.11.0)
|
rspec-support (3.12.1)
|
||||||
|
rubocop (1.60.2)
|
||||||
|
json (~> 2.3)
|
||||||
|
language_server-protocol (>= 3.17.0)
|
||||||
|
parallel (~> 1.10)
|
||||||
|
parser (>= 3.3.0.2)
|
||||||
|
rainbow (>= 2.2.2, < 4.0)
|
||||||
|
regexp_parser (>= 1.8, < 3.0)
|
||||||
|
rexml (>= 3.2.5, < 4.0)
|
||||||
|
rubocop-ast (>= 1.30.0, < 2.0)
|
||||||
|
ruby-progressbar (~> 1.7)
|
||||||
|
unicode-display_width (>= 2.4.0, < 3.0)
|
||||||
|
rubocop-ast (1.30.0)
|
||||||
|
parser (>= 3.2.1.0)
|
||||||
|
ruby-progressbar (1.13.0)
|
||||||
|
ruby-vips (2.2.0)
|
||||||
|
ffi (~> 1.12)
|
||||||
ruby2_keywords (0.0.5)
|
ruby2_keywords (0.0.5)
|
||||||
rufus-scheduler (3.8.1)
|
rufus-scheduler (3.9.1)
|
||||||
fugit (~> 1.1, >= 1.1.6)
|
fugit (~> 1.1, >= 1.1.6)
|
||||||
sidekiq (6.4.1)
|
sanitize (6.1.0)
|
||||||
connection_pool (>= 2.2.2)
|
crass (~> 1.0.2)
|
||||||
|
nokogiri (>= 1.12.0)
|
||||||
|
sentry-rails (5.16.1)
|
||||||
|
railties (>= 5.0)
|
||||||
|
sentry-ruby (~> 5.16.1)
|
||||||
|
sentry-ruby (5.16.1)
|
||||||
|
concurrent-ruby (~> 1.0, >= 1.0.2)
|
||||||
|
sidekiq (6.5.12)
|
||||||
|
connection_pool (>= 2.2.5, < 3)
|
||||||
rack (~> 2.0)
|
rack (~> 2.0)
|
||||||
redis (>= 4.2.0)
|
redis (>= 4.5.0, < 5)
|
||||||
sidekiq-scheduler (3.1.1)
|
sidekiq-scheduler (5.0.3)
|
||||||
e2mmap
|
|
||||||
redis (>= 3, < 5)
|
|
||||||
rufus-scheduler (~> 3.2)
|
rufus-scheduler (~> 3.2)
|
||||||
sidekiq (>= 3)
|
sidekiq (>= 6, < 8)
|
||||||
thwait
|
|
||||||
tilt (>= 1.4.0)
|
tilt (>= 1.4.0)
|
||||||
sprockets (4.0.2)
|
solargraph (0.50.0)
|
||||||
|
backport (~> 1.2)
|
||||||
|
benchmark
|
||||||
|
bundler (~> 2.0)
|
||||||
|
diff-lcs (~> 1.4)
|
||||||
|
e2mmap
|
||||||
|
jaro_winkler (~> 1.5)
|
||||||
|
kramdown (~> 2.3)
|
||||||
|
kramdown-parser-gfm (~> 1.1)
|
||||||
|
parser (~> 3.0)
|
||||||
|
rbs (~> 2.0)
|
||||||
|
reverse_markdown (~> 2.0)
|
||||||
|
rubocop (~> 1.38)
|
||||||
|
thor (~> 1.0)
|
||||||
|
tilt (~> 2.0)
|
||||||
|
yard (~> 0.9, >= 0.9.24)
|
||||||
|
sprockets (4.2.1)
|
||||||
concurrent-ruby (~> 1.0)
|
concurrent-ruby (~> 1.0)
|
||||||
rack (> 1, < 3)
|
rack (>= 2.2.4, < 4)
|
||||||
sprockets-rails (3.4.2)
|
sprockets-rails (3.4.2)
|
||||||
actionpack (>= 5.2)
|
actionpack (>= 5.2)
|
||||||
activesupport (>= 5.2)
|
activesupport (>= 5.2)
|
||||||
sprockets (>= 3.0.0)
|
sprockets (>= 3.0.0)
|
||||||
sqlite3 (1.4.2)
|
sqlite3 (1.7.2)
|
||||||
stimulus-rails (1.0.2)
|
mini_portile2 (~> 2.8.0)
|
||||||
|
sqlite3 (1.7.2-arm64-darwin)
|
||||||
|
sqlite3 (1.7.2-x86_64-linux)
|
||||||
|
stimulus-rails (1.3.3)
|
||||||
railties (>= 6.0.0)
|
railties (>= 6.0.0)
|
||||||
strscan (3.0.1)
|
stringio (3.1.0)
|
||||||
thor (1.2.1)
|
thor (1.3.0)
|
||||||
thwait (0.2.0)
|
tilt (2.3.0)
|
||||||
e2mmap
|
timeout (0.4.1)
|
||||||
tilt (2.0.10)
|
turbo-rails (1.5.0)
|
||||||
timeout (0.2.0)
|
|
||||||
turbo-rails (1.0.1)
|
|
||||||
actionpack (>= 6.0.0)
|
actionpack (>= 6.0.0)
|
||||||
|
activejob (>= 6.0.0)
|
||||||
railties (>= 6.0.0)
|
railties (>= 6.0.0)
|
||||||
tzinfo (2.0.4)
|
tzinfo (2.0.6)
|
||||||
|
concurrent-ruby (~> 1.0)
|
||||||
|
unicode-display_width (2.5.0)
|
||||||
|
uri (0.13.0)
|
||||||
|
view_component (3.10.0)
|
||||||
|
activesupport (>= 5.2.0, < 8.0)
|
||||||
concurrent-ruby (~> 1.0)
|
concurrent-ruby (~> 1.0)
|
||||||
view_component (2.49.0)
|
|
||||||
activesupport (>= 5.0.0, < 8.0)
|
|
||||||
method_source (~> 1.0)
|
method_source (~> 1.0)
|
||||||
warden (1.2.9)
|
warden (1.2.9)
|
||||||
rack (>= 2.0.9)
|
rack (>= 2.0.9)
|
||||||
web-console (4.2.0)
|
web-console (4.2.1)
|
||||||
actionview (>= 6.0.0)
|
actionview (>= 6.0.0)
|
||||||
activemodel (>= 6.0.0)
|
activemodel (>= 6.0.0)
|
||||||
bindex (>= 0.4.0)
|
bindex (>= 0.4.0)
|
||||||
railties (>= 6.0.0)
|
railties (>= 6.0.0)
|
||||||
webmock (3.14.0)
|
webmock (3.19.1)
|
||||||
addressable (>= 2.8.0)
|
addressable (>= 2.8.0)
|
||||||
crack (>= 0.3.2)
|
crack (>= 0.3.2)
|
||||||
hashdiff (>= 0.4.0, < 2.0.0)
|
hashdiff (>= 0.4.0, < 2.0.0)
|
||||||
websocket-driver (0.7.5)
|
webrick (1.8.1)
|
||||||
|
websocket-driver (0.7.6)
|
||||||
websocket-extensions (>= 0.1.0)
|
websocket-extensions (>= 0.1.0)
|
||||||
websocket-extensions (0.1.5)
|
websocket-extensions (0.1.5)
|
||||||
xpath (3.2.0)
|
xpath (3.2.0)
|
||||||
nokogiri (~> 1.8)
|
nokogiri (~> 1.8)
|
||||||
zeitwerk (2.5.4)
|
yard (0.9.34)
|
||||||
|
zeitwerk (2.6.12)
|
||||||
|
|
||||||
PLATFORMS
|
PLATFORMS
|
||||||
|
arm64-darwin-22
|
||||||
|
ruby
|
||||||
x86_64-linux
|
x86_64-linux
|
||||||
|
|
||||||
DEPENDENCIES
|
DEPENDENCIES
|
||||||
|
aws-sdk-s3
|
||||||
|
bcrypt (~> 3.1)
|
||||||
capybara
|
capybara
|
||||||
cssbundling-rails
|
cssbundling-rails
|
||||||
database_cleaner
|
database_cleaner
|
||||||
devise
|
devise (~> 4.9.0)
|
||||||
devise_ldap_authenticatable
|
devise_ldap_authenticatable
|
||||||
|
discourse_api
|
||||||
dotenv-rails
|
dotenv-rails
|
||||||
|
down
|
||||||
factory_bot_rails
|
factory_bot_rails
|
||||||
|
faker
|
||||||
faraday
|
faraday
|
||||||
|
flipper
|
||||||
|
flipper-active_record
|
||||||
|
flipper-ui
|
||||||
|
image_processing (~> 1.12.2)
|
||||||
importmap-rails
|
importmap-rails
|
||||||
jbuilder (~> 2.7)
|
jbuilder (~> 2.7)
|
||||||
letter_opener
|
letter_opener
|
||||||
letter_opener_web
|
letter_opener_web
|
||||||
listen (~> 3.2)
|
listen (~> 3.2)
|
||||||
|
lnurl
|
||||||
lockbox
|
lockbox
|
||||||
|
manifique (~> 1.1.0)
|
||||||
net-ldap
|
net-ldap
|
||||||
pg (~> 1.2.3)
|
nostr (~> 0.6.0)
|
||||||
|
pagy (~> 6.0, >= 6.0.2)
|
||||||
|
pg (~> 1.5)
|
||||||
puma (~> 4.1)
|
puma (~> 4.1)
|
||||||
rails (~> 7.0.2)
|
rails (~> 7.1)
|
||||||
|
rails-controller-testing
|
||||||
|
rails-settings-cached (~> 2.8.3)
|
||||||
rqrcode (~> 2.0)
|
rqrcode (~> 2.0)
|
||||||
rspec-rails
|
rspec-rails
|
||||||
sidekiq
|
sentry-rails
|
||||||
|
sentry-ruby
|
||||||
|
sidekiq (< 7)
|
||||||
sidekiq-scheduler
|
sidekiq-scheduler
|
||||||
|
solargraph
|
||||||
sprockets-rails
|
sprockets-rails
|
||||||
sqlite3 (~> 1.4)
|
sqlite3 (~> 1.7.2)
|
||||||
stimulus-rails
|
stimulus-rails
|
||||||
turbo-rails
|
turbo-rails
|
||||||
tzinfo-data
|
tzinfo-data
|
||||||
view_component
|
view_component
|
||||||
warden
|
warden
|
||||||
web-console (>= 3.3.0)
|
web-console (~> 4.2)
|
||||||
webmock
|
webmock
|
||||||
|
|
||||||
BUNDLED WITH
|
BUNDLED WITH
|
||||||
2.3.7
|
2.5.5
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
web: bin/rails server -p 3000
|
web: bin/rails server -b 0.0.0.0 -p 3000
|
||||||
css: yarn build:css --watch
|
css: yarn build:css --watch
|
||||||
|
|||||||
112
README.md
112
README.md
@@ -7,43 +7,131 @@ credentials, invites, donations, etc..
|
|||||||
|
|
||||||
## Development
|
## Development
|
||||||
|
|
||||||
|
### Quick Start
|
||||||
|
|
||||||
|
The easiest way to get a working development setup is using Docker Compose like
|
||||||
|
so:
|
||||||
|
|
||||||
|
1. Make sure [Docker Compose is installed][1] and Docker is running (included in
|
||||||
|
Docker Desktop)
|
||||||
|
3. Run `docker compose up --build` and wait until all services have started
|
||||||
|
(389ds might take an extra minute to be ready). This will take a while when
|
||||||
|
running for the first time, so you might want to do something else in the
|
||||||
|
meantime.
|
||||||
|
4. `docker-compose exec ldap dsconf localhost backend create --suffix="dc=kosmos,dc=org" --be-name="dev"`
|
||||||
|
5. `docker compose run web rails ldap:setup`
|
||||||
|
6. `docker compose run web rails db:setup`
|
||||||
|
|
||||||
|
After these steps, you should have a working Rails app with a handful of test
|
||||||
|
users running on [http://localhost:3000](http://localhost:3000).
|
||||||
|
Log in with username "admin" and password "admin is admin". All users listed on
|
||||||
|
[http://localhost:3000/admin/users](http://localhost:3000/admin/users)
|
||||||
|
have the password "user is user".
|
||||||
|
|
||||||
### Rails app
|
### Rails app
|
||||||
|
|
||||||
|
_Note: when using Docker Compose, prefix the following commands with `docker-compose
|
||||||
|
run web`._
|
||||||
|
|
||||||
Installing dependencies:
|
Installing dependencies:
|
||||||
|
|
||||||
bundle install
|
bundle install
|
||||||
yarn install
|
yarn install
|
||||||
|
|
||||||
Setting up local database (SQLite):
|
Migrating the local database (after schema changes):
|
||||||
|
|
||||||
bundle exec rails db:create
|
|
||||||
bundle exec rails db:migrate
|
bundle exec rails db:migrate
|
||||||
|
|
||||||
Running the dev server and auto-building CSS files on change:
|
Running the dev server, and auto-building CSS files on change _(automatic with Docker Compose)_:
|
||||||
|
|
||||||
bin/dev
|
bin/dev
|
||||||
|
|
||||||
Running the background workers (requires Redis):
|
Running the background workers (requires Redis) _(automatic with Docker Compose)_:
|
||||||
|
|
||||||
bundle exec sidekiq -C config/sidekiq.yml
|
bundle exec sidekiq -C config/sidekiq.yml
|
||||||
|
|
||||||
Running all specs:
|
Running the test suite:
|
||||||
|
|
||||||
bundle exec rspec
|
bundle exec rspec
|
||||||
|
|
||||||
### LDAP server
|
Running the test suite with Docker Compose requires overriding the Rails
|
||||||
|
environment:
|
||||||
|
|
||||||
TODO make it easy to run a local Kosmos LDAP server for development, without
|
docker-compose run -e "RAILS_ENV=test" web rspec
|
||||||
manual LDIF imports etc. (or provide a staging instance)
|
|
||||||
|
### Docker Compose
|
||||||
|
|
||||||
|
Services/containers are configured in `docker-compose.yml`.
|
||||||
|
|
||||||
|
You can run services selectively, for example if you want to run the Rails app
|
||||||
|
and test suite on the host machine. Just add the service names of the
|
||||||
|
containers you want to run to the `up` command, like so:
|
||||||
|
|
||||||
|
docker-compose up ldap redis
|
||||||
|
|
||||||
|
#### LDAP server
|
||||||
|
|
||||||
|
After creating the Docker container for the first time (or after deleting it),
|
||||||
|
you need to run the following command once, in order to create the dirsrv
|
||||||
|
back-end:
|
||||||
|
|
||||||
|
docker-compose exec ldap dsconf localhost backend create --suffix="dc=kosmos,dc=org" --be-name="dev"
|
||||||
|
|
||||||
|
Now you can seed the back-end with data using this Rails task:
|
||||||
|
|
||||||
|
bundle exec rails ldap:setup
|
||||||
|
|
||||||
|
The setup task will first delete any existing entries in the directory tree
|
||||||
|
("dc=kosmos,dc=org"), and then create our development entries.
|
||||||
|
|
||||||
|
Note that all 389ds data is stored in the `389ds-data` volume. So if you want
|
||||||
|
to start over with a fresh installation, delete both that volume as well as the
|
||||||
|
container.
|
||||||
|
|
||||||
|
#### Minio / remoteStorage
|
||||||
|
|
||||||
|
If you want to run remoteStorage accounts locally, you will have to create the
|
||||||
|
respective bucket first. With the `minio` container running (run by default
|
||||||
|
when using Docker Compose), follow these steps:
|
||||||
|
|
||||||
|
* `docker compose up web redis minio liquor-cabinet`
|
||||||
|
* Head to http://localhost:9001 and log in with user `minioadmin`, password
|
||||||
|
`minioadmin`
|
||||||
|
* Create a new bucket called `remotestorage` (or whatever you
|
||||||
|
change the `S3_BUCKET` config to)
|
||||||
|
* Create a new key with ID "dev-key" and secret "123456789" (or whatever you
|
||||||
|
change `S3_ACCESS_KEY` and `S3_SECRET_KEY` to). Leave the policy field empty,
|
||||||
|
as it will automatically allow access to the bucket you created.
|
||||||
|
|
||||||
|
### Adding npm modules to use with Stimulus controllers
|
||||||
|
|
||||||
|
The following command downloads the specified npm module to `vendor/javascript`
|
||||||
|
and adds an entry for it to `config/importmap.rb`.
|
||||||
|
|
||||||
|
bin/importmap pin bech32 --download
|
||||||
|
|
||||||
|
### Solargraph
|
||||||
|
|
||||||
|
[Solargraph](https://solargraph.org/) is a Ruby language server, which you may
|
||||||
|
use with your editor to add features like auto-completion and syntax
|
||||||
|
validation. You can add inline documentation for bundled gems with this
|
||||||
|
command:
|
||||||
|
|
||||||
|
bundle exec yard gems
|
||||||
|
|
||||||
## Documentation
|
## Documentation
|
||||||
|
|
||||||
|
### Rails
|
||||||
|
|
||||||
* [Ruby on Rails](https://guides.rubyonrails.org/)
|
* [Ruby on Rails](https://guides.rubyonrails.org/)
|
||||||
* [Sass](https://sass-lang.com/documentation)
|
* [Pagination](https://ddnexus.github.io/pagy/)
|
||||||
|
|
||||||
### Front-end
|
### Front-end
|
||||||
|
|
||||||
* [Tailwind CSS](https://tailwindcss.com/)
|
* [Tailwind CSS](https://tailwindcss.com/)
|
||||||
|
* [Sass](https://sass-lang.com/documentation)
|
||||||
|
* [Stimulus](https://stimulus.hotwired.dev/handbook/)
|
||||||
|
* [Tailwind Stimulus Components](https://github.com/excid3/tailwindcss-stimulus-components)
|
||||||
|
|
||||||
### Testing
|
### Testing
|
||||||
|
|
||||||
@@ -60,6 +148,12 @@ manual LDIF imports etc. (or provide a staging instance)
|
|||||||
* [Sidekiq](https://github.com/mperham/sidekiq/wiki/)
|
* [Sidekiq](https://github.com/mperham/sidekiq/wiki/)
|
||||||
* [ActiveJob](https://github.com/mperham/sidekiq/wiki/Active-Job)
|
* [ActiveJob](https://github.com/mperham/sidekiq/wiki/Active-Job)
|
||||||
|
|
||||||
|
### Feature Flags
|
||||||
|
|
||||||
|
* [Flipper](https://www.flippercloud.io/docs/get-started/self-hosted)
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
[GNU Affero General Public License v3.0](https://choosealicense.com/licenses/agpl-3.0/)
|
[GNU Affero General Public License v3.0](https://choosealicense.com/licenses/agpl-3.0/)
|
||||||
|
|
||||||
|
[1]: https://docs.docker.com/compose/install/
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
//= link_tree ../images
|
//= link_tree ../images
|
||||||
//= link_tree ../../javascript .js
|
//= link_tree ../../javascript .js
|
||||||
//= link_tree ../builds
|
//= link_tree ../builds
|
||||||
|
//= link_tree ../../../vendor/javascript .js
|
||||||
|
|||||||
@@ -2,8 +2,12 @@
|
|||||||
@import "tailwindcss/components";
|
@import "tailwindcss/components";
|
||||||
@import "tailwindcss/utilities";
|
@import "tailwindcss/utilities";
|
||||||
|
|
||||||
|
@import "components/animations";
|
||||||
@import "components/base";
|
@import "components/base";
|
||||||
@import "components/buttons";
|
@import "components/buttons";
|
||||||
|
@import "components/dashboard_services";
|
||||||
@import "components/forms";
|
@import "components/forms";
|
||||||
@import "components/links";
|
@import "components/links";
|
||||||
@import "components/notifications";
|
@import "components/notifications";
|
||||||
|
@import "components/pagination";
|
||||||
|
@import "components/tables";
|
||||||
|
|||||||
16
app/assets/stylesheets/components/animations.css
Normal file
16
app/assets/stylesheets/components/animations.css
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
@keyframes scaleIn {
|
||||||
|
from {
|
||||||
|
transform: scale(0.5);
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
transform: scale(1);
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.animate-scale-in {
|
||||||
|
animation-name: scaleIn;
|
||||||
|
animation-duration: 0.15s;
|
||||||
|
animation-timing-function: cubic-bezier(0.2, 0, 0.13, 1);
|
||||||
|
}
|
||||||
@@ -1,11 +1,16 @@
|
|||||||
@layer base {
|
@layer base {
|
||||||
body {
|
html {
|
||||||
@apply leading-none
|
font-size: 14px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* h1, h2, h3 { */
|
body {
|
||||||
/* @apply font-light; */
|
@apply leading-none bg-cover bg-fixed;
|
||||||
/* } */
|
background-image: linear-gradient(35deg, rgba(255,0,255,0.2) 0, rgba(13,79,153,0.8) 100%), url('/img/bg-1.jpg');
|
||||||
|
}
|
||||||
|
|
||||||
|
body#admin {
|
||||||
|
background-image: linear-gradient(35deg, rgba(255,0,255,0.2) 0, rgba(153,12,14,0.9) 100%), url('/img/bg-1.jpg');
|
||||||
|
}
|
||||||
|
|
||||||
h1 {
|
h1 {
|
||||||
@apply text-3xl uppercase;
|
@apply text-3xl uppercase;
|
||||||
@@ -19,6 +24,10 @@
|
|||||||
@apply text-xl mb-6;
|
@apply text-xl mb-6;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
h4 {
|
||||||
|
@apply font-bold mb-4 leading-6;
|
||||||
|
}
|
||||||
|
|
||||||
main section {
|
main section {
|
||||||
@apply pt-8 sm:pt-12;
|
@apply pt-8 sm:pt-12;
|
||||||
}
|
}
|
||||||
@@ -26,4 +35,24 @@
|
|||||||
main section:first-of-type {
|
main section:first-of-type {
|
||||||
@apply pt-0;
|
@apply pt-0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
main p {
|
||||||
|
@apply mb-4 leading-6;
|
||||||
|
}
|
||||||
|
|
||||||
|
main p:last-child {
|
||||||
|
@apply mb-0;
|
||||||
|
}
|
||||||
|
|
||||||
|
main ul {
|
||||||
|
@apply mb-6;
|
||||||
|
}
|
||||||
|
|
||||||
|
main ul:last-child {
|
||||||
|
@apply mb-0;
|
||||||
|
}
|
||||||
|
|
||||||
|
main ul li {
|
||||||
|
@apply leading-6;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,17 +1,25 @@
|
|||||||
@layer components {
|
@layer components {
|
||||||
.btn {
|
.btn {
|
||||||
@apply font-semibold rounded-md leading-none cursor-pointer text-center
|
@apply inline-block font-semibold rounded-md leading-none cursor-pointer text-center
|
||||||
transition-colors duration-75 focus:outline-none focus:ring-4;
|
transition-colors duration-75 focus:outline-none focus:ring-4;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn-md {
|
.btn-md {
|
||||||
@apply btn;
|
@apply btn;
|
||||||
@apply py-2.5 px-5 shadow-md;
|
@apply py-3 px-6;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn-sm {
|
.btn-sm {
|
||||||
@apply btn;
|
@apply btn;
|
||||||
@apply py-1 px-2 text-sm shadow-sm;
|
@apply py-1 px-2 text-sm;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-icon {
|
||||||
|
@apply py-2 px-3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-outline {
|
||||||
|
@apply py-2 border-2 border-gray-100 hover:bg-gray-100;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn-gray {
|
.btn-gray {
|
||||||
@@ -24,8 +32,23 @@
|
|||||||
focus:ring-blue-400 focus:ring-opacity-75;
|
focus:ring-blue-400 focus:ring-opacity-75;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.btn-emerald {
|
||||||
|
@apply bg-emerald-500 hover:bg-emerald-600 text-white
|
||||||
|
focus:ring-emerald-400 focus:ring-opacity-75;
|
||||||
|
}
|
||||||
|
|
||||||
.btn-red {
|
.btn-red {
|
||||||
@apply bg-red-600 hover:bg-red-700 text-white
|
@apply bg-red-600 hover:bg-red-700 text-white
|
||||||
focus:ring-red-500 focus:ring-opacity-75;
|
focus:ring-red-500 focus:ring-opacity-75;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.btn-outline-purple {
|
||||||
|
@apply border-2 border-purple-500 hover:bg-purple-100
|
||||||
|
focus:ring-purple-400 focus:ring-opacity-75;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn:disabled {
|
||||||
|
@apply bg-gray-100 hover:bg-gray-200 text-gray-400
|
||||||
|
focus:ring-gray-300 focus:ring-opacity-75;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
5
app/assets/stylesheets/components/dashboard_services.css
Normal file
5
app/assets/stylesheets/components/dashboard_services.css
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
@layer components {
|
||||||
|
.services > div > a {
|
||||||
|
background-image: linear-gradient(110deg, rgba(255,255,255,0.99) 20%, rgba(255,255,255,0.88) 100%);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,11 +1,20 @@
|
|||||||
@layer components {
|
@layer components {
|
||||||
input[type=text], input[type=email], input[type=password],
|
input[type=text], input[type=email], input[type=password],
|
||||||
input[type=number], select {
|
input[type=number], select, textarea {
|
||||||
@apply mt-1 rounded-md bg-gray-100 focus:bg-white
|
@apply rounded-md bg-gray-100 focus:bg-white
|
||||||
border-transparent focus:border-transparent focus:ring-2
|
border-transparent focus:border-transparent focus:ring-2
|
||||||
focus:ring-blue-600 focus:ring-opacity-75;
|
focus:ring-blue-600 focus:ring-opacity-75;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
input[type=text]:disabled,
|
||||||
|
input[type=email]:disabled {
|
||||||
|
@apply text-gray-700;
|
||||||
|
}
|
||||||
|
|
||||||
|
input.field_with_errors {
|
||||||
|
@apply border-b-red-600;
|
||||||
|
}
|
||||||
|
|
||||||
.field_with_errors {
|
.field_with_errors {
|
||||||
@apply inline-block;
|
@apply inline-block;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,10 +5,4 @@
|
|||||||
&:visited { @apply text-indigo-600; }
|
&:visited { @apply text-indigo-600; }
|
||||||
&:active { @apply text-red-600; }
|
&:active { @apply text-red-600; }
|
||||||
}
|
}
|
||||||
|
|
||||||
.devise-links {
|
|
||||||
a {
|
|
||||||
@apply ks-text-link;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
45
app/assets/stylesheets/components/pagination.css
Normal file
45
app/assets/stylesheets/components/pagination.css
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
@layer components {
|
||||||
|
.pagy-nav.pagination {
|
||||||
|
@apply isolate inline-flex -space-x-px rounded-md shadow-sm;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagy-nav .page:not(.prev):not(.next) {
|
||||||
|
@apply hidden sm:inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagy-nav .page.next a {
|
||||||
|
@apply relative inline-flex items-center rounded-r-md border
|
||||||
|
border-gray-300 bg-white px-3 py-2 text-sm font-medium
|
||||||
|
text-gray-500 hover:bg-gray-100 focus:z-20;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagy-nav .page.prev a {
|
||||||
|
@apply relative inline-flex items-center rounded-l-md border
|
||||||
|
border-gray-300 bg-white px-3 py-2 text-sm font-medium
|
||||||
|
text-gray-500 hover:bg-gray-100 focus:z-20;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagy-nav .page.next.disabled {
|
||||||
|
@apply relative inline-flex items-center rounded-r-md border
|
||||||
|
border-gray-300 bg-gray-100 px-3 py-2 text-sm font-medium
|
||||||
|
text-gray-400 focus:z-20;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagy-nav .page.prev.disabled {
|
||||||
|
@apply relative inline-flex items-center rounded-l-md border
|
||||||
|
border-gray-300 bg-gray-100 px-3 py-2 text-sm font-medium
|
||||||
|
text-gray-400 focus:z-20;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagy-nav .page a, .page.gap {
|
||||||
|
@apply bg-white border-gray-300 text-gray-500 hover:bg-gray-100 relative
|
||||||
|
inline-flex items-center border px-4 py-2 text-sm font-medium
|
||||||
|
focus:z-20;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagy-nav .page.active {
|
||||||
|
@apply z-10 border-indigo-500 bg-indigo-50 text-indigo-600 relative
|
||||||
|
inline-flex items-center border px-4 py-2 text-sm font-medium
|
||||||
|
focus:z-20;
|
||||||
|
}
|
||||||
|
}
|
||||||
36
app/assets/stylesheets/components/tables.css
Normal file
36
app/assets/stylesheets/components/tables.css
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
@layer components {
|
||||||
|
table {
|
||||||
|
@apply w-full;
|
||||||
|
}
|
||||||
|
|
||||||
|
table thead tr {
|
||||||
|
@apply text-left;
|
||||||
|
}
|
||||||
|
|
||||||
|
table thead th {
|
||||||
|
@apply pb-3.5 text-sm font-normal uppercase text-gray-500;
|
||||||
|
}
|
||||||
|
|
||||||
|
table tbody th {
|
||||||
|
@apply text-left font-normal text-gray-500;
|
||||||
|
}
|
||||||
|
|
||||||
|
table th:not(:last-of-type),
|
||||||
|
table td:not(:last-of-type) {
|
||||||
|
@apply pr-2;
|
||||||
|
}
|
||||||
|
|
||||||
|
table td, tbody th {
|
||||||
|
@apply py-2;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.divided {
|
||||||
|
@apply divide-y divide-gray-300;
|
||||||
|
}
|
||||||
|
table.divided tbody {
|
||||||
|
@apply divide-y divide-gray-200;
|
||||||
|
}
|
||||||
|
table.divided td, table.divided tbody th {
|
||||||
|
@apply py-3;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
@import "legacy/layout";
|
|
||||||
@import "legacy/main_nav";
|
|
||||||
@@ -1,118 +0,0 @@
|
|||||||
@import "variables";
|
|
||||||
@import "mediaqueries";
|
|
||||||
|
|
||||||
body {
|
|
||||||
background: linear-gradient(35deg, rgba(255,0,255,0.2) 0, rgba(13,79,153,0.8) 100%),
|
|
||||||
url('/img/bg-1.jpg');
|
|
||||||
background-size: cover;
|
|
||||||
background-attachment: fixed;
|
|
||||||
}
|
|
||||||
|
|
||||||
body#admin {
|
|
||||||
background: linear-gradient(35deg, rgba(255,0,255,0.2) 0, rgba(153,12,14,0.9) 100%),
|
|
||||||
url('/img/bg-1.jpg');
|
|
||||||
background-size: cover;
|
|
||||||
background-attachment: fixed;
|
|
||||||
}
|
|
||||||
|
|
||||||
.ks-site-icon {
|
|
||||||
svg {
|
|
||||||
display: inline-block;
|
|
||||||
height: 1.875rem;
|
|
||||||
vertical-align: top;
|
|
||||||
width: auto;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#wrapper {
|
|
||||||
width: 100%;
|
|
||||||
text-align: center;
|
|
||||||
|
|
||||||
> header {
|
|
||||||
margin: 0 auto;
|
|
||||||
padding: 4rem 0;
|
|
||||||
text-align: center;
|
|
||||||
background: linear-gradient(35deg, rgba(255,0,255,0.2) 0, rgba(13,79,153,0.8) 100%),
|
|
||||||
url('/img/bg-1.jpg');
|
|
||||||
background-size: cover;
|
|
||||||
|
|
||||||
@include media-max(small) {
|
|
||||||
padding: 3rem 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
h1 {
|
|
||||||
color: #fff;
|
|
||||||
|
|
||||||
span.project-name {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
p.current-user {
|
|
||||||
color: rgba(255,255,255,0.6);
|
|
||||||
|
|
||||||
@include media-max(small) {
|
|
||||||
font-size: 0.85rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
a {
|
|
||||||
color: rgba(255,255,255,0.6);
|
|
||||||
transition: color 0.1s linear;
|
|
||||||
|
|
||||||
&:hover, &:active {
|
|
||||||
color: #fff;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
main {
|
|
||||||
p {
|
|
||||||
line-height: 1.5rem;
|
|
||||||
margin-bottom: 1rem;
|
|
||||||
|
|
||||||
&.notice {
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ul {
|
|
||||||
margin-bottom: 1.5rem;
|
|
||||||
|
|
||||||
li {
|
|
||||||
line-height: 1.5rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
table {
|
|
||||||
width: 100%;
|
|
||||||
|
|
||||||
th {
|
|
||||||
color: $text-color-discreet;
|
|
||||||
font-weight: normal;
|
|
||||||
text-transform: uppercase;
|
|
||||||
font-size: 0.85rem;
|
|
||||||
padding-bottom: 0.825rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
td {
|
|
||||||
padding-top: 0.5rem;
|
|
||||||
padding-bottom: 0.5rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.grid {
|
|
||||||
display: grid;
|
|
||||||
|
|
||||||
&.services {
|
|
||||||
grid-template-columns: 1fr 1fr 1fr;
|
|
||||||
grid-row-gap: 1rem;
|
|
||||||
grid-column-gap: 2rem;
|
|
||||||
|
|
||||||
@include media-max(small) {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,54 +0,0 @@
|
|||||||
@import "variables";
|
|
||||||
@import "mediaqueries";
|
|
||||||
|
|
||||||
#main-nav {
|
|
||||||
width: 100%;
|
|
||||||
text-align: center;
|
|
||||||
background-color: #efefef;
|
|
||||||
|
|
||||||
.wrapper {
|
|
||||||
width: $content-width;
|
|
||||||
max-width: $content-max-width;
|
|
||||||
margin: 0 auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
ul {
|
|
||||||
@include media-max(large) {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr 1fr 1fr 1fr;
|
|
||||||
}
|
|
||||||
|
|
||||||
li {
|
|
||||||
@include media-min(large) {
|
|
||||||
display: inline;
|
|
||||||
}
|
|
||||||
|
|
||||||
@include media-max(large) {
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
|
|
||||||
a {
|
|
||||||
display: inline-block;
|
|
||||||
padding: 1.5rem 2rem;
|
|
||||||
text-decoration: none;
|
|
||||||
color: $text-color-discreet;
|
|
||||||
|
|
||||||
@include media-max(large) {
|
|
||||||
display: block;
|
|
||||||
text-align: center;
|
|
||||||
padding-left: 0;
|
|
||||||
padding-right: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@include media-max(small) {
|
|
||||||
font-size: 0.85rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.active {
|
|
||||||
color: $text-color-body;
|
|
||||||
border-bottom: 2px solid #4ea2df;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
$breakpoints-max: (
|
|
||||||
small: 600px,
|
|
||||||
medium: 960px,
|
|
||||||
large: 1280px
|
|
||||||
);
|
|
||||||
|
|
||||||
$breakpoints-min: (
|
|
||||||
small: 601px,
|
|
||||||
medium: 961px,
|
|
||||||
large: 1281px
|
|
||||||
);
|
|
||||||
|
|
||||||
@mixin media-max($screen-size) {
|
|
||||||
@if map-has-key($breakpoints-max, $screen-size) {
|
|
||||||
@media (max-width: map-get($breakpoints-max, $screen-size)) {
|
|
||||||
@content;
|
|
||||||
}
|
|
||||||
} @else {
|
|
||||||
// Debugging
|
|
||||||
@warn "'#{$screen-size}' has not been declared as a breakpoint."
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@mixin media-min($screen-size) {
|
|
||||||
@if map-has-key($breakpoints-min, $screen-size) {
|
|
||||||
@media (min-width: map-get($breakpoints-min, $screen-size)) {
|
|
||||||
@content;
|
|
||||||
}
|
|
||||||
} @else {
|
|
||||||
// Debugging
|
|
||||||
@warn "'#{$screen-size}' has not been declared as a breakpoint."
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
$content-width: 800px;
|
|
||||||
$content-max-width: 100%;
|
|
||||||
|
|
||||||
$text-color-body: #222;
|
|
||||||
$text-color-discreet: #888;
|
|
||||||
|
|
||||||
$background-color-notice: #efffc4;
|
|
||||||
$background-color-alert: #fff4c2;
|
|
||||||
|
|
||||||
$color-blue: #0d4f99;
|
|
||||||
$color-purple: #8955a0;
|
|
||||||
$color-red-bright: #c00;
|
|
||||||
$color-red-dark: #990c0e;
|
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
<% if @image_url %>
|
||||||
|
<%= image_tag @image_url, class: "h-full w-full" %>
|
||||||
|
<% else %>
|
||||||
|
<%= render partial: "icons/remotestorage", locals: { custom_class: "h-full w-full p-0.5 text-gray-200" } %>
|
||||||
|
<% end %>
|
||||||
21
app/components/app_catalog/web_app_icon_component.rb
Normal file
21
app/components/app_catalog/web_app_icon_component.rb
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module AppCatalog
|
||||||
|
class WebAppIconComponent < ViewComponent::Base
|
||||||
|
def initialize(web_app:)
|
||||||
|
if web_app&.icon&.attached?
|
||||||
|
@image_url = image_url_for(web_app.icon)
|
||||||
|
elsif web_app&.apple_touch_icon&.attached?
|
||||||
|
@image_url = image_url_for(web_app.apple_touch_icon)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def image_url_for(attachment)
|
||||||
|
if Setting.s3_enabled?
|
||||||
|
s3_image_url(attachment)
|
||||||
|
else
|
||||||
|
Rails.application.routes.url_helpers.rails_blob_path(attachment, only_path: true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
15
app/components/app_info_component.html.erb
Normal file
15
app/components/app_info_component.html.erb
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<div class="flex">
|
||||||
|
<div class="<%= @icon_container_class %>">
|
||||||
|
<%= image_tag(@icon_path, class: 'h-full w-full') %>
|
||||||
|
</div>
|
||||||
|
<div class="flex-1 px-4">
|
||||||
|
<h4 class="sm:pt-2 mb-2 text-lg font-bold"><%= @name %></h4>
|
||||||
|
<p class="leading-snug"><%= @description %></p>
|
||||||
|
<p class="leading-snug flex flex-wrap gap-3">
|
||||||
|
<% @links.each do |link| %>
|
||||||
|
<a href="<%= link[1] %>" target="_blank"
|
||||||
|
class="flex-0 btn-sm btn-gray"><%= link[0] %></a>
|
||||||
|
<% end %>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
19
app/components/app_info_component.rb
Normal file
19
app/components/app_info_component.rb
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class AppInfoComponent < ViewComponent::Base
|
||||||
|
def initialize(name:, description:, icon_path: , icon_fill_box: false, links: [])
|
||||||
|
@name = name
|
||||||
|
@description = description
|
||||||
|
@icon_path = icon_path
|
||||||
|
@icon_container_class = icon_container_class(icon_fill_box)
|
||||||
|
@links = links
|
||||||
|
end
|
||||||
|
|
||||||
|
def icon_container_class(icon_fill_box)
|
||||||
|
str = "flex-0 h-16 w-16 sm:h-28 sm:w-28 bg-white rounded-3xl overflow-hidden"
|
||||||
|
unless icon_fill_box
|
||||||
|
str += " p-2 border border-gray-200"
|
||||||
|
end
|
||||||
|
str
|
||||||
|
end
|
||||||
|
end
|
||||||
34
app/components/dropdown_component.html.erb
Normal file
34
app/components/dropdown_component.html.erb
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<div data-controller="dropdown" data-action="click->dropdown#toggle click@window->dropdown#hide">
|
||||||
|
<div class="relative inline-block">
|
||||||
|
<div role="button" tabindex="0" data-dropdown-target="button"
|
||||||
|
class="inline-block select-none">
|
||||||
|
<% if @size == :large %>
|
||||||
|
<span class="appearance-none flex items-center inline-block">
|
||||||
|
<span class="p-2 bg-gray-50 hover:bg-gray-100 rounded-full">
|
||||||
|
<%= render partial: "icons/#{@icon_name}",
|
||||||
|
locals: { custom_class: "inline text-gray-500 h-6 w-6" } %>
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
<% elsif @size == :small %>
|
||||||
|
<span class="appearance-none flex items-center inline-block">
|
||||||
|
<span class="text-gray-500 hover:text-blue-600">
|
||||||
|
<%= render partial: "icons/#{@icon_name}",
|
||||||
|
locals: { custom_class: "inline h-4 w-4" } %>
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
<div data-dropdown-target="menu"
|
||||||
|
data-transition-enter="transition ease-out duration-200"
|
||||||
|
data-transition-enter-from="opacity-0 translate-y-1"
|
||||||
|
data-transition-enter-to="opacity-100 translate-y-0"
|
||||||
|
data-transition-leave="transition ease-in duration-150"
|
||||||
|
data-transition-leave-from="opacity-100 translate-y-0"
|
||||||
|
data-transition-leave-to="opacity-0 translate-y-1"
|
||||||
|
class="hidden absolute top-4 right-0 z-10 mt-5 flex w-screen max-w-max">
|
||||||
|
<div class="bg-white shadow-lg rounded border overflow-hidden w-auto">
|
||||||
|
<%= content %>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
8
app/components/dropdown_component.rb
Normal file
8
app/components/dropdown_component.rb
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class DropdownComponent < ViewComponent::Base
|
||||||
|
def initialize(size: :large, icon_name: "kebap-menu")
|
||||||
|
@size = size.to_sym
|
||||||
|
@icon_name = icon_name
|
||||||
|
end
|
||||||
|
end
|
||||||
6
app/components/dropdown_link_component.html.erb
Normal file
6
app/components/dropdown_link_component.html.erb
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<%= link_to @href, class: @class, data: {
|
||||||
|
'dropdown-target': "menuItem",
|
||||||
|
'action': "keydown.up->dropdown#previousItem:prevent keydown.down->dropdown#nextItem:prevent"
|
||||||
|
} do %>
|
||||||
|
<%= content %>
|
||||||
|
<% end %>
|
||||||
18
app/components/dropdown_link_component.rb
Normal file
18
app/components/dropdown_link_component.rb
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class DropdownLinkComponent < ViewComponent::Base
|
||||||
|
def initialize(href:, separator: false, add_class: nil)
|
||||||
|
@href = href
|
||||||
|
@class = class_str(separator, add_class)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def class_str(separator, add_class)
|
||||||
|
str = "no-underline block px-5 py-3 text-sm text-gray-900 bg-white
|
||||||
|
hover:bg-gray-100 focus:bg-gray-100 whitespace-no-wrap"
|
||||||
|
str = "#{str} border-t" if separator
|
||||||
|
str = "#{str} #{add_class}" if add_class
|
||||||
|
str
|
||||||
|
end
|
||||||
|
end
|
||||||
45
app/components/form_elements/fieldset_component.html.erb
Normal file
45
app/components/form_elements/fieldset_component.html.erb
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
<%= tag.public_send(@tag, class: "mb-6 last:mb-0", data: {
|
||||||
|
:'field-name' => @field_name
|
||||||
|
}) do %>
|
||||||
|
<% if @positioning == :vertical %>
|
||||||
|
<label class="block">
|
||||||
|
<p class="font-bold <%= @descripton.present? ? "mb-1" : "mb-2" %>">
|
||||||
|
<%= @title %>
|
||||||
|
</p>
|
||||||
|
<% if @descripton.present? %>
|
||||||
|
<p class="text-gray-500">
|
||||||
|
<%= @descripton %>
|
||||||
|
</p>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<%= tag.p class: "flex gap-x-1", data: {
|
||||||
|
controller: @resettable ? "settings--resettable-field" : nil,
|
||||||
|
} do %>
|
||||||
|
<%= content %>
|
||||||
|
<% if @resettable %>
|
||||||
|
<button type="button"
|
||||||
|
class="relative grow-0 shrink-0 btn-md btn-outline text-red-700"
|
||||||
|
title="Reset to default value"
|
||||||
|
data-settings--resettable-field-target="resetButton"
|
||||||
|
data-action="settings--resettable-field#resetField">
|
||||||
|
Reset
|
||||||
|
</button>
|
||||||
|
<% end %>
|
||||||
|
<% end %>
|
||||||
|
</label>
|
||||||
|
<% elsif @positioning == :horizontal %>
|
||||||
|
<label class="block flex items-center justify-between">
|
||||||
|
<div class="flex flex-col">
|
||||||
|
<label class="font-bold mb-1"><%= @title %></label>
|
||||||
|
<% if @descripton.present? %>
|
||||||
|
<p class="text-gray-500"><%= @descripton %></p>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
<div class="relative ml-4 inline-flex flex-shrink-0">
|
||||||
|
<%= content %>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
<% else %>
|
||||||
|
<p>Invalid <code>positioning<code> argument for <code>FieldsetComponent</code>.</p>
|
||||||
|
<% end %>
|
||||||
|
<% end %>
|
||||||
16
app/components/form_elements/fieldset_component.rb
Normal file
16
app/components/form_elements/fieldset_component.rb
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module FormElements
|
||||||
|
class FieldsetComponent < ViewComponent::Base
|
||||||
|
def initialize(tag: "li", positioning: :vertical,
|
||||||
|
title:, description: nil,
|
||||||
|
field_name: nil, resettable: false)
|
||||||
|
@tag = tag
|
||||||
|
@positioning = positioning
|
||||||
|
@title = title
|
||||||
|
@descripton = description
|
||||||
|
@field_name = field_name
|
||||||
|
@resettable = resettable
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
<%= render FormElements::FieldsetComponent.new(
|
||||||
|
title: @title,
|
||||||
|
description: @description,
|
||||||
|
field_name: "setting_#{@key.to_s}",
|
||||||
|
resettable: @resettable
|
||||||
|
) do %>
|
||||||
|
<%= method("#{@type}_field").call :setting, @key,
|
||||||
|
value: Setting.public_send(@key),
|
||||||
|
placeholder: @placeholder,
|
||||||
|
data: {
|
||||||
|
:'default-value' => Setting.get_field(@key)[:default]
|
||||||
|
},
|
||||||
|
class: "w-full" %>
|
||||||
|
<% end %>
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module FormElements
|
||||||
|
class FieldsetResettableSettingComponent < ViewComponent::Base
|
||||||
|
def initialize(tag: "li", key:, type: :text, title:, description: nil, placeholder: nil)
|
||||||
|
@tag = tag
|
||||||
|
@positioning = :vertical
|
||||||
|
@title = title
|
||||||
|
@description = description
|
||||||
|
@key = key.to_sym
|
||||||
|
@type = type
|
||||||
|
@resettable = is_resettable?(@key)
|
||||||
|
@placeholder = placeholder
|
||||||
|
end
|
||||||
|
|
||||||
|
def is_resettable?(key)
|
||||||
|
default_value = Setting.get_field(key)[:default]
|
||||||
|
default_value.present? && (default_value != Setting.send(key))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
<%= tag.public_send @tag, class: "flex items-center justify-between mb-6 last:mb-0",
|
||||||
|
data: @form_enabled ? {
|
||||||
|
controller: "settings--toggle",
|
||||||
|
:'settings--toggle-switch-enabled-value' => @enabled.to_s
|
||||||
|
} : nil do %>
|
||||||
|
<div class="flex flex-col">
|
||||||
|
<label class="font-bold mb-1"><%= @title %></label>
|
||||||
|
<% if @description.present? %>
|
||||||
|
<p class="text-gray-500"><%= @description %></p>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
<div class="relative ml-4 inline-flex flex-shrink-0">
|
||||||
|
<%= render FormElements::ToggleComponent.new(
|
||||||
|
enabled: @enabled,
|
||||||
|
input_enabled: @input_enabled,
|
||||||
|
class_names: @form_enabled ? "hidden" : nil,
|
||||||
|
data: {
|
||||||
|
:'settings--toggle-target' => "button",
|
||||||
|
action: "settings--toggle#toggleSwitch"
|
||||||
|
}) %>
|
||||||
|
<% if @form_enabled %>
|
||||||
|
<% if @attribute.present? %>
|
||||||
|
<%= @form.check_box @attribute, {
|
||||||
|
checked: @enabled,
|
||||||
|
data: { :'settings--toggle-target' => "checkbox" }
|
||||||
|
}, "true", "false" %>
|
||||||
|
<% else %>
|
||||||
|
<input name="<%= @field_name %>" type="hidden" value="false" autocomplete="off">
|
||||||
|
<%= check_box_tag @field_name, "true", @enabled, {
|
||||||
|
data: { :'settings--toggle-target' => "checkbox" }
|
||||||
|
} %>
|
||||||
|
<% end %>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
19
app/components/form_elements/fieldset_toggle_component.rb
Normal file
19
app/components/form_elements/fieldset_toggle_component.rb
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module FormElements
|
||||||
|
class FieldsetToggleComponent < ViewComponent::Base
|
||||||
|
def initialize(tag: "li", form: nil, attribute: nil, field_name: nil,
|
||||||
|
enabled: false, input_enabled: true, title:, description: nil)
|
||||||
|
@tag = tag
|
||||||
|
@form = form
|
||||||
|
@attribute = attribute
|
||||||
|
@field_name = field_name
|
||||||
|
@form_enabled = @form.present? || @field_name.present?
|
||||||
|
@enabled = enabled
|
||||||
|
@input_enabled = input_enabled
|
||||||
|
@title = title
|
||||||
|
@description = description
|
||||||
|
@button_text = @enabled ? "Switch off" : "Switch on"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
15
app/components/form_elements/toggle_component.html.erb
Normal file
15
app/components/form_elements/toggle_component.html.erb
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<%= button_tag type: "button", name: "toggle", data: @data,
|
||||||
|
role: "switch", aria: { checked: @enabled.to_s },
|
||||||
|
tabindex: @tabindex, disabled: !@input_enabled,
|
||||||
|
class: "#{ @enabled ? 'bg-blue-600' : 'bg-gray-200' }
|
||||||
|
#{ @class_names.present? ? @class_names : '' }
|
||||||
|
relative inline-flex h-6 w-11 flex-shrink-0 cursor-pointer
|
||||||
|
rounded-full border-2 border-transparent transition-colors
|
||||||
|
duration-200 ease-in-out focus:outline-none focus:ring-2
|
||||||
|
focus:ring-blue-600 focus:ring-offset-2" do %>
|
||||||
|
<span class="sr-only"><%= @button_text %></span>
|
||||||
|
<span aria-hidden="true" data-settings--toggle-target="switch"
|
||||||
|
class="<%= @enabled ? 'translate-x-5' : 'translate-x-0' %>
|
||||||
|
pointer-events-none inline-block h-5 w-5 transform rounded-full
|
||||||
|
bg-white shadow ring-0 transition duration-200 ease-in-out"></span>
|
||||||
|
<% end %>
|
||||||
13
app/components/form_elements/toggle_component.rb
Normal file
13
app/components/form_elements/toggle_component.rb
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module FormElements
|
||||||
|
class ToggleComponent < ViewComponent::Base
|
||||||
|
def initialize(enabled:, input_enabled: true, data: nil, class_names: nil, tabindex: nil)
|
||||||
|
@enabled = !!enabled
|
||||||
|
@input_enabled = input_enabled
|
||||||
|
@data = data
|
||||||
|
@class_names = class_names
|
||||||
|
@tabindex = tabindex
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
<main class="w-full max-w-6xl mx-auto pb-12 px-4 md:px-6 lg:px-8">
|
<main class="w-full max-w-6xl mx-auto pb-12 px-4 md:px-6 lg:px-8">
|
||||||
<div class="bg-white rounded-lg shadow px-6 sm:px-12 py-8 sm:py-12">
|
<div class="md:min-h-[50vh] bg-white rounded-lg shadow px-6 sm:px-12 py-8 sm:py-12">
|
||||||
<%= content %>
|
<%= content %>
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<main class="w-full max-w-6xl mx-auto pb-12 px-4 md:px-6 lg:px-8">
|
<main class="w-full max-w-6xl mx-auto pb-12 px-4 md:px-6 lg:px-8">
|
||||||
<div class="bg-white rounded-lg shadow">
|
<div class="bg-white rounded-lg shadow">
|
||||||
<div class="divide-y divide-gray-200 lg:grid lg:grid-cols-12 lg:divide-y-0 lg:divide-x">
|
<div class="md:min-h-[50vh] divide-y divide-gray-200 lg:grid lg:grid-cols-12 lg:divide-y-0 lg:divide-x">
|
||||||
<aside class="py-6 sm:py-8 lg:col-span-3">
|
<aside class="py-6 sm:py-8 lg:col-span-3">
|
||||||
<nav class="space-y-1">
|
<nav class="space-y-1">
|
||||||
<%= render partial: @sidenav_partial %>
|
<%= render partial: @sidenav_partial %>
|
||||||
|
|||||||
10
app/components/main_with_tabnav_component.html.erb
Normal file
10
app/components/main_with_tabnav_component.html.erb
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
<main class="w-full max-w-6xl mx-auto pb-12 px-4 md:px-6 lg:px-8">
|
||||||
|
<div class="md:min-h-[50vh] bg-white rounded-lg shadow">
|
||||||
|
<div class="px-6 sm:px-12 pt-2 sm:pt-4">
|
||||||
|
<%= render partial: @tabnav_partial %>
|
||||||
|
</div>
|
||||||
|
<div class="px-6 sm:px-12 py-8 sm:py-12">
|
||||||
|
<%= content %>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
7
app/components/main_with_tabnav_component.rb
Normal file
7
app/components/main_with_tabnav_component.rb
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class MainWithTabnavComponent < ViewComponent::Base
|
||||||
|
def initialize(tabnav_partial:)
|
||||||
|
@tabnav_partial = tabnav_partial
|
||||||
|
end
|
||||||
|
end
|
||||||
30
app/components/modal_component.html.erb
Normal file
30
app/components/modal_component.html.erb
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
<div tabindex="-1" class="relative z-10">
|
||||||
|
<!-- Modal Background -->
|
||||||
|
<div class="hidden fixed inset-0 bg-black bg-opacity-80 overflow-y-auto flex items-center justify-center"
|
||||||
|
data-modal-target="background"
|
||||||
|
data-action="click->modal#closeBackground"
|
||||||
|
data-transition-enter="transition-all ease-in-out duration-100"
|
||||||
|
data-transition-enter-from="bg-opacity-0"
|
||||||
|
data-transition-enter-to="bg-opacity-80"
|
||||||
|
data-transition-leave="transition-all ease-in-out duration-100"
|
||||||
|
data-transition-leave-from="bg-opacity-80"
|
||||||
|
data-transition-leave-to="bg-opacity-0">
|
||||||
|
|
||||||
|
<!-- Modal Container -->
|
||||||
|
<div data-modal-target="container"
|
||||||
|
class="relative m-4 max-h-screen w-auto max-w-full
|
||||||
|
hidden animate-scale-in fixed inset-0 overflow-y-auto flex items-center justify-center">
|
||||||
|
<!-- Modal Card -->
|
||||||
|
<div class="m-1 bg-white rounded shadow">
|
||||||
|
<div class="p-8">
|
||||||
|
<%= content %>
|
||||||
|
<% if @show_close_button %>
|
||||||
|
<div class="flex justify-end items-center flex-wrap mt-6">
|
||||||
|
<button class="btn-md btn-blue" data-action="click->modal#close:prevent">Close</button>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
5
app/components/modal_component.rb
Normal file
5
app/components/modal_component.rb
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
class ModalComponent < ViewComponent::Base
|
||||||
|
def initialize(show_close_button: true)
|
||||||
|
@show_close_button = show_close_button
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -34,6 +34,8 @@ class NotificationComponent < ViewComponent::Base
|
|||||||
'alert-octagon'
|
'alert-octagon'
|
||||||
when 'alert'
|
when 'alert'
|
||||||
'alert-octagon'
|
'alert-octagon'
|
||||||
|
when 'warning'
|
||||||
|
'alert-octagon'
|
||||||
else
|
else
|
||||||
'info'
|
'info'
|
||||||
end
|
end
|
||||||
|
|||||||
6
app/components/qr_code_modal_component.html.erb
Normal file
6
app/components/qr_code_modal_component.html.erb
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<%= render ModalComponent.new do %>
|
||||||
|
<% if @descripton.present? %>
|
||||||
|
<p class="mb-6"><%= @description %></p>
|
||||||
|
<% end %>
|
||||||
|
<p><%= raw @qr_code_svg %></p>
|
||||||
|
<% end %>
|
||||||
24
app/components/qr_code_modal_component.rb
Normal file
24
app/components/qr_code_modal_component.rb
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
require "rqrcode"
|
||||||
|
|
||||||
|
class QrCodeModalComponent < ViewComponent::Base
|
||||||
|
def initialize(qr_content:, description: nil)
|
||||||
|
@description = description
|
||||||
|
@qr_code_svg = qr_code_svg(qr_content)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def qr_code_svg(content)
|
||||||
|
qr_code = RQRCode::QRCode.new(content)
|
||||||
|
qr_code.as_svg(
|
||||||
|
color: "000",
|
||||||
|
shape_rendering: "crispEdges",
|
||||||
|
module_size: 6,
|
||||||
|
standalone: true,
|
||||||
|
use_path: true,
|
||||||
|
svg_attributes: {
|
||||||
|
class: 'inline-block'
|
||||||
|
}
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
3
app/components/quickstats_container_component.html.erb
Normal file
3
app/components/quickstats_container_component.html.erb
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<dl class="grid grid-cols-2 lg:grid-cols-4 gap-6 sm:gap-12">
|
||||||
|
<%= content %>
|
||||||
|
</dl>
|
||||||
4
app/components/quickstats_container_component.rb
Normal file
4
app/components/quickstats_container_component.rb
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class QuickstatsContainerComponent < ViewComponent::Base
|
||||||
|
end
|
||||||
18
app/components/quickstats_item_component.html.erb
Normal file
18
app/components/quickstats_item_component.html.erb
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
<div class="">
|
||||||
|
<dt class="mb-2 text-gray-500">
|
||||||
|
<%= @title %>
|
||||||
|
</dt>
|
||||||
|
<dd>
|
||||||
|
<% if @type == :number %>
|
||||||
|
<span class="text-2xl"><%= number_with_delimiter @value %></span>
|
||||||
|
<% else %>
|
||||||
|
<span class="text-2xl"><%= @value %></span>
|
||||||
|
<% end %>
|
||||||
|
<% if @unit %>
|
||||||
|
<span><%= @unit %></span>
|
||||||
|
<% end %>
|
||||||
|
<% if @meta %>
|
||||||
|
<span class="text-gray-500"><%= @meta %></span>
|
||||||
|
<% end %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
13
app/components/quickstats_item_component.rb
Normal file
13
app/components/quickstats_item_component.rb
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class QuickstatsItemComponent < ViewComponent::Base
|
||||||
|
def initialize(type:, title:, value:, unit: nil, meta: nil, icon_name: nil, icon_color_class: nil)
|
||||||
|
@type = type
|
||||||
|
@title = title
|
||||||
|
@value = value
|
||||||
|
@unit = unit
|
||||||
|
@meta = meta
|
||||||
|
@icon_name = icon_name
|
||||||
|
@icon_color_class = icon_color_class
|
||||||
|
end
|
||||||
|
end
|
||||||
26
app/components/rs_auth_component.html.erb
Normal file
26
app/components/rs_auth_component.html.erb
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
<div class="flex items-center gap-4">
|
||||||
|
<div class="h-16 w-16 flex-none">
|
||||||
|
<%= render AppCatalog::WebAppIconComponent.new(web_app: @web_app) %>
|
||||||
|
</div>
|
||||||
|
<div class="flex-grow">
|
||||||
|
<h4 class="mb-1 text-lg font-bold">
|
||||||
|
<%= @web_app&.name || @auth.app_name %>
|
||||||
|
</h4>
|
||||||
|
<p class="text-sm text-gray-500">
|
||||||
|
<%= @auth.client_id %>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<%= render DropdownComponent.new do %>
|
||||||
|
<%= render DropdownLinkComponent.new(
|
||||||
|
href: launch_app_services_storage_rs_auth_url(@auth)
|
||||||
|
) do %>
|
||||||
|
Launch app
|
||||||
|
<% end %>
|
||||||
|
<%= render DropdownLinkComponent.new(
|
||||||
|
href: revoke_services_storage_rs_auth_url(@auth),
|
||||||
|
separator: true, add_class: "text-red-700"
|
||||||
|
) do %>
|
||||||
|
Revoke access
|
||||||
|
<% end %>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
8
app/components/rs_auth_component.rb
Normal file
8
app/components/rs_auth_component.rb
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class RsAuthComponent < ViewComponent::Base
|
||||||
|
def initialize(auth:)
|
||||||
|
@auth = auth
|
||||||
|
@web_app = auth.web_app
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
<div class="w-[72vw] md:w-[500px]">
|
||||||
|
<header class="absolute z-10 h-36 sm:h-44 inset-x-1 top-1 rounded-t
|
||||||
|
bg-cover bg-center bg-gray-50"
|
||||||
|
style="background-image: url('<%= @profile["banner"]%>');">
|
||||||
|
<div class="inline-block z-20 size-28 sm:size-32 ml-4 mt-16 sm:mt-20">
|
||||||
|
<% if @profile["picture"].present? %>
|
||||||
|
<img src="<%= @profile["picture"] %>"
|
||||||
|
class="inline-block size:28 sm:size-32 rounded-full border-2 border-white" />
|
||||||
|
<% else %>
|
||||||
|
<span class="inline-block size:28 sm:size-32 overflow-hidden rounded-full border-2 border-white bg-gray-100">
|
||||||
|
<svg class="size-full text-gray-300" fill="currentColor" viewBox="0 0 24 24">
|
||||||
|
<path d="M24 20.993V24H0v-2.996A14.977 14.977 0 0112.004 15c4.904 0 9.26 2.354 11.996 5.993zM16.002 8.999a4 4 0 11-8 0 4 4 0 018 0z" />
|
||||||
|
</svg>
|
||||||
|
</span>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
<main class="mt-44 sm:mt-52">
|
||||||
|
<%= form_for(@user, url: setting_path(:nostr), html: { :method => :put }) do |f| %>
|
||||||
|
<%= render FormElements::FieldsetComponent.new(tag: "div", title: "Display name") do %>
|
||||||
|
<%= f.text_field :display_name, value: @display_name, class: "w-full sm:w-3/5" %>
|
||||||
|
<% if @validation_errors.present? && @validation_errors[:display_name].present? %>
|
||||||
|
<p class="error-msg mt-2"><%= @validation_errors[:display_name].first %></p>
|
||||||
|
<% end %>
|
||||||
|
<% end %>
|
||||||
|
<%= render FormElements::FieldsetComponent.new(tag: "div", title: "Nostr address (NIP-05)") do %>
|
||||||
|
<%= f.text_field :nip05_address, value: @profile["nip05"], class: "w-full sm:w-3/5" %>
|
||||||
|
<% if @validation_errors.present? && @validation_errors[:nip05_address].present? %>
|
||||||
|
<p class="error-msg mt-2"><%= @validation_errors[:nip05_address].first %></p>
|
||||||
|
<% end %>
|
||||||
|
<% end %>
|
||||||
|
<%= render FormElements::FieldsetComponent.new(tag: "div", title: "Ligtning address for Zaps") do %>
|
||||||
|
<%= f.text_field :lud16_address, value: @profile["lud16"], class: "w-full sm:w-3/5" %>
|
||||||
|
<% if @validation_errors.present? && @validation_errors[:lud16_address].present? %>
|
||||||
|
<p class="error-msg mt-2"><%= @validation_errors[:lud16_address].first %></p>
|
||||||
|
<% end %>
|
||||||
|
<% end %>
|
||||||
|
<% end %>
|
||||||
|
</main>
|
||||||
|
<footer>
|
||||||
|
<%# <%= @profile.inspect %>
|
||||||
|
<%# <%= @profile_event.inspect %>
|
||||||
|
</footer>
|
||||||
|
</div>
|
||||||
28
app/components/settings/nostr_edit_profile_component.rb
Normal file
28
app/components/settings/nostr_edit_profile_component.rb
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Settings
|
||||||
|
class NostrEditProfileComponent < ViewComponent::Base
|
||||||
|
def initialize(user:, profile_event:)
|
||||||
|
if profile_event.present?
|
||||||
|
@user = user
|
||||||
|
@profile_event = profile_event
|
||||||
|
@profile = JSON.parse(profile_event["content"])
|
||||||
|
@display_name = @profile["display_name"] || @profile["displayName"]
|
||||||
|
|
||||||
|
if @profile["nip05"].present? && @profile["nip05"] == @user.address
|
||||||
|
# "Your profile's Nostr address is set to <strong>#{ user_address }</strong>"
|
||||||
|
else
|
||||||
|
# "Your profile's Nostr address is not set to <strong>#{ user_address }</strong> yet"
|
||||||
|
end
|
||||||
|
|
||||||
|
if @profile["lud16"].present? && @profile["lud16"] == @user.address
|
||||||
|
# "Your profile's Lightning address is set to <strong>#{ user_address }</strong>"
|
||||||
|
else
|
||||||
|
# "Your profile's Lightning address is not set to <strong>#{ user_address }</strong> yet"
|
||||||
|
end
|
||||||
|
else
|
||||||
|
# "We could not find a profile for your public key"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
<% @statuses.each do |status| %>
|
||||||
|
<%= render StatusTextComponent.new(
|
||||||
|
text: status[:text],
|
||||||
|
icon_name: status[:icon_name],
|
||||||
|
icon_color: status[:icon_color]
|
||||||
|
) %>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<% if @status == 1 %>
|
||||||
|
<p class="mt-8">
|
||||||
|
<button class="btn-md btn-blue">
|
||||||
|
Edit my profile
|
||||||
|
</button>
|
||||||
|
</p>
|
||||||
|
<% elsif @status == 2 %>
|
||||||
|
<p class="mt-8">
|
||||||
|
<button class="btn-md btn-blue">
|
||||||
|
Create my profile
|
||||||
|
</button>
|
||||||
|
</p>
|
||||||
|
<% end %>
|
||||||
53
app/components/settings/nostr_profile_status_component.rb
Normal file
53
app/components/settings/nostr_profile_status_component.rb
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Settings
|
||||||
|
class NostrProfileStatusComponent < ViewComponent::Base
|
||||||
|
def initialize(profile_event:, user_address:)
|
||||||
|
@statuses = []
|
||||||
|
|
||||||
|
if profile_event.present?
|
||||||
|
profile = JSON.parse(profile_event["content"])
|
||||||
|
|
||||||
|
@statuses.push({
|
||||||
|
text: "You have a public Nostr profile",
|
||||||
|
icon_name: "check-circle",
|
||||||
|
icon_color: "emerald-500"
|
||||||
|
})
|
||||||
|
|
||||||
|
if profile["nip05"].present? && profile["nip05"] == user_address
|
||||||
|
@statuses.push({
|
||||||
|
text: "Your profile's Nostr address is set to <strong>#{ user_address }</strong>",
|
||||||
|
icon_name: "check-circle",
|
||||||
|
icon_color: "emerald-500"
|
||||||
|
})
|
||||||
|
else
|
||||||
|
@statuses.push({
|
||||||
|
text: "Your profile's Nostr address is not set to <strong>#{ user_address }</strong> yet",
|
||||||
|
icon_name: "alert-octagon",
|
||||||
|
icon_color: "amber-500"
|
||||||
|
})
|
||||||
|
end
|
||||||
|
|
||||||
|
if profile["lud16"].present? && profile["lud16"] == user_address
|
||||||
|
@statuses.push({
|
||||||
|
text: "Your profile's Lightning address is set to <strong>#{ user_address }</strong>",
|
||||||
|
icon_name: "check-circle",
|
||||||
|
icon_color: "emerald-500"
|
||||||
|
})
|
||||||
|
else
|
||||||
|
@statuses.push({
|
||||||
|
text: "Your profile's Lightning address is not set to <strong>#{ user_address }</strong> yet",
|
||||||
|
icon_name: "alert-octagon",
|
||||||
|
icon_color: "amber-500"
|
||||||
|
})
|
||||||
|
end
|
||||||
|
else
|
||||||
|
@statuses.push({
|
||||||
|
text: "We could not find a profile for your public key",
|
||||||
|
icon_name: "alert-octagon",
|
||||||
|
icon_color: "amber-500"
|
||||||
|
})
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
<%= render StatusTextComponent.new(
|
||||||
|
text: @text,
|
||||||
|
icon_name: @icon_name,
|
||||||
|
icon_color: @icon_color) %>
|
||||||
|
|
||||||
|
<% if @status == 1 %>
|
||||||
|
<p class="mt-8">
|
||||||
|
<button class="btn-md btn-blue">
|
||||||
|
Add the relay to my list
|
||||||
|
</button>
|
||||||
|
</p>
|
||||||
|
<% elsif @status == 2 %>
|
||||||
|
<p class="mt-8">
|
||||||
|
<button class="btn-md btn-blue">
|
||||||
|
Set up default relays
|
||||||
|
</button>
|
||||||
|
</p>
|
||||||
|
<% end %>
|
||||||
34
app/components/settings/nostr_relay_status_component.rb
Normal file
34
app/components/settings/nostr_relay_status_component.rb
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Settings
|
||||||
|
class NostrRelayStatusComponent < ViewComponent::Base
|
||||||
|
def initialize(nip65_event:)
|
||||||
|
if nip65_event.present?
|
||||||
|
if relay_urls(nip65_event).any? { |r| r.include?("wss://nostr.kosmos.org") }
|
||||||
|
@text = "You have a relay list, and the Kosmos relay is part of it"
|
||||||
|
@icon_name = "check-circle"
|
||||||
|
@icon_color = "emerald-500"
|
||||||
|
@status = 0
|
||||||
|
else
|
||||||
|
@text = "The Kosmos relay is missing from your relay list"
|
||||||
|
@icon_name = "alert-octagon"
|
||||||
|
@icon_color = "amber-500"
|
||||||
|
@status = 1
|
||||||
|
end
|
||||||
|
else
|
||||||
|
@text = "We could not find a relay list for your public key"
|
||||||
|
@icon_name = "alert-octagon"
|
||||||
|
@icon_color = "amber-500"
|
||||||
|
@status = 2
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def relay_urls(nip65_event)
|
||||||
|
nip65_event["tags"].select{ |t| t[0] == "r" }.map{ |t| t[1] }
|
||||||
|
# @inbox_relay_urls = relay_tags&.select{ |t| t[2] == "read" }&.map{ |t| t[1] }
|
||||||
|
# @outbox_relay_urls = relay_tags&.select{ |t| t[2] != "read" }&.map{ |t| t[1] }
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,4 +1,8 @@
|
|||||||
<%= link_to @path, class: @link_class do %>
|
<%= link_to @path, class: @link_class, title: (@disabled ? "Coming soon" : nil) do %>
|
||||||
|
<% if @icon.present? %>
|
||||||
<%= render partial: "icons/#{@icon}", locals: { custom_class: @icon_class } %>
|
<%= render partial: "icons/#{@icon}", locals: { custom_class: @icon_class } %>
|
||||||
|
<% elsif @text_icon.present? %>
|
||||||
|
<span class="mr-3"><%= @text_icon %></span>
|
||||||
|
<% end %>
|
||||||
<span class="truncate"><%= @name %></span>
|
<span class="truncate"><%= @name %></span>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
|||||||
@@ -1,10 +1,13 @@
|
|||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
class SidenavLinkComponent < ViewComponent::Base
|
class SidenavLinkComponent < ViewComponent::Base
|
||||||
def initialize(name:, path:, icon:, active: false, disabled: false)
|
def initialize(name:, level: 1, path:, icon: nil, text_icon: nil,
|
||||||
|
active: false, disabled: false)
|
||||||
@name = name
|
@name = name
|
||||||
|
@level = level
|
||||||
@path = path
|
@path = path
|
||||||
@icon = icon
|
@icon = icon
|
||||||
|
@text_icon = text_icon
|
||||||
@active = active
|
@active = active
|
||||||
@disabled = disabled
|
@disabled = disabled
|
||||||
@link_class = class_names_link(path)
|
@link_class = class_names_link(path)
|
||||||
@@ -12,12 +15,15 @@ class SidenavLinkComponent < ViewComponent::Base
|
|||||||
end
|
end
|
||||||
|
|
||||||
def class_names_link(path)
|
def class_names_link(path)
|
||||||
|
px = @level == 1 ? "px-4" : "pl-8 pr-4"
|
||||||
|
base = "#{px} py-2 group border-l-4 flex items-center text-base font-medium"
|
||||||
|
|
||||||
if @active
|
if @active
|
||||||
"bg-teal-50 border-teal-500 text-teal-700 hover:bg-teal-50 hover:text-teal-700 group border-l-4 px-4 py-2 flex items-center text-base font-medium"
|
"#{base} bg-teal-50 border-teal-500 text-teal-700 hover:bg-teal-50 hover:text-teal-700"
|
||||||
elsif @disabled
|
elsif @disabled
|
||||||
"border-transparent text-gray-400 hover:bg-gray-50 group border-l-4 px-4 py-2 flex items-center text-base font-medium"
|
"#{base} border-transparent text-gray-400 hover:bg-gray-50"
|
||||||
else
|
else
|
||||||
"border-transparent text-gray-900 hover:bg-gray-50 hover:text-gray-900 group border-l-4 px-4 py-2 flex items-center text-base font-medium"
|
"#{base} border-transparent text-gray-900 hover:bg-gray-50 hover:text-gray-900"
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
8
app/components/status_text_component.html.erb
Normal file
8
app/components/status_text_component.html.erb
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
<p class="flex gap-x-4 items-center">
|
||||||
|
<span class="inline-block h-6 w-6 grow-0 text-<%= @icon_color %>">
|
||||||
|
<%= render "icons/#{@icon_name}" %>
|
||||||
|
</span>
|
||||||
|
<span>
|
||||||
|
<%= raw @text %>
|
||||||
|
</span>
|
||||||
|
</p>
|
||||||
7
app/components/status_text_component.rb
Normal file
7
app/components/status_text_component.rb
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
class StatusTextComponent < ViewComponent::Base
|
||||||
|
def initialize(text:, icon_name:, icon_color:)
|
||||||
|
@text = text
|
||||||
|
@icon_name = icon_name
|
||||||
|
@icon_color = icon_color
|
||||||
|
end
|
||||||
|
end
|
||||||
3
app/components/tabnav_link_component.html.erb
Normal file
3
app/components/tabnav_link_component.html.erb
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<%= link_to @path, class: @link_class do %>
|
||||||
|
<%= @name %>
|
||||||
|
<% end %>
|
||||||
21
app/components/tabnav_link_component.rb
Normal file
21
app/components/tabnav_link_component.rb
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class TabnavLinkComponent < ViewComponent::Base
|
||||||
|
def initialize(name:, path:, active: false, disabled: false)
|
||||||
|
@name = name
|
||||||
|
@path = path
|
||||||
|
@active = active
|
||||||
|
@disabled = disabled
|
||||||
|
@link_class = class_names_link(path)
|
||||||
|
end
|
||||||
|
|
||||||
|
def class_names_link(path)
|
||||||
|
if @active
|
||||||
|
"border-indigo-500 text-indigo-600 w-1/2 py-4 px-1 text-center border-b-2"
|
||||||
|
elsif @disabled
|
||||||
|
"border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300 w-1/2 py-4 px-1 text-center border-b-2"
|
||||||
|
else
|
||||||
|
"border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300 w-1/2 py-4 px-1 text-center border-b-2"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
22
app/components/wallet_summary_component.html.erb
Normal file
22
app/components/wallet_summary_component.html.erb
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
<section class="w-full grid grid-cols-1 md:grid-cols-12 md:mb-0">
|
||||||
|
<div class="md:col-span-8">
|
||||||
|
<p>
|
||||||
|
Send and receive sats via the Bitcoin Lightning Network.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div class="md:col-span-4 mt-4 md:mt-0">
|
||||||
|
<p class="font-mono md:text-right mb-0 p-4 border border-gray-300 rounded-lg overflow-hidden">
|
||||||
|
<% if @balance %>
|
||||||
|
<span class="text-2xl"><%= number_with_delimiter @balance %></span>
|
||||||
|
<span class="text-xl">sats</span>
|
||||||
|
<br>
|
||||||
|
<span class="text-sm text-gray-500">Available balance</span>
|
||||||
|
<% else %>
|
||||||
|
<span class="text-2xl">n/a</span>
|
||||||
|
<span class="text-xl">sats</span>
|
||||||
|
<br>
|
||||||
|
<span class="text-sm text-gray-500">Balance unavailable</span>
|
||||||
|
<% end %>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
8
app/components/wallet_summary_component.rb
Normal file
8
app/components/wallet_summary_component.rb
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class WalletSummaryComponent < ViewComponent::Base
|
||||||
|
def initialize(balance:)
|
||||||
|
@balance = balance
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
7
app/controllers/account_controller.rb
Normal file
7
app/controllers/account_controller.rb
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
class AccountController < ApplicationController
|
||||||
|
before_action :authenticate_user!
|
||||||
|
|
||||||
|
def index
|
||||||
|
@current_section = :account
|
||||||
|
end
|
||||||
|
end
|
||||||
9
app/controllers/admin/app_catalog/web_apps_controller.rb
Normal file
9
app/controllers/admin/app_catalog/web_apps_controller.rb
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
class Admin::AppCatalog::WebAppsController < Admin::AppCatalogController
|
||||||
|
def index
|
||||||
|
@pagy, @web_apps = pagy(AppCatalog::WebApp.order('created_at desc'))
|
||||||
|
|
||||||
|
@stats = {
|
||||||
|
known_apps: AppCatalog::WebApp.count
|
||||||
|
}
|
||||||
|
end
|
||||||
|
end
|
||||||
9
app/controllers/admin/app_catalog_controller.rb
Normal file
9
app/controllers/admin/app_catalog_controller.rb
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
class Admin::AppCatalogController < Admin::BaseController
|
||||||
|
before_action :set_current_section
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_current_section
|
||||||
|
@current_section = :app_catalog
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
class Admin::BaseController < ApplicationController
|
class Admin::BaseController < ApplicationController
|
||||||
|
include Pagy::Backend
|
||||||
|
|
||||||
before_action :authenticate_user!
|
before_action :authenticate_user!
|
||||||
before_action :authorize_admin
|
before_action :authorize_admin
|
||||||
@@ -7,5 +8,4 @@ class Admin::BaseController < ApplicationController
|
|||||||
def set_context
|
def set_context
|
||||||
@context = :admin
|
@context = :admin
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -3,13 +3,16 @@ class Admin::DonationsController < Admin::BaseController
|
|||||||
before_action :set_current_section, only: [:index, :show, :new, :edit]
|
before_action :set_current_section, only: [:index, :show, :new, :edit]
|
||||||
|
|
||||||
# GET /donations
|
# GET /donations
|
||||||
# GET /donations.json
|
|
||||||
def index
|
def index
|
||||||
@donations = Donation.all
|
@pagy, @donations = pagy(Donation.completed.order('paid_at desc'))
|
||||||
|
|
||||||
|
@stats = {
|
||||||
|
overall_sats: @donations.sum("amount_sats"),
|
||||||
|
donor_count: Donation.completed.count(:user_id)
|
||||||
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
# GET /donations/1
|
# GET /donations/1
|
||||||
# GET /donations/1.json
|
|
||||||
def show
|
def show
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -23,43 +26,41 @@ class Admin::DonationsController < Admin::BaseController
|
|||||||
end
|
end
|
||||||
|
|
||||||
# POST /donations
|
# POST /donations
|
||||||
# POST /donations.json
|
|
||||||
def create
|
def create
|
||||||
@donation = Donation.new(donation_params)
|
@donation = Donation.new(donation_params)
|
||||||
|
|
||||||
respond_to do |format|
|
if @donation.paid_at == nil
|
||||||
if @donation.save
|
@donation.errors.add(:paid_at, message: "is required")
|
||||||
format.html { redirect_to admin_donation_url(@donation), notice: 'Donation was successfully created.' }
|
render :new, status: :unprocessable_entity and return
|
||||||
format.json { render :show, status: :created, location: @donation }
|
end
|
||||||
else
|
|
||||||
format.html { render :new }
|
if @donation.save
|
||||||
format.json { render json: @donation.errors, status: :unprocessable_entity }
|
redirect_to admin_donation_url(@donation), flash: {
|
||||||
end
|
success: 'Donation was successfully created.'
|
||||||
|
}
|
||||||
|
else
|
||||||
|
render :new, status: :unprocessable_entity
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# PATCH/PUT /donations/1
|
# PUT /donations/1
|
||||||
# PATCH/PUT /donations/1.json
|
|
||||||
def update
|
def update
|
||||||
respond_to do |format|
|
if @donation.update(donation_params)
|
||||||
if @donation.update(donation_params)
|
redirect_to admin_donation_url(@donation), flash: {
|
||||||
format.html { redirect_to admin_donation_url(@donation), notice: 'Donation was successfully updated.' }
|
success: 'Donation was successfully updated.'
|
||||||
format.json { render :show, status: :ok, location: @donation }
|
}
|
||||||
else
|
else
|
||||||
format.html { render :edit }
|
render :edit, status: :unprocessable_entity
|
||||||
format.json { render json: @donation.errors, status: :unprocessable_entity }
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# DELETE /donations/1
|
# DELETE /donations/1
|
||||||
# DELETE /donations/1.json
|
|
||||||
def destroy
|
def destroy
|
||||||
@donation.destroy
|
@donation.destroy
|
||||||
respond_to do |format|
|
|
||||||
format.html { redirect_to admin_donations_url, notice: 'Donation was successfully destroyed.' }
|
redirect_to admin_donations_url, flash: {
|
||||||
format.json { head :no_content }
|
success: 'Donation was successfully destroyed.'
|
||||||
end
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
@@ -70,7 +71,10 @@ class Admin::DonationsController < Admin::BaseController
|
|||||||
|
|
||||||
# Only allow a list of trusted parameters through.
|
# Only allow a list of trusted parameters through.
|
||||||
def donation_params
|
def donation_params
|
||||||
params.require(:donation).permit(:user_id, :amount_sats, :amount_eur, :amount_usd, :public_name, :paid_at)
|
params.require(:donation).permit(
|
||||||
|
:user_id, :donation_method,
|
||||||
|
:amount_sats, :fiat_amount, :fiat_currency,
|
||||||
|
:public_name, :paid_at)
|
||||||
end
|
end
|
||||||
|
|
||||||
def set_current_section
|
def set_current_section
|
||||||
|
|||||||
@@ -1,8 +1,12 @@
|
|||||||
class Admin::InvitationsController < Admin::BaseController
|
class Admin::InvitationsController < Admin::BaseController
|
||||||
def index
|
def index
|
||||||
@current_section = :invitations
|
@current_section = :invitations
|
||||||
@invitations_unused_count = Invitation.unused.count
|
@pagy, @invitations_used = pagy(Invitation.used.order('used_at desc'))
|
||||||
@users_with_referrals_count = Invitation.used.distinct.count(:user_id)
|
|
||||||
@invitations_used = Invitation.used.order('used_at desc')
|
@stats = {
|
||||||
|
available: Invitation.unused.count,
|
||||||
|
accepted: @invitations_used.length,
|
||||||
|
users_with_referrals: Invitation.used.distinct.count(:user_id)
|
||||||
|
}
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
class Admin::LdapUsersController < Admin::BaseController
|
|
||||||
before_action :set_current_section
|
|
||||||
|
|
||||||
def index
|
|
||||||
attributes = %w{dn cn uid mail admin}
|
|
||||||
filter = Net::LDAP::Filter.eq("uid", "*")
|
|
||||||
|
|
||||||
@ou = params[:ou] || "kosmos.org"
|
|
||||||
treebase = "ou=#{@ou},cn=users,dc=kosmos,dc=org"
|
|
||||||
|
|
||||||
entries = ldap_client.search(base: treebase, filter: filter, attributes: attributes)
|
|
||||||
entries.sort_by! { |e| e.cn[0] }
|
|
||||||
|
|
||||||
@entries = entries.collect do |e|
|
|
||||||
{
|
|
||||||
uid: e.uid.first,
|
|
||||||
mail: e.try(:mail) ? e.mail.first : nil,
|
|
||||||
admin: e.try(:admin) ? 'admin' : nil
|
|
||||||
# password: e.userpassword.first
|
|
||||||
}
|
|
||||||
end
|
|
||||||
# ldap_client.get_operation_result
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def ldap_client
|
|
||||||
ldap_client ||= Net::LDAP.new host: ldap_config['host'],
|
|
||||||
port: ldap_config['port'],
|
|
||||||
encryption: ldap_config['ssl'],
|
|
||||||
auth: {
|
|
||||||
method: :simple,
|
|
||||||
username: ldap_config['admin_user'],
|
|
||||||
password: ldap_config['admin_password']
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def ldap_config
|
|
||||||
ldap_config ||= YAML.load(ERB.new(File.read("#{Rails.root}/config/ldap.yml")).result)[Rails.env]
|
|
||||||
end
|
|
||||||
|
|
||||||
def set_current_section
|
|
||||||
@current_section = :ldap_users
|
|
||||||
end
|
|
||||||
end
|
|
||||||
21
app/controllers/admin/lightning_controller.rb
Normal file
21
app/controllers/admin/lightning_controller.rb
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
class Admin::LightningController < Admin::BaseController
|
||||||
|
before_action :check_feature_enabled
|
||||||
|
|
||||||
|
def index
|
||||||
|
@current_section = :lightning
|
||||||
|
|
||||||
|
@users = User.pluck(:cn, :ou, :ln_account)
|
||||||
|
@accounts = LndhubAccount.with_balances.order(balance: :desc).to_a
|
||||||
|
|
||||||
|
@ln = {}
|
||||||
|
@ln[:current_balance] = LndhubAccount.current.joins(:ledgers).sum("account_ledgers.amount")
|
||||||
|
@ln[:users_with_sats] = @accounts.length
|
||||||
|
end
|
||||||
|
|
||||||
|
def check_feature_enabled
|
||||||
|
if !Setting.lndhub_admin_enabled?
|
||||||
|
flash[:alert] = "Lightning Admin UI not enabled"
|
||||||
|
redirect_to admin_root_path and return
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
20
app/controllers/admin/settings/registrations_controller.rb
Normal file
20
app/controllers/admin/settings/registrations_controller.rb
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
class Admin::Settings::RegistrationsController < Admin::SettingsController
|
||||||
|
def show
|
||||||
|
end
|
||||||
|
|
||||||
|
def update
|
||||||
|
update_settings
|
||||||
|
|
||||||
|
redirect_to admin_settings_registrations_path, flash: {
|
||||||
|
success: "Settings saved"
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def setting_params
|
||||||
|
params.require(:setting).permit([
|
||||||
|
:reserved_usernames, default_services: []
|
||||||
|
])
|
||||||
|
end
|
||||||
|
end
|
||||||
32
app/controllers/admin/settings/services_controller.rb
Normal file
32
app/controllers/admin/settings/services_controller.rb
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
class Admin::Settings::ServicesController < Admin::SettingsController
|
||||||
|
before_action :set_service, only: [:show, :update]
|
||||||
|
|
||||||
|
def index
|
||||||
|
redirect_to admin_settings_service_path("btcpay")
|
||||||
|
end
|
||||||
|
|
||||||
|
def show
|
||||||
|
end
|
||||||
|
|
||||||
|
def update
|
||||||
|
update_settings
|
||||||
|
|
||||||
|
redirect_to admin_settings_service_path(@service), flash: {
|
||||||
|
success: "Settings saved"
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_subsection
|
||||||
|
@subsection = "services"
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_service
|
||||||
|
@service = params[:service]
|
||||||
|
|
||||||
|
if @service.blank?
|
||||||
|
redirect_to admin_settings_services_path and return
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
49
app/controllers/admin/settings_controller.rb
Normal file
49
app/controllers/admin/settings_controller.rb
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
class Admin::SettingsController < Admin::BaseController
|
||||||
|
before_action :set_current_section
|
||||||
|
|
||||||
|
def index
|
||||||
|
end
|
||||||
|
|
||||||
|
def update_settings
|
||||||
|
@errors = ActiveModel::Errors.new(Setting.new)
|
||||||
|
changed_keys = []
|
||||||
|
|
||||||
|
setting_params.keys.each do |key|
|
||||||
|
next if clean_param(key).nil? ||
|
||||||
|
(Setting.send(key).to_s == clean_param(key))
|
||||||
|
|
||||||
|
changed_keys.push(key)
|
||||||
|
setting = Setting.new(var: key)
|
||||||
|
setting.value = clean_param(key)
|
||||||
|
unless setting.valid?
|
||||||
|
@errors.merge!(setting.errors)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
if @errors.any?
|
||||||
|
render :show and return
|
||||||
|
end
|
||||||
|
|
||||||
|
changed_keys.each do |key|
|
||||||
|
Setting.send("#{key}=", clean_param(key))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_current_section
|
||||||
|
@current_section = :settings
|
||||||
|
end
|
||||||
|
|
||||||
|
def setting_params
|
||||||
|
params.require(:setting).permit(Setting.editable_keys.map(&:to_sym))
|
||||||
|
end
|
||||||
|
|
||||||
|
def clean_param(key)
|
||||||
|
if Setting.get_field(key)[:type] == :string
|
||||||
|
setting_params[key].strip
|
||||||
|
else
|
||||||
|
setting_params[key]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
62
app/controllers/admin/users_controller.rb
Normal file
62
app/controllers/admin/users_controller.rb
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
class Admin::UsersController < Admin::BaseController
|
||||||
|
before_action :set_user, except: [:index]
|
||||||
|
before_action :set_current_section
|
||||||
|
|
||||||
|
# GET /admin/users
|
||||||
|
def index
|
||||||
|
ldap = LdapService.new
|
||||||
|
@ou = Setting.primary_domain
|
||||||
|
@pagy, @users = pagy(User.where(ou: @ou).order(cn: :asc))
|
||||||
|
|
||||||
|
@stats = {
|
||||||
|
users_confirmed: User.where(ou: @ou).confirmed.count,
|
||||||
|
users_pending: User.where(ou: @ou).pending.count
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
# GET /admin/users/:username
|
||||||
|
def show
|
||||||
|
if Setting.lndhub_admin_enabled?
|
||||||
|
@lndhub_user = @user.lndhub_user
|
||||||
|
end
|
||||||
|
|
||||||
|
@services_enabled = @user.services_enabled
|
||||||
|
|
||||||
|
@avatar = LdapManager::FetchAvatar.call(cn: @user.cn)
|
||||||
|
end
|
||||||
|
|
||||||
|
# POST /admin/users/:username/invitations
|
||||||
|
def create_invitations
|
||||||
|
amount = params[:amount].to_i
|
||||||
|
notify_user = ActiveRecord::Type::Boolean.new.cast(params[:notify_user])
|
||||||
|
|
||||||
|
CreateInvitations.call(user: @user, amount: amount, notify: notify_user)
|
||||||
|
|
||||||
|
redirect_to admin_user_path(@user.cn), flash: {
|
||||||
|
success: "Added #{amount} invitations to #{@user.cn}'s account"
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
# DELETE /admin/users/:username/invitations
|
||||||
|
def delete_invitations
|
||||||
|
invitations = @user.invitations.unused
|
||||||
|
amount = invitations.count
|
||||||
|
|
||||||
|
invitations.destroy_all
|
||||||
|
|
||||||
|
redirect_to admin_user_path(@user.cn), flash: {
|
||||||
|
success: "Removed #{amount} invitations from #{@user.cn}'s account"
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_user
|
||||||
|
@user = User.find_by(cn: params[:username], ou: Setting.primary_domain)
|
||||||
|
http_status :not_found unless @user
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_current_section
|
||||||
|
@current_section = :users
|
||||||
|
end
|
||||||
|
end
|
||||||
5
app/controllers/api/base_controller.rb
Normal file
5
app/controllers/api/base_controller.rb
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
class Api::BaseController < ApplicationController
|
||||||
|
|
||||||
|
layout false
|
||||||
|
|
||||||
|
end
|
||||||
37
app/controllers/api/btcpay_controller.rb
Normal file
37
app/controllers/api/btcpay_controller.rb
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
class Api::BtcpayController < Api::BaseController
|
||||||
|
before_action :require_feature_enabled
|
||||||
|
before_action :set_cors_access_control_headers
|
||||||
|
|
||||||
|
def onchain_btc_balance
|
||||||
|
balance = BtcpayManager::FetchOnchainWalletBalance.call
|
||||||
|
render json: balance
|
||||||
|
rescue => error
|
||||||
|
Rails.logger.warn "Failed to fetch BTC wallet balance: #{error.message}"
|
||||||
|
render json: { error: 'Failed to fetch wallet balance' },
|
||||||
|
status: 500
|
||||||
|
end
|
||||||
|
|
||||||
|
def lightning_btc_balance
|
||||||
|
balance = BtcpayManager::FetchLightningWalletBalance.call
|
||||||
|
render json: balance
|
||||||
|
rescue => error
|
||||||
|
Rails.logger.warn "Failed to fetch BTC lightning balance: #{error.message}"
|
||||||
|
render json: { error: 'Failed to fetch wallet balance' },
|
||||||
|
status: 500
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def require_feature_enabled
|
||||||
|
unless Setting.btcpay_publish_wallet_balances
|
||||||
|
http_status :not_found and return
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_cors_access_control_headers
|
||||||
|
return unless Rails.env.development?
|
||||||
|
headers['Access-Control-Allow-Origin'] = "*"
|
||||||
|
headers['Access-Control-Allow-Headers'] = "*"
|
||||||
|
headers['Access-Control-Allow-Methods'] = "GET"
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -3,6 +3,18 @@ class ApplicationController < ActionController::Base
|
|||||||
render :text => exception, :status => 500
|
render :text => exception, :status => 500
|
||||||
end
|
end
|
||||||
|
|
||||||
|
before_action :sentry_set_user
|
||||||
|
|
||||||
|
def sentry_set_user
|
||||||
|
return unless Setting.sentry_enabled
|
||||||
|
|
||||||
|
if user_signed_in?
|
||||||
|
Sentry.set_user(id: current_user.id, username: current_user.cn)
|
||||||
|
else
|
||||||
|
Sentry.set_user({})
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def require_user_signed_in
|
def require_user_signed_in
|
||||||
unless user_signed_in?
|
unless user_signed_in?
|
||||||
redirect_to welcome_path and return
|
redirect_to welcome_path and return
|
||||||
@@ -25,4 +37,35 @@ class ApplicationController < ActionController::Base
|
|||||||
format.any { head status }
|
format.any { head status }
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def after_sign_in_path_for(user)
|
||||||
|
session[:user_return_to] || root_path
|
||||||
|
end
|
||||||
|
|
||||||
|
def lndhub_authenticate(options={})
|
||||||
|
if session[:ln_auth_token].present? && !options[:force_reauth]
|
||||||
|
@ln_auth_token = session[:ln_auth_token]
|
||||||
|
else
|
||||||
|
lndhub = Lndhub.new
|
||||||
|
auth_token = lndhub.authenticate(current_user)
|
||||||
|
session[:ln_auth_token] = auth_token
|
||||||
|
@ln_auth_token = auth_token
|
||||||
|
end
|
||||||
|
rescue => e
|
||||||
|
Sentry.capture_exception(e) if Setting.sentry_enabled?
|
||||||
|
end
|
||||||
|
|
||||||
|
def lndhub_fetch_balance
|
||||||
|
@balance = LndhubManager::FetchUserBalance.call(auth_token: @ln_auth_token)
|
||||||
|
rescue AuthError
|
||||||
|
lndhub_authenticate(force_reauth: true)
|
||||||
|
raise if @fetch_balance_retried
|
||||||
|
@fetch_balance_retried = true
|
||||||
|
lndhub_fetch_balance
|
||||||
|
end
|
||||||
|
|
||||||
|
def nostr_event_from_params
|
||||||
|
params.permit!
|
||||||
|
params[:signed_event].to_h.symbolize_keys
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
129
app/controllers/contributions/donations_controller.rb
Normal file
129
app/controllers/contributions/donations_controller.rb
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
class Contributions::DonationsController < ApplicationController
|
||||||
|
include BtcpayHelper
|
||||||
|
|
||||||
|
before_action :authenticate_user!
|
||||||
|
before_action :set_donation_methods, only: [:index, :create]
|
||||||
|
before_action :require_donation_method_enabled, only: [:create]
|
||||||
|
before_action :validate_donation_params, only: [:create]
|
||||||
|
before_action :set_donation, only: [:confirm_btcpay]
|
||||||
|
|
||||||
|
# GET /contributions/donations
|
||||||
|
def index
|
||||||
|
@current_section = :contributions
|
||||||
|
@donations_completed = current_user.donations.completed.order('paid_at desc')
|
||||||
|
@donations_pending = current_user.donations.processing.order('created_at desc')
|
||||||
|
|
||||||
|
if Setting.lndhub_enabled?
|
||||||
|
begin
|
||||||
|
lndhub_authenticate
|
||||||
|
lndhub_fetch_balance
|
||||||
|
rescue
|
||||||
|
@balance = 0
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# POST /contributions/donations
|
||||||
|
def create
|
||||||
|
if params[:currency] == "sats"
|
||||||
|
fiat_amount = nil
|
||||||
|
fiat_currency = nil
|
||||||
|
amount_sats = params[:amount]
|
||||||
|
else
|
||||||
|
fiat_amount = params[:amount].to_i
|
||||||
|
fiat_currency = params[:currency]
|
||||||
|
amount_sats = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
@donation = current_user.donations.create!(
|
||||||
|
donation_method: params[:donation_method],
|
||||||
|
payment_method: nil,
|
||||||
|
paid_at: nil,
|
||||||
|
amount_sats: amount_sats,
|
||||||
|
fiat_amount: (fiat_amount.nil? ? nil : fiat_amount * 100), # store in cents
|
||||||
|
fiat_currency: fiat_currency,
|
||||||
|
public_name: params[:public_name]
|
||||||
|
)
|
||||||
|
|
||||||
|
case params[:donation_method]
|
||||||
|
when "btcpay"
|
||||||
|
res = BtcpayManager::CreateInvoice.call(
|
||||||
|
amount: fiat_amount || (amount_sats.to_f / 100000000),
|
||||||
|
currency: fiat_currency || "BTC",
|
||||||
|
redirect_url: confirm_btcpay_contributions_donation_url(@donation)
|
||||||
|
)
|
||||||
|
|
||||||
|
@donation.update! btcpay_invoice_id: res["id"]
|
||||||
|
|
||||||
|
redirect_to btcpay_checkout_url(res["id"]), allow_other_host: true
|
||||||
|
else
|
||||||
|
redirect_to contributions_donations_url, flash: {
|
||||||
|
error: "Donation method currently not available"
|
||||||
|
}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def confirm_btcpay
|
||||||
|
redirect_to contributions_donations_url and return if @donation.completed?
|
||||||
|
|
||||||
|
invoice = BtcpayManager::FetchInvoice.call(invoice_id: @donation.btcpay_invoice_id)
|
||||||
|
|
||||||
|
if @donation.amount_sats.present?
|
||||||
|
# TODO make default fiat currency configurable and/or determine from user's
|
||||||
|
# i18n browser settings
|
||||||
|
@donation.fiat_currency = "EUR"
|
||||||
|
exchange_rate = BtcpayManager::FetchExchangeRate.call(fiat_currency: @donation.fiat_currency)
|
||||||
|
@donation.fiat_amount = (((@donation.amount_sats.to_f / 100000000) * exchange_rate) * 100).to_i
|
||||||
|
else
|
||||||
|
amt_str = invoice["paymentMethods"].first["amount"]
|
||||||
|
@donation.amount_sats = amt_str.tr(".","").sub(/0*$/, "").to_i
|
||||||
|
end
|
||||||
|
|
||||||
|
case invoice["status"]
|
||||||
|
when "Settled"
|
||||||
|
@donation.paid_at = DateTime.now
|
||||||
|
@donation.payment_status = "settled"
|
||||||
|
@donation.save!
|
||||||
|
flash_message = { success: "Thank you!" }
|
||||||
|
when "Processing"
|
||||||
|
unless @donation.processing?
|
||||||
|
@donation.payment_status = "processing"
|
||||||
|
@donation.save!
|
||||||
|
flash_message = { success: "Thank you! We will send you an email when the payment is confirmed." }
|
||||||
|
BtcpayCheckDonationJob.set(wait: 20.seconds).perform_later(@donation)
|
||||||
|
end
|
||||||
|
when "Expired"
|
||||||
|
flash_message = { warning: "The payment request for this donation has expired" }
|
||||||
|
else
|
||||||
|
flash_message = { warning: "Could not determine status of payment" }
|
||||||
|
end
|
||||||
|
|
||||||
|
redirect_to contributions_donations_url, flash: flash_message
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_donation
|
||||||
|
@donation = current_user.donations.find_by(id: params[:id])
|
||||||
|
http_status :not_found unless @donation.present?
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_donation_methods
|
||||||
|
@donation_methods = []
|
||||||
|
@donation_methods.push :btcpay if Setting.btcpay_enabled?
|
||||||
|
@donation_methods.push :lndhub if Setting.lndhub_enabled?
|
||||||
|
@donation_methods.push :opencollective if Setting.opencollective_enabled?
|
||||||
|
end
|
||||||
|
|
||||||
|
def require_donation_method_enabled
|
||||||
|
http_status :forbidden unless @donation_methods.include?(
|
||||||
|
params[:donation_method].to_sym
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
def validate_donation_params
|
||||||
|
if !%w[EUR USD sats].include?(params[:currency]) || (params[:amount].to_i <= 0)
|
||||||
|
http_status :unprocessable_entity
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
8
app/controllers/contributions/projects_controller.rb
Normal file
8
app/controllers/contributions/projects_controller.rb
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
class Contributions::ProjectsController < ApplicationController
|
||||||
|
before_action :authenticate_user!
|
||||||
|
|
||||||
|
# GET /contributions
|
||||||
|
def index
|
||||||
|
@current_section = :contributions
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -2,6 +2,6 @@ class DashboardController < ApplicationController
|
|||||||
before_action :require_user_signed_in
|
before_action :require_user_signed_in
|
||||||
|
|
||||||
def index
|
def index
|
||||||
@current_section = :dashboard
|
@current_section = :services
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
17
app/controllers/discourse/sso_controller.rb
Normal file
17
app/controllers/discourse/sso_controller.rb
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
class Discourse::SsoController < ApplicationController
|
||||||
|
before_action :authenticate_user!
|
||||||
|
|
||||||
|
def connect
|
||||||
|
secret = Setting.discourse_connect_secret
|
||||||
|
sso = DiscourseApi::SingleSignOn.parse(request.query_string, secret)
|
||||||
|
sso.external_id = current_user.id
|
||||||
|
sso.email = current_user.email
|
||||||
|
sso.username = current_user.cn
|
||||||
|
sso.name = current_user.display_name
|
||||||
|
sso.admin = current_user.is_admin?
|
||||||
|
sso.sso_secret = secret
|
||||||
|
|
||||||
|
redirect_to sso.to_url("#{Setting.discourse_public_url}/session/sso_login"),
|
||||||
|
allow_other_host: true
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
class DonationsController < ApplicationController
|
|
||||||
before_action :require_user_signed_in
|
|
||||||
|
|
||||||
# GET /donations
|
|
||||||
# GET /donations.json
|
|
||||||
def index
|
|
||||||
@donations = current_user.donations.completed
|
|
||||||
@current_section = :contributions
|
|
||||||
end
|
|
||||||
end
|
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
class InvitationsController < ApplicationController
|
class InvitationsController < ApplicationController
|
||||||
before_action :require_user_signed_in, except: ["show"]
|
before_action :authenticate_user!, except: ["show"]
|
||||||
before_action :require_user_signed_out, only: ["show"]
|
before_action :require_user_signed_out, only: ["show"]
|
||||||
|
|
||||||
# GET /invitations
|
# GET /invitations
|
||||||
def index
|
def index
|
||||||
@invitations_unused = current_user.invitations.unused
|
@invitations_unused = current_user.invitations.unused
|
||||||
@invitations_used = current_user.invitations.used
|
@invitations_used = current_user.invitations.used.order('used_at desc')
|
||||||
@current_section = :invitations
|
@current_section = :invitations
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -27,7 +27,10 @@ class InvitationsController < ApplicationController
|
|||||||
|
|
||||||
respond_to do |format|
|
respond_to do |format|
|
||||||
if @invitation.save
|
if @invitation.save
|
||||||
format.html { redirect_to @invitation, notice: 'Invitation was successfully created.' }
|
format.html do redirect_to @invitation, flash: {
|
||||||
|
success: 'Invitation was successfully created.'
|
||||||
|
}
|
||||||
|
end
|
||||||
format.json { render :show, status: :created, location: @invitation }
|
format.json { render :show, status: :created, location: @invitation }
|
||||||
else
|
else
|
||||||
format.html { render :new }
|
format.html { render :new }
|
||||||
|
|||||||
@@ -1,75 +1,161 @@
|
|||||||
class LnurlpayController < ApplicationController
|
class LnurlpayController < ApplicationController
|
||||||
before_action :find_user_by_address
|
before_action :check_service_available
|
||||||
|
before_action :find_user
|
||||||
|
before_action :set_cors_access_control_headers
|
||||||
|
|
||||||
MIN_SATS = 100
|
MIN_SATS = 10
|
||||||
MAX_SATS = 1_000_000
|
MAX_SATS = 1_000_000
|
||||||
MAX_COMMENT_CHARS = 100
|
MAX_COMMENT_CHARS = 100
|
||||||
|
|
||||||
|
# GET /.well-known/lnurlp/:username
|
||||||
def index
|
def index
|
||||||
render json: {
|
res = {
|
||||||
status: "OK",
|
status: "OK",
|
||||||
callback: "https://accounts.kosmos.org/lnurlpay/#{@user.address}/invoice",
|
callback: "https://#{Setting.accounts_domain}/lnurlpay/#{@user.cn}/invoice",
|
||||||
tag: "payRequest",
|
tag: "payRequest",
|
||||||
maxSendable: MAX_SATS * 1000, # msat
|
maxSendable: MAX_SATS * 1000, # msat
|
||||||
minSendable: MIN_SATS * 1000, # msat
|
minSendable: MIN_SATS * 1000, # msat
|
||||||
metadata: metadata(@user.address),
|
metadata: metadata(@user.address),
|
||||||
commentAllowed: MAX_COMMENT_CHARS
|
commentAllowed: MAX_COMMENT_CHARS
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if Setting.nostr_enabled?
|
||||||
|
res[:allowsNostr] = true
|
||||||
|
res[:nostrPubkey] = Setting.nostr_public_key
|
||||||
|
end
|
||||||
|
|
||||||
|
render json: res
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# GET /.well-known/keysend/:username
|
||||||
|
def keysend
|
||||||
|
http_status :not_found and return unless Setting.lndhub_keysend_enabled?
|
||||||
|
|
||||||
|
render json: {
|
||||||
|
status: "OK",
|
||||||
|
tag: "keysend",
|
||||||
|
pubkey: Setting.lndhub_public_key,
|
||||||
|
customData: [{
|
||||||
|
customKey: "696969",
|
||||||
|
customValue: @user.ln_account
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
# GET /lnurlpay/:username/invoice
|
||||||
def invoice
|
def invoice
|
||||||
amount = params[:amount].to_i / 1000 # msats
|
amount = params[:amount].to_i / 1000 # msats to sats
|
||||||
address = params[:address]
|
|
||||||
comment = params[:comment] || ""
|
comment = params[:comment] || ""
|
||||||
|
address = @user.address
|
||||||
|
|
||||||
if !valid_amount?(amount)
|
if !valid_amount?(amount)
|
||||||
render json: { status: "ERROR", reason: "Invalid amount" }
|
render json: { status: "ERROR", reason: "Invalid amount" }
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
if !valid_comment?(comment)
|
if params[:nostr].present? && Setting.nostr_enabled?
|
||||||
render json: { status: "ERROR", reason: "Comment too long" }
|
handle_zap_request amount, params[:nostr], params[:lnurl]
|
||||||
return
|
else
|
||||||
|
handle_pay_request address, amount, comment
|
||||||
end
|
end
|
||||||
|
|
||||||
memo = "Sats for #{address}"
|
|
||||||
memo = "#{memo}: \"#{comment}\"" if comment.present?
|
|
||||||
|
|
||||||
payment_request = @user.ln_create_invoice({
|
|
||||||
amount: amount, # we create invoices in sats
|
|
||||||
memo: memo,
|
|
||||||
description_hash: Digest::SHA2.hexdigest(metadata(address)),
|
|
||||||
})
|
|
||||||
|
|
||||||
render json: {
|
|
||||||
status: "OK",
|
|
||||||
successAction: {
|
|
||||||
tag: "message",
|
|
||||||
message: "Sats received. Thank you!"
|
|
||||||
},
|
|
||||||
routes: [],
|
|
||||||
pr: payment_request
|
|
||||||
}
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def find_user_by_address
|
def set_cors_access_control_headers
|
||||||
address = params[:address].split("@")
|
headers['Access-Control-Allow-Origin'] = "*"
|
||||||
@user = User.where(cn: address.first, ou: address.last).first
|
headers['Access-Control-Allow-Headers'] = "*"
|
||||||
http_status :not_found if @user.nil?
|
headers['Access-Control-Allow-Methods'] = "GET"
|
||||||
end
|
end
|
||||||
|
|
||||||
def metadata(address)
|
def check_service_available
|
||||||
"[[\"text/identifier\", \"#{address}\"], [\"text/plain\", \"Send sats, receive thanks.\"]]"
|
http_status :not_found unless Setting.lndhub_enabled?
|
||||||
end
|
end
|
||||||
|
|
||||||
def valid_amount?(amount_in_sats)
|
def find_user
|
||||||
amount_in_sats <= MAX_SATS && amount_in_sats >= MIN_SATS
|
@user = User.where(cn: params[:username], ou: Setting.primary_domain).first
|
||||||
end
|
http_status :not_found if @user.nil?
|
||||||
|
end
|
||||||
|
|
||||||
def valid_comment?(comment)
|
def metadata(address)
|
||||||
comment.length <= MAX_COMMENT_CHARS
|
"[[\"text/identifier\",\"#{address}\"],[\"text/plain\",\"Sats for #{address}\"]]"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def valid_amount?(amount_in_sats)
|
||||||
|
amount_in_sats <= MAX_SATS && amount_in_sats >= MIN_SATS
|
||||||
|
end
|
||||||
|
|
||||||
|
def valid_comment?(comment)
|
||||||
|
comment.length <= MAX_COMMENT_CHARS
|
||||||
|
end
|
||||||
|
|
||||||
|
def handle_pay_request(address, amount, comment)
|
||||||
|
if !valid_comment?(comment)
|
||||||
|
render json: { status: "ERROR", reason: "Comment too long" }
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
desc = "To #{address}"
|
||||||
|
desc = "#{desc}: \"#{comment}\"" if comment.present?
|
||||||
|
|
||||||
|
invoice = LndhubManager::CreateUserInvoice.call(
|
||||||
|
user: @user, payload: {
|
||||||
|
amount: amount, # sats
|
||||||
|
description: desc,
|
||||||
|
description_hash: Digest::SHA256.hexdigest(metadata(address)),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
render json: {
|
||||||
|
status: "OK",
|
||||||
|
successAction: {
|
||||||
|
tag: "message",
|
||||||
|
message: "Sats received. Thank you!"
|
||||||
|
},
|
||||||
|
routes: [],
|
||||||
|
pr: invoice["payment_request"]
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def nostr_event_from_payload(nostr_param)
|
||||||
|
event_obj = JSON.parse(nostr_param).transform_keys(&:to_sym)
|
||||||
|
Nostr::Event.new(**event_obj)
|
||||||
|
rescue => e
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
|
||||||
|
def valid_zap_request?(amount, event, lnurl)
|
||||||
|
NostrManager::VerifyZapRequest.call(
|
||||||
|
amount: amount, event: event, lnurl: lnurl
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
def handle_zap_request(amount, nostr_param, lnurl_param)
|
||||||
|
event = nostr_event_from_payload(nostr_param)
|
||||||
|
|
||||||
|
unless event.present? && valid_zap_request?(amount*1000, event, lnurl_param)
|
||||||
|
render json: { status: "ERROR", reason: "Invalid zap request" }
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
# TODO might want to use the existing invoice and zap record if there are
|
||||||
|
# multiple calls with the same zap request
|
||||||
|
|
||||||
|
desc = "Zap for #{@user.address}"
|
||||||
|
desc = "#{desc}: \"#{event.content}\"" if event.content.present?
|
||||||
|
|
||||||
|
invoice = LndhubManager::CreateUserInvoice.call(
|
||||||
|
user: @user, payload: {
|
||||||
|
amount: amount, # sats
|
||||||
|
description: desc,
|
||||||
|
description_hash: Digest::SHA256.hexdigest(event.to_json),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
@user.zaps.create! request: event,
|
||||||
|
payment_request: invoice["payment_request"],
|
||||||
|
amount: amount
|
||||||
|
|
||||||
|
render json: { status: "OK", pr: invoice["payment_request"] }
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
131
app/controllers/rs/oauth_controller.rb
Normal file
131
app/controllers/rs/oauth_controller.rb
Normal file
@@ -0,0 +1,131 @@
|
|||||||
|
class Rs::OauthController < ApplicationController
|
||||||
|
before_action :require_signed_in_with_username, only: :new
|
||||||
|
before_action :authenticate_user!, only: :create
|
||||||
|
|
||||||
|
def new
|
||||||
|
@user = User.where(cn: params[:username].downcase, ou: Setting.primary_domain).first
|
||||||
|
@scopes = parse_scopes params[:scope]
|
||||||
|
@redirect_uri = params[:redirect_uri]
|
||||||
|
@client_id = params[:client_id]
|
||||||
|
@state = params[:state]
|
||||||
|
@root_access_requested = (@scopes & [":r",":rw"]).any?
|
||||||
|
|
||||||
|
@denial_url = url_with_state("#{@redirect_uri}#error=access_denied", @state)
|
||||||
|
|
||||||
|
@expire_at_dates = [["Never", nil],
|
||||||
|
["In 1 month", 1.month.from_now],
|
||||||
|
["In 1 day", 1.day.from_now]]
|
||||||
|
|
||||||
|
http_status :bad_request and return unless @redirect_uri.present?
|
||||||
|
|
||||||
|
unless current_user == @user
|
||||||
|
sign_out :user
|
||||||
|
|
||||||
|
redirect_to new_rs_oauth_url(@user.cn,
|
||||||
|
scope: params[:scope],
|
||||||
|
redirect_uri: params[:redirect_uri],
|
||||||
|
client_id: params[:client_id],
|
||||||
|
state: params[:state])
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
unless @client_id.present?
|
||||||
|
redirect_to(url_with_state("#{@redirect_uri}#error=invalid_request", @state),
|
||||||
|
allow_other_host: true) and return
|
||||||
|
end
|
||||||
|
|
||||||
|
if @scopes.empty?
|
||||||
|
redirect_to(url_with_state("#{@redirect_uri}#error=invalid_scope", @state),
|
||||||
|
allow_other_host: true) and return
|
||||||
|
end
|
||||||
|
|
||||||
|
unless hostname_of(@client_id) == hostname_of(@redirect_uri)
|
||||||
|
redirect_to(url_with_state("#{@redirect_uri}#error=invalid_client", @state),
|
||||||
|
allow_other_host: true) and return
|
||||||
|
end
|
||||||
|
|
||||||
|
@client_id.gsub!(/http(s)?:\/\//, "")
|
||||||
|
|
||||||
|
if auth = current_user.remote_storage_authorizations.valid.where(permissions: @scopes, client_id: @client_id).first
|
||||||
|
redirect_to(url_with_state("#{@redirect_uri}#access_token=#{auth.token}", @state),
|
||||||
|
allow_other_host: true) and return
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def create
|
||||||
|
unless current_user.id.to_s == params[:user_id]
|
||||||
|
Rails.logger.info("NO MATCH: #{params[:user_id]}, #{current_user.id}")
|
||||||
|
http_status :forbidden and return
|
||||||
|
end
|
||||||
|
|
||||||
|
permissions = parse_scopes params[:scope]
|
||||||
|
redirect_uri = params[:redirect_uri].presence
|
||||||
|
client_id = params[:client_id].presence
|
||||||
|
state = params[:state].presence
|
||||||
|
expire_at = params[:expire_at].presence
|
||||||
|
|
||||||
|
http_status :bad_request and return unless redirect_uri.present?
|
||||||
|
|
||||||
|
if permissions.empty?
|
||||||
|
redirect_to(url_with_state("#{redirect_uri}#error=invalid_scope", state),
|
||||||
|
allow_other_host: true) and return
|
||||||
|
end
|
||||||
|
|
||||||
|
unless client_id.present?
|
||||||
|
redirect_to(url_with_state("#{redirect_uri}#error=invalid_request", state),
|
||||||
|
allow_other_host: true) and return
|
||||||
|
end
|
||||||
|
|
||||||
|
unless hostname_of(client_id) == hostname_of(redirect_uri)
|
||||||
|
redirect_to(url_with_state("#{redirect_uri}#error=invalid_client", state),
|
||||||
|
allow_other_host: true) and return
|
||||||
|
end
|
||||||
|
|
||||||
|
client_id.gsub!(/http(s)?:\/\//, "")
|
||||||
|
|
||||||
|
auth = current_user.remote_storage_authorizations.create!(
|
||||||
|
permissions: permissions,
|
||||||
|
client_id: client_id,
|
||||||
|
redirect_uri: redirect_uri,
|
||||||
|
app_name: client_id,
|
||||||
|
expire_at: expire_at
|
||||||
|
)
|
||||||
|
|
||||||
|
redirect_to url_with_state("#{redirect_uri}#access_token=#{auth.token}", state),
|
||||||
|
allow_other_host: true
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def require_signed_in_with_username
|
||||||
|
unless user_signed_in?
|
||||||
|
session[:user_return_to] = request.url
|
||||||
|
redirect_to new_user_session_path(cn: params[:username], ou: Setting.primary_domain)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def hostname_of(uri)
|
||||||
|
uri.gsub(/http(s)?:\/\//, "").split(":")[0].split("/")[0]
|
||||||
|
end
|
||||||
|
|
||||||
|
def parse_scopes(scope_string)
|
||||||
|
return [] if scope_string.blank?
|
||||||
|
|
||||||
|
scopes = scope_string.
|
||||||
|
gsub(/\[|\]/, "").
|
||||||
|
gsub(/\,/, " ").
|
||||||
|
gsub(/\/:/, ":").
|
||||||
|
split(/\s/).map(&:strip).
|
||||||
|
reject(&:empty?)
|
||||||
|
|
||||||
|
scopes = [":r"] if scopes.include?("*:r")
|
||||||
|
scopes = [":rw"] if scopes.include?("*:rw")
|
||||||
|
|
||||||
|
scopes
|
||||||
|
end
|
||||||
|
|
||||||
|
def url_with_state(url, state)
|
||||||
|
state ? "#{url}&state=#{CGI.escape(state)}" : url
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
class SecurityController < ApplicationController
|
|
||||||
before_action :require_user_signed_in
|
|
||||||
|
|
||||||
def index
|
|
||||||
@current_section = :security
|
|
||||||
end
|
|
||||||
end
|
|
||||||
9
app/controllers/services/base_controller.rb
Normal file
9
app/controllers/services/base_controller.rb
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
class Services::BaseController < ApplicationController
|
||||||
|
before_action :set_current_section
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_current_section
|
||||||
|
@current_section = :services
|
||||||
|
end
|
||||||
|
end
|
||||||
14
app/controllers/services/chat_controller.rb
Normal file
14
app/controllers/services/chat_controller.rb
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
class Services::ChatController < Services::BaseController
|
||||||
|
before_action :authenticate_user!
|
||||||
|
before_action :require_service_available
|
||||||
|
|
||||||
|
def show
|
||||||
|
@service_enabled = current_user.service_enabled?(:ejabberd)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def require_service_available
|
||||||
|
http_status :not_found unless Setting.ejabberd_enabled?
|
||||||
|
end
|
||||||
|
end
|
||||||
34
app/controllers/services/email_controller.rb
Normal file
34
app/controllers/services/email_controller.rb
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
class Services::EmailController < Services::BaseController
|
||||||
|
before_action :authenticate_user!
|
||||||
|
before_action :require_service_available
|
||||||
|
before_action :require_feature_enabled
|
||||||
|
|
||||||
|
def show
|
||||||
|
ldap_entry = current_user.ldap_entry
|
||||||
|
|
||||||
|
@service_enabled = ldap_entry[:email_password].present?
|
||||||
|
@maildrop = ldap_entry[:email_maildrop]
|
||||||
|
@email_forwarding_active = @maildrop.present? &&
|
||||||
|
@maildrop.split("@").first != current_user.cn
|
||||||
|
end
|
||||||
|
|
||||||
|
def new_password
|
||||||
|
if session[:new_email_password].present?
|
||||||
|
@new_password = session.delete(:new_email_password)
|
||||||
|
else
|
||||||
|
redirect_to setting_path(:email)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def require_service_available
|
||||||
|
http_status :not_found unless Setting.email_enabled?
|
||||||
|
end
|
||||||
|
|
||||||
|
def require_feature_enabled
|
||||||
|
unless Flipper.enabled?(:email, current_user)
|
||||||
|
http_status :forbidden
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user