mirror of
https://github.com/signalapp/Signal-Server.git
synced 2025-12-11 01:40:22 +00:00
Compare commits
2336 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
17a3c90286 | ||
|
|
6341770768 | ||
|
|
308437ec93 | ||
|
|
d3d4916d6c | ||
|
|
d2fa00f0c6 | ||
|
|
d6c9652a70 | ||
|
|
0d20b73e76 | ||
|
|
3c655cdd5a | ||
|
|
fc5cd3a9ca | ||
|
|
83ab926f96 | ||
|
|
56e54e0724 | ||
|
|
544e4fb89a | ||
|
|
966c3a8f47 | ||
|
|
c2ab72c77e | ||
|
|
4468ee3142 | ||
|
|
c82c2c0ba4 | ||
|
|
6e595a0959 | ||
|
|
a79d709039 | ||
|
|
538a07542e | ||
|
|
07ed765250 | ||
|
|
2e497b5834 | ||
|
|
61b3cecd17 | ||
|
|
a4a666bb80 | ||
|
|
c14621a09f | ||
|
|
d0a8899daf | ||
|
|
65dbcb3e5f | ||
|
|
7f725b67c4 | ||
|
|
e25252dc69 | ||
|
|
8b65c11e1e | ||
|
|
320c5eac53 | ||
|
|
8199e0d2d5 | ||
|
|
53387f5a0c | ||
|
|
7d171a79d7 | ||
|
|
3b99bb9e78 | ||
|
|
132f026c75 | ||
|
|
abd0f9630c | ||
|
|
a4508ec84f | ||
|
|
6119b6ab89 | ||
|
|
307ac47ce0 | ||
|
|
4032ddd4fd | ||
|
|
98c8dc05f1 | ||
|
|
4c677ec2da | ||
|
|
c05692e417 | ||
|
|
1e7aa89664 | ||
|
|
ae1edf3c5c | ||
|
|
b17f41c3e8 | ||
|
|
08db4ba54b | ||
|
|
cb6cc39679 | ||
|
|
b6bf6c994c | ||
|
|
3bb4709563 | ||
|
|
b280c768a4 | ||
|
|
d23e89fb9c | ||
|
|
3a27bd0318 | ||
|
|
616513edaf | ||
|
|
09a51020e9 | ||
|
|
cb8cb94d1a | ||
|
|
2440dc0089 | ||
|
|
2336eef333 | ||
|
|
a0e948627c | ||
|
|
88159af588 | ||
|
|
38b77bb550 | ||
|
|
e72d1d0b6f | ||
|
|
1891622e69 | ||
|
|
628a112b38 | ||
|
|
50f5d760c9 | ||
|
|
7292a88ea3 | ||
|
|
07cb3ab576 | ||
|
|
27b749abbd | ||
|
|
27f67a077c | ||
|
|
393e15815b | ||
|
|
a7f1cd25b9 | ||
|
|
953cd2ae0c | ||
|
|
a84a7dbc3d | ||
|
|
24d01f1ab2 | ||
|
|
06eb890761 | ||
|
|
6d0345d327 | ||
|
|
1c67233eb0 | ||
|
|
b4281c5a70 | ||
|
|
5f6b66dad6 | ||
|
|
c2be0af9d9 | ||
|
|
c111e9a35a | ||
|
|
a53a85d788 | ||
|
|
a44c18e9b7 | ||
|
|
4d78437fe4 | ||
|
|
2bfe2c8ff8 | ||
|
|
65da844d70 | ||
|
|
5275c27ee1 | ||
|
|
390580a19d | ||
|
|
147917454f | ||
|
|
39562775d9 | ||
|
|
4a0ef1f834 | ||
|
|
85b16b674d | ||
|
|
ab5d8ba120 | ||
|
|
28076335e0 | ||
|
|
9e9333424f | ||
|
|
6f0faae4ce | ||
|
|
0d24828539 | ||
|
|
0a6d724f2c | ||
|
|
8956e1e0cf | ||
|
|
c9ae991aa3 | ||
|
|
421d594507 | ||
|
|
9c03f2e468 | ||
|
|
1175ff5867 | ||
|
|
55df593561 | ||
|
|
a06a663b94 | ||
|
|
3d2f7e731f | ||
|
|
2575372639 | ||
|
|
faa6e8324a | ||
|
|
d0e3fb1901 | ||
|
|
04287c5073 | ||
|
|
0c76fdd36c | ||
|
|
d582942244 | ||
|
|
3636626e09 | ||
|
|
3e0919106d | ||
|
|
d385838dc1 | ||
|
|
e28f1e8ceb | ||
|
|
3d875f1ce5 | ||
|
|
c4c5397b44 | ||
|
|
6b6f9b2405 | ||
|
|
7d4a8d03a4 | ||
|
|
e9119da040 | ||
|
|
aa36dc95ef | ||
|
|
a6f9409a39 | ||
|
|
41a113e22c | ||
|
|
4cfcdb0c96 | ||
|
|
36050f580e | ||
|
|
98760b631b | ||
|
|
d00aa1e77a | ||
|
|
dce391a248 | ||
|
|
c252118cfc | ||
|
|
e9fd32de79 | ||
|
|
788246a56f | ||
|
|
bc02fe3831 | ||
|
|
d290aad27b | ||
|
|
6754ec5e10 | ||
|
|
1ba00a66eb | ||
|
|
1dd7d33e23 | ||
|
|
e200548e35 | ||
|
|
fdf7b69996 | ||
|
|
92d36b725f | ||
|
|
4e131858ca | ||
|
|
a45d95905e | ||
|
|
0fdfdabf2a | ||
|
|
a25e967978 | ||
|
|
38e30c7513 | ||
|
|
e38e5fa17d | ||
|
|
c1f9bedf2f | ||
|
|
dd5d0ea2b3 | ||
|
|
42fd29d38b | ||
|
|
bf6d3aa324 | ||
|
|
023ccc6563 | ||
|
|
da49db5b9e | ||
|
|
cc8dda28cc | ||
|
|
47300c1d44 | ||
|
|
d31550d444 | ||
|
|
51f37350eb | ||
|
|
ecfa161da8 | ||
|
|
e3778c17ea | ||
|
|
cbc95415b7 | ||
|
|
776c0aa488 | ||
|
|
327eb0219d | ||
|
|
8507b6a1f0 | ||
|
|
a853748303 | ||
|
|
192e884e4a | ||
|
|
7001ad1445 | ||
|
|
5cfb133f79 | ||
|
|
5df24edebf | ||
|
|
95d0293a96 | ||
|
|
f5a2efb57c | ||
|
|
e4b9ae4eee | ||
|
|
bc1ac5a37f | ||
|
|
96ac56faac | ||
|
|
f0bc444388 | ||
|
|
8584f47d95 | ||
|
|
f6235b8c08 | ||
|
|
d452e90470 | ||
|
|
418a869451 | ||
|
|
cf89e2215c | ||
|
|
a4ca1ef1a8 | ||
|
|
c38572307d | ||
|
|
20902df122 | ||
|
|
d31ddb72f3 | ||
|
|
d5f2d86bd2 | ||
|
|
2ce8bcd565 | ||
|
|
75c92eaa93 | ||
|
|
0445adcac3 | ||
|
|
c45ff61954 | ||
|
|
06dd4c5026 | ||
|
|
058caadf4f | ||
|
|
7b7d309105 | ||
|
|
63be7b93ce | ||
|
|
578ea12b59 | ||
|
|
364e59be57 | ||
|
|
fece4dac9e | ||
|
|
ce85c1aabc | ||
|
|
0ac2ce5e72 | ||
|
|
391c800bf5 | ||
|
|
9c27b58194 | ||
|
|
f6471cf8f9 | ||
|
|
f21e9bcc4d | ||
|
|
1eaff753a6 | ||
|
|
9b3a8897cd | ||
|
|
40f8cddfb2 | ||
|
|
c29d5de1eb | ||
|
|
d94c171d63 | ||
|
|
2717967d61 | ||
|
|
53203dbcef | ||
|
|
9e66f8ac11 | ||
|
|
796fb3b4cd | ||
|
|
473ecbdf2d | ||
|
|
7b3703506b | ||
|
|
5816f76bbe | ||
|
|
355996bafc | ||
|
|
c2bb46f41d | ||
|
|
12f76c24b1 | ||
|
|
4b8ebc9a17 | ||
|
|
42a109e593 | ||
|
|
8064e68873 | ||
|
|
3dc0d0bb92 | ||
|
|
2bb8f92af1 | ||
|
|
5b7d5d2b93 | ||
|
|
2b27db18d8 | ||
|
|
f3c811cc03 | ||
|
|
df415208a4 | ||
|
|
77fd01bd9f | ||
|
|
fa3a9570d6 | ||
|
|
c06a5ac96c | ||
|
|
33467b42da | ||
|
|
13fb641113 | ||
|
|
53f17c2baa | ||
|
|
06a57ef811 | ||
|
|
86a09b16ff | ||
|
|
c70d7535b9 | ||
|
|
8541360bf3 | ||
|
|
2a832d36d7 | ||
|
|
1578c89475 | ||
|
|
5c13e54149 | ||
|
|
8e74cf6633 | ||
|
|
bab6b36e4d | ||
|
|
f75e616397 | ||
|
|
941a9c3b39 | ||
|
|
7ba0f604e6 | ||
|
|
cf8a4cc939 | ||
|
|
ee78daeeef | ||
|
|
2f6b0b1a55 | ||
|
|
c048074c31 | ||
|
|
5ca89709e3 | ||
|
|
5a88ff0811 | ||
|
|
de68c251f8 | ||
|
|
7c9ae3561d | ||
|
|
b608ece57e | ||
|
|
8dfffebaf1 | ||
|
|
109a3bb2b9 | ||
|
|
fef37f739b | ||
|
|
7a5615182a | ||
|
|
02a7003ffe | ||
|
|
1571f14815 | ||
|
|
9cb098ad8a | ||
|
|
6283f5952d | ||
|
|
9b9edbae0e | ||
|
|
491155d1cf | ||
|
|
54207254f1 | ||
|
|
1395dcc0be | ||
|
|
2a68d9095d | ||
|
|
a984b3640e | ||
|
|
f9c1e411aa | ||
|
|
f6cbc32ee7 | ||
|
|
602614acf6 | ||
|
|
3854b7d472 | ||
|
|
5e25481088 | ||
|
|
fe86e15d80 | ||
|
|
179b4a69eb | ||
|
|
eee6307789 | ||
|
|
9fc5002619 | ||
|
|
faa6ae284a | ||
|
|
8b4355b21d | ||
|
|
e8835da740 | ||
|
|
75854e104e | ||
|
|
93d06e3f4d | ||
|
|
c560b9229c | ||
|
|
935e268dec | ||
|
|
3a1c716c73 | ||
|
|
f3457502a6 | ||
|
|
7ded802df4 | ||
|
|
d3cd1d1b15 | ||
|
|
f5a75c6319 | ||
|
|
ae3a5c5f5e | ||
|
|
43792e2426 | ||
|
|
551d639951 | ||
|
|
c367a71223 | ||
|
|
d259ef0348 | ||
|
|
288cbf4a80 | ||
|
|
ba5e5a780f | ||
|
|
73fa3c3fe4 | ||
|
|
579eb85175 | ||
|
|
b2b20072ae | ||
|
|
a2c4d3fe95 | ||
|
|
31e2be2e4d | ||
|
|
9f5d97e1c6 | ||
|
|
baaae6cd9f | ||
|
|
ed398aa7b9 | ||
|
|
6e2ae42dab | ||
|
|
7f832ad783 | ||
|
|
2ce6f8cb6c | ||
|
|
2574125199 | ||
|
|
41bf2b2c42 | ||
|
|
51bac394ec | ||
|
|
b696649c9d | ||
|
|
b4828ad8de | ||
|
|
639d634426 | ||
|
|
5358fc4f43 | ||
|
|
6a654ab90b | ||
|
|
99eda80a78 | ||
|
|
a6182acc9c | ||
|
|
2241e4d8ea | ||
|
|
cbbdea1ba4 | ||
|
|
05e7c98620 | ||
|
|
1f1d618dea | ||
|
|
b18117ef89 | ||
|
|
44cb796574 | ||
|
|
ccf60ffc4b | ||
|
|
f69db11f42 | ||
|
|
96a680dcf0 | ||
|
|
c8367c9b7a | ||
|
|
c612663490 | ||
|
|
de5d967d18 | ||
|
|
7fc63f7847 | ||
|
|
49009cbcad | ||
|
|
b5fbeffb86 | ||
|
|
146655e997 | ||
|
|
87d66f04d8 | ||
|
|
bb27dd0c3b | ||
|
|
f45a1c232f | ||
|
|
d7a3c12bbe | ||
|
|
a1e84f5a88 | ||
|
|
b758737907 | ||
|
|
c488c14d25 | ||
|
|
5e0cca0702 | ||
|
|
8559e46e4a | ||
|
|
4bc00e00e3 | ||
|
|
3e777df86c | ||
|
|
278b4e810d | ||
|
|
346c7cd743 | ||
|
|
867bf97d8f | ||
|
|
8a67949168 | ||
|
|
5baa51d547 | ||
|
|
616db337e1 | ||
|
|
3895871462 | ||
|
|
a87b84fbe2 | ||
|
|
b2f0ace9db | ||
|
|
20c95e2606 | ||
|
|
22dccaeddb | ||
|
|
e611a70ba4 | ||
|
|
66845d7080 | ||
|
|
4ea7278c6f | ||
|
|
2b2e26f14b | ||
|
|
b496ef8d6f | ||
|
|
7f5e83141d | ||
|
|
2d1ca98605 | ||
|
|
eaa4c318e3 | ||
|
|
31373fd1ba | ||
|
|
9086246947 | ||
|
|
7855b70682 | ||
|
|
0ce87153e5 | ||
|
|
dba1711e8d | ||
|
|
a70b057e1c | ||
|
|
9a5ffea0ad | ||
|
|
96f4b771ea | ||
|
|
3df143dd3d | ||
|
|
d78d7c726e | ||
|
|
d0ad580c7d | ||
|
|
4a8a2a70b5 | ||
|
|
20a71b7df2 | ||
|
|
68412b3901 | ||
|
|
93a7c60a15 | ||
|
|
31e5058b15 | ||
|
|
14cff958e9 | ||
|
|
9628f147f1 | ||
|
|
13e346d4eb | ||
|
|
e507ce2f26 | ||
|
|
9c62622733 | ||
|
|
62aa0cef39 | ||
|
|
401953313a | ||
|
|
4d2403d619 | ||
|
|
c5f261305d | ||
|
|
394f58f6cc | ||
|
|
674bf1b0e0 | ||
|
|
606ddd8a9b | ||
|
|
e23a1fac50 | ||
|
|
342323a7e6 | ||
|
|
efb410444b | ||
|
|
17c9b4c5d3 | ||
|
|
706de8e2f1 | ||
|
|
23bc11f3b6 | ||
|
|
4eb7dde1c8 | ||
|
|
064861b930 | ||
|
|
afa910bbd7 | ||
|
|
6aceb24fd2 | ||
|
|
d94e86781f | ||
|
|
0d4a3b1ad4 | ||
|
|
acfcb18f29 | ||
|
|
f7ff8e3837 | ||
|
|
048e17c62b | ||
|
|
d89b4f7e95 | ||
|
|
795b226b90 | ||
|
|
e485c380e0 | ||
|
|
bb4f4bc441 | ||
|
|
65b49b2d9c | ||
|
|
9e7010f185 | ||
|
|
3bb8e5bb00 | ||
|
|
2a4d1da2ca | ||
|
|
6b71b66bd2 | ||
|
|
ebf24fb125 | ||
|
|
46d64b949e | ||
|
|
6919354520 | ||
|
|
a42fe9bfb0 | ||
|
|
ee1f8b34ea | ||
|
|
c910fa406d | ||
|
|
559205e33f | ||
|
|
c0756e9c60 | ||
|
|
bf1190696e | ||
|
|
71dd0890de | ||
|
|
e5acdf1402 | ||
|
|
0f08b6bb59 | ||
|
|
6198a7b69a | ||
|
|
067aee6664 | ||
|
|
138a2ebbd0 | ||
|
|
296f6a7a88 | ||
|
|
069ffa9921 | ||
|
|
f42fd8a840 | ||
|
|
10f27af6f2 | ||
|
|
0bbd34d060 | ||
|
|
282daeb0dc | ||
|
|
d33b313c11 | ||
|
|
fb7316c9ae | ||
|
|
279b0a51d9 | ||
|
|
6547d5ebf3 | ||
|
|
4f1ef9a039 | ||
|
|
4c80714d19 | ||
|
|
077ead71a5 | ||
|
|
caba110266 | ||
|
|
0fdb23c1e9 | ||
|
|
13a84f0c72 | ||
|
|
669bd58e33 | ||
|
|
6e82740a9b | ||
|
|
7ea43a728d | ||
|
|
71b38356b1 | ||
|
|
5a99708f56 | ||
|
|
24191d9599 | ||
|
|
482ea8eb40 | ||
|
|
1dae05651f | ||
|
|
5164e92538 | ||
|
|
f89a20dbc7 | ||
|
|
3a4c5a2bfb | ||
|
|
5e1334e8de | ||
|
|
fa6e3d3690 | ||
|
|
9383e7716b | ||
|
|
cfe34fbf0f | ||
|
|
9fe110625c | ||
|
|
975f753c2b | ||
|
|
e6237480f8 | ||
|
|
966d4e29d4 | ||
|
|
26f876a2cb | ||
|
|
ab9e6ac48a | ||
|
|
c1d6c04ab2 | ||
|
|
888cec3d56 | ||
|
|
1461bcc2c2 | ||
|
|
11f1cf80bd | ||
|
|
c675cc8b26 | ||
|
|
0011b8925b | ||
|
|
73ea6e4251 | ||
|
|
e4441dddbb | ||
|
|
8d1d56f694 | ||
|
|
2015ba77ca | ||
|
|
7033a0f68f | ||
|
|
6ada76da7f | ||
|
|
cbdec0cb22 | ||
|
|
de6e9d31c9 | ||
|
|
f0a6be32fc | ||
|
|
5c4855cca6 | ||
|
|
2e1e380418 | ||
|
|
d07f0b4f71 | ||
|
|
aaa2a6eef1 | ||
|
|
b1f56c3324 | ||
|
|
da5c0ae4b6 | ||
|
|
1e1394560d | ||
|
|
bae0196bcf | ||
|
|
3398955c1a | ||
|
|
a1b925d1e0 | ||
|
|
31c0c3275f | ||
|
|
0a4392f700 | ||
|
|
eb86986cf4 | ||
|
|
1053a47e42 | ||
|
|
99b1f48e0e | ||
|
|
c21eb6aa50 | ||
|
|
6dddf54222 | ||
|
|
9e3eb2319e | ||
|
|
1d8dcda815 | ||
|
|
ee52a84262 | ||
|
|
eb51e81faa | ||
|
|
d41ef1df18 | ||
|
|
66d47aff2c | ||
|
|
c931103712 | ||
|
|
ad1aeea74b | ||
|
|
ae7f8af03e | ||
|
|
a52c91a665 | ||
|
|
94bf3a3902 | ||
|
|
f5a539e128 | ||
|
|
24480b2090 | ||
|
|
a124b3abe9 | ||
|
|
090d722b61 | ||
|
|
d27ec6fe8d | ||
|
|
8d34f3447b | ||
|
|
72b52965b9 | ||
|
|
ae7077c643 | ||
|
|
11598e855f | ||
|
|
534c577f59 | ||
|
|
7762afc497 | ||
|
|
a3fe4b9980 | ||
|
|
598599cd14 | ||
|
|
07cd69ab34 | ||
|
|
3b764bed7a | ||
|
|
c91d5c2fdb | ||
|
|
40f7e6e994 | ||
|
|
ee9aa9ce12 | ||
|
|
08304bf375 | ||
|
|
8b8c6237be | ||
|
|
c0837104cd | ||
|
|
fe21d014f7 | ||
|
|
75c5032cd3 | ||
|
|
f84e7aebd0 | ||
|
|
c379a3d297 | ||
|
|
eedeaaecee | ||
|
|
64eeb1e361 | ||
|
|
e07597eba7 | ||
|
|
5f2656710c | ||
|
|
1af53f2612 | ||
|
|
c89cfa4927 | ||
|
|
bbde93a3c7 | ||
|
|
b01b76d78f | ||
|
|
75c22038eb | ||
|
|
3c1705994d | ||
|
|
439d2f5df8 | ||
|
|
d2bc3c7360 | ||
|
|
9734433f00 | ||
|
|
5bd08800bb | ||
|
|
3032415141 | ||
|
|
ba58a95a0f | ||
|
|
aa4bd92fee | ||
|
|
f37c76dab1 | ||
|
|
863969c77c | ||
|
|
2383aaaa3d | ||
|
|
715d1157ad | ||
|
|
4aaae3f445 | ||
|
|
8359ef73f4 | ||
|
|
c6bb649adb | ||
|
|
e333cbd94d | ||
|
|
cc9a825279 | ||
|
|
5189cbe5c7 | ||
|
|
d1d6e5c652 | ||
|
|
3e5087e60b | ||
|
|
93c3cea912 | ||
|
|
e824b861d4 | ||
|
|
e8dd1e0bf2 | ||
|
|
533afa4c6e | ||
|
|
559026933d | ||
|
|
7864405efd | ||
|
|
a5575902de | ||
|
|
5b9bce59e1 | ||
|
|
041aed2d72 | ||
|
|
02a296e500 | ||
|
|
98e41f9a37 | ||
|
|
6a71d369e2 | ||
|
|
75661fa800 | ||
|
|
df5498e1c0 | ||
|
|
c0af911197 | ||
|
|
44bc90e5ab | ||
|
|
5c1cde1b28 | ||
|
|
3172b571c6 | ||
|
|
17e8b77e88 | ||
|
|
8011935a3b | ||
|
|
3f3052c23c | ||
|
|
8f17f45339 | ||
|
|
009e2eeb97 | ||
|
|
c70fa48835 | ||
|
|
bd5f5c407b | ||
|
|
2bc573a53d | ||
|
|
537d61d5bd | ||
|
|
09519ae942 | ||
|
|
2a67b2e610 | ||
|
|
8161f55a82 | ||
|
|
ecee189ad8 | ||
|
|
ef0900f3ac | ||
|
|
383d744bd8 | ||
|
|
c2ba8ab562 | ||
|
|
cd49ea43c0 | ||
|
|
53aa45a2bb | ||
|
|
83e0a19561 | ||
|
|
6a5d475198 | ||
|
|
49ccbba2e3 | ||
|
|
41735ed40e | ||
|
|
2d11a433c9 | ||
|
|
e79ab2521f | ||
|
|
fb1f99da87 | ||
|
|
08c6a8c2e5 | ||
|
|
ce3835e176 | ||
|
|
39f6eadbb9 | ||
|
|
16dba09b61 | ||
|
|
d5ebf2f2ed | ||
|
|
8a8e6e7b49 | ||
|
|
34e21b9f7b | ||
|
|
98a31d1474 | ||
|
|
72a0c1be0f | ||
|
|
5b25e38e41 | ||
|
|
2fb400280b | ||
|
|
79ad09524e | ||
|
|
5f8accb492 | ||
|
|
6fcadc2297 | ||
|
|
3f4e1522eb | ||
|
|
6304c84cdb | ||
|
|
894297efa9 | ||
|
|
a51a7a0901 | ||
|
|
372e131e25 | ||
|
|
6c6e6a4975 | ||
|
|
cd66a1ceb7 | ||
|
|
feb59deb28 | ||
|
|
489519a982 | ||
|
|
a96865d0f5 | ||
|
|
12e11609a9 | ||
|
|
5b404095b0 | ||
|
|
6a6555e2d5 | ||
|
|
49489a6021 | ||
|
|
8cd93d68e4 | ||
|
|
f3b9a8d97f | ||
|
|
b91a69d8b3 | ||
|
|
624e40e3b7 | ||
|
|
23a076a204 | ||
|
|
016141a05d | ||
|
|
a064b25a14 | ||
|
|
bd40e32f3b | ||
|
|
81a21c0d5f | ||
|
|
6478210330 | ||
|
|
aa1c37fe26 | ||
|
|
6ee23b0186 | ||
|
|
40eb445592 | ||
|
|
ce7d687205 | ||
|
|
758900b7a8 | ||
|
|
539b62a829 | ||
|
|
2866f1b213 | ||
|
|
fc1465c05d | ||
|
|
bc887ec6fa | ||
|
|
84b3d324bb | ||
|
|
fc10108788 | ||
|
|
fbbc1bec58 | ||
|
|
2059bb5ef8 | ||
|
|
b080a5db4d | ||
|
|
b4aabd799b | ||
|
|
92f035bc2a | ||
|
|
18a6df34bd | ||
|
|
b1274125c9 | ||
|
|
dceebc1c8d | ||
|
|
6aadb4b458 | ||
|
|
703405b874 | ||
|
|
d1735c7e57 | ||
|
|
1f815b49dd | ||
|
|
a9339b7037 | ||
|
|
f2c6ca182d | ||
|
|
b946c27a20 | ||
|
|
9fd6358518 | ||
|
|
8a8a848fac | ||
|
|
aeb9f67266 | ||
|
|
e08c5a412e | ||
|
|
a7443a9ece | ||
|
|
54fe3b9a43 | ||
|
|
ba522b1691 | ||
|
|
739c5bf22c | ||
|
|
7cdadeb791 | ||
|
|
dadf43b93e | ||
|
|
bd820e6d2e | ||
|
|
19f7b207b7 | ||
|
|
a398e2269c | ||
|
|
2e28fb97a4 | ||
|
|
5c68d83a93 | ||
|
|
0b7c3ad745 | ||
|
|
0cde06557d | ||
|
|
27844fe692 | ||
|
|
779051ef9f | ||
|
|
d13741fbd5 | ||
|
|
f7f870fe62 | ||
|
|
de59aa099d | ||
|
|
57a478b898 | ||
|
|
3e8d79e147 | ||
|
|
a46045d987 | ||
|
|
662c905b80 | ||
|
|
31022aeb79 | ||
|
|
b3e6a50dee | ||
|
|
d29764d11f | ||
|
|
f8e4f6727a | ||
|
|
63d05df8a3 | ||
|
|
52d13d1d62 | ||
|
|
f58a320223 | ||
|
|
3e01bc1174 | ||
|
|
d1ada7f998 | ||
|
|
095fc8140e | ||
|
|
73c368ea86 | ||
|
|
ce5edbb7fc | ||
|
|
a680639718 | ||
|
|
becf6afbdd | ||
|
|
1dda015c6a | ||
|
|
a0427ecf8c | ||
|
|
cfd31e98ff | ||
|
|
bcb89924b4 | ||
|
|
23f9199439 | ||
|
|
1f6318a919 | ||
|
|
b0667b258b | ||
|
|
4c3a48f5be | ||
|
|
33fb7a72de | ||
|
|
2c808e369c | ||
|
|
906d0be382 | ||
|
|
1c9a3c6105 | ||
|
|
2aaddd721f | ||
|
|
4e2284b83f | ||
|
|
d5d9978e48 | ||
|
|
d45659ac76 | ||
|
|
13a07dc6cd | ||
|
|
51b7a8d868 | ||
|
|
df9c0051c9 | ||
|
|
331ff83cd5 | ||
|
|
44838d6238 | ||
|
|
5400abb065 | ||
|
|
f47fefb73e | ||
|
|
cdef745a7a | ||
|
|
1a1eab4ec0 | ||
|
|
3a966ef345 | ||
|
|
be20c04cd8 | ||
|
|
d09dcc90fe | ||
|
|
1fd1207bf6 | ||
|
|
0117fc12c7 | ||
|
|
ef9a7fda9a | ||
|
|
13447df1e0 | ||
|
|
3608c5bfb0 | ||
|
|
34dbff6786 | ||
|
|
a6066bfc2f | ||
|
|
8579190cdf | ||
|
|
917f667229 | ||
|
|
317a551bdb | ||
|
|
27e9271473 | ||
|
|
11dff6c546 | ||
|
|
e6712937ca | ||
|
|
cf8887bb5a | ||
|
|
696340f780 | ||
|
|
86ddcbaa08 | ||
|
|
2144d2a8d8 | ||
|
|
f7af861b31 | ||
|
|
208a09b3ae | ||
|
|
831023e41d | ||
|
|
ff627793d6 | ||
|
|
f971c76a99 | ||
|
|
8f41176c76 | ||
|
|
31bbbbb5e0 | ||
|
|
effcd6038d | ||
|
|
12be7d49c2 | ||
|
|
14863b575e | ||
|
|
32a95f96ff | ||
|
|
b757d4b334 | ||
|
|
bd03d910fe | ||
|
|
01ef855157 | ||
|
|
817866caf3 | ||
|
|
158d65c6a7 | ||
|
|
62022c7de1 | ||
|
|
a824b5575d | ||
|
|
78819d5382 | ||
|
|
d128bc782a | ||
|
|
530b2a310f | ||
|
|
d5b0d99a54 | ||
|
|
43be72d076 | ||
|
|
9558944e22 | ||
|
|
0f6c866c8d | ||
|
|
bac78e9291 | ||
|
|
c22ea78672 | ||
|
|
a85afe827d | ||
|
|
abaed821ec | ||
|
|
6fa9dcd954 | ||
|
|
819d59cd79 | ||
|
|
2f88f0eedb | ||
|
|
74ff491671 | ||
|
|
eac48a6617 | ||
|
|
19617c14f8 | ||
|
|
fc7291c3e8 | ||
|
|
88db808298 | ||
|
|
5193abdab3 | ||
|
|
a315c9be92 | ||
|
|
fc1541591a | ||
|
|
ae97c4db9f | ||
|
|
26bc5973b5 | ||
|
|
e52b8c8423 | ||
|
|
7395489bac | ||
|
|
b384ed7f5c | ||
|
|
e3afcae7d3 | ||
|
|
9faeed7b20 | ||
|
|
49adcca80e | ||
|
|
49c43a6816 | ||
|
|
84f85ae098 | ||
|
|
3d581941ab | ||
|
|
d2d39baede | ||
|
|
111f5ba024 | ||
|
|
ce3fb7fa99 | ||
|
|
fc421d3f21 | ||
|
|
71bea759c6 | ||
|
|
bf1dd791a5 | ||
|
|
4c99577c08 | ||
|
|
5d5c63e6d4 | ||
|
|
42ff3f8432 | ||
|
|
be6ef76486 | ||
|
|
bc297e6d34 | ||
|
|
3a526dcbd7 | ||
|
|
7883352b74 | ||
|
|
982d122d18 | ||
|
|
d8d94407c6 | ||
|
|
28cfc54170 | ||
|
|
2ee7279743 | ||
|
|
eb1b073385 | ||
|
|
c634185b6f | ||
|
|
827a3af419 | ||
|
|
2c33d22a30 | ||
|
|
b41ed9d810 | ||
|
|
58d3a12eff | ||
|
|
88c4b2be97 | ||
|
|
6cbd57f19f | ||
|
|
5522376584 | ||
|
|
5089c37d28 | ||
|
|
1ccf24e68c | ||
|
|
411f7298f2 | ||
|
|
5b0214c6f2 | ||
|
|
735573e61b | ||
|
|
c545cff1b3 | ||
|
|
cbd9681e3e | ||
|
|
ca876e40ca | ||
|
|
76f5a71727 | ||
|
|
117de2382d | ||
|
|
25e7036451 | ||
|
|
3131bd3dd9 | ||
|
|
1cf9397bbd | ||
|
|
c97be15e79 | ||
|
|
164fc40990 | ||
|
|
6456af6284 | ||
|
|
81212cc13a | ||
|
|
6f0750790c | ||
|
|
3e61b5c49d | ||
|
|
50c4df4f45 | ||
|
|
1eb946f5fe | ||
|
|
7bd402b48d | ||
|
|
90444d5b91 | ||
|
|
5ee093f87c | ||
|
|
623743286c | ||
|
|
67067f1d2d | ||
|
|
07f9bb112e | ||
|
|
417d48c452 | ||
|
|
358412c78a | ||
|
|
215621a9b0 | ||
|
|
c3f53c4dd9 | ||
|
|
01514f83a0 | ||
|
|
c10b64c367 | ||
|
|
722055c8b5 | ||
|
|
680e501f83 | ||
|
|
f13f7a5ff4 | ||
|
|
5290656c3b | ||
|
|
93fbb87741 | ||
|
|
ce76c5c117 | ||
|
|
e663e1b0a6 | ||
|
|
20cdd09171 | ||
|
|
f98dd80941 | ||
|
|
f84736bd32 | ||
|
|
9995f271c8 | ||
|
|
cf59d849b0 | ||
|
|
ee3b91e4fb | ||
|
|
77f134ddca | ||
|
|
8913192b7e | ||
|
|
94ac3f6cc8 | ||
|
|
b89de860d3 | ||
|
|
f8c623074b | ||
|
|
1160af9522 | ||
|
|
3056ea8cbc | ||
|
|
28e3b23e8c | ||
|
|
fbaf4a09e2 | ||
|
|
cfa8cbedc1 | ||
|
|
be4c46e674 | ||
|
|
bacf524ae6 | ||
|
|
aa65d34c36 | ||
|
|
0cd3640f13 | ||
|
|
c595d9415c | ||
|
|
1a604d8c79 | ||
|
|
f76e6705c0 | ||
|
|
10cd60738a | ||
|
|
89470ff536 | ||
|
|
79b8202452 | ||
|
|
d252e579f4 | ||
|
|
30b2c2b5ad | ||
|
|
282f39141e | ||
|
|
85e4de6933 | ||
|
|
0b993098a8 | ||
|
|
1880773fb9 | ||
|
|
00c9023e74 | ||
|
|
d59eabd9d7 | ||
|
|
2a3ea13c9e | ||
|
|
6906336dfb | ||
|
|
514b94a5cb | ||
|
|
df01be2dca | ||
|
|
10c6f885fd | ||
|
|
224e6dac31 | ||
|
|
3b1eb3a9db | ||
|
|
e320626c6e | ||
|
|
03dac2bf7e | ||
|
|
730303567f | ||
|
|
57ff9f86f5 | ||
|
|
bfd2c32d4e | ||
|
|
e9a3d52d7f | ||
|
|
ac7eb88194 | ||
|
|
d45154f2aa | ||
|
|
760462f8fb | ||
|
|
1999bd2bcb | ||
|
|
46110d4d65 | ||
|
|
5752853bba | ||
|
|
02d06af3fc | ||
|
|
09e0934eaf | ||
|
|
b100f09205 | ||
|
|
670b69df24 | ||
|
|
03a531e1b0 | ||
|
|
13ecbe7e53 | ||
|
|
17047513c3 | ||
|
|
7bd7d0e84e | ||
|
|
4571042814 | ||
|
|
9cb89b42bf | ||
|
|
9a4453c414 | ||
|
|
5fa22bc073 | ||
|
|
bf32b766a5 | ||
|
|
f0a8b5a54a | ||
|
|
8e68e0e037 | ||
|
|
b81b811400 | ||
|
|
b41f97233e | ||
|
|
350de1c759 | ||
|
|
055e8d80a1 | ||
|
|
dfb8a419e7 | ||
|
|
030a791d69 | ||
|
|
cf495ef7cf | ||
|
|
8fdbcbef44 | ||
|
|
30c9968928 | ||
|
|
f357ad098f | ||
|
|
1a8c40c02a | ||
|
|
20677d4be1 | ||
|
|
c448c37cc9 | ||
|
|
f117d9ff4d | ||
|
|
2dbd7ffc75 | ||
|
|
fac4538f6f | ||
|
|
01e526af25 | ||
|
|
7e805d1592 | ||
|
|
c63bebb3e7 | ||
|
|
0e6cfb460d | ||
|
|
cd6b2512e1 | ||
|
|
4f6b132449 | ||
|
|
b7c611a466 | ||
|
|
0163242c8a | ||
|
|
7fa17e33e9 | ||
|
|
e4dbb8efe7 | ||
|
|
89256fb5b3 | ||
|
|
59e401f41e | ||
|
|
4b42dd1db3 | ||
|
|
6196856a7c | ||
|
|
0e8d4f9a61 | ||
|
|
97d2d97ee7 | ||
|
|
62315f423c | ||
|
|
5ee56b022c | ||
|
|
6c37b658ac | ||
|
|
1f53900345 | ||
|
|
deece33a0d | ||
|
|
13053da97f | ||
|
|
4c019aef15 | ||
|
|
bab5e5769b | ||
|
|
f68390e96f | ||
|
|
76cbf734ad | ||
|
|
17ba630014 | ||
|
|
7057476048 | ||
|
|
3121867f72 | ||
|
|
435410b004 | ||
|
|
f190462879 | ||
|
|
7c0ff67625 | ||
|
|
ac72c8b2de | ||
|
|
20208ae528 | ||
|
|
6c6f073bc2 | ||
|
|
0663fe30df | ||
|
|
2c0a75586b | ||
|
|
b6cb23cbb5 | ||
|
|
ee555285ed | ||
|
|
b75456acf3 | ||
|
|
be6d6351b9 | ||
|
|
abafa2ccac | ||
|
|
53e6f419b6 | ||
|
|
b75dec40ac | ||
|
|
0f4f775ee2 | ||
|
|
5974328d9c | ||
|
|
a472774734 | ||
|
|
166d203e8e | ||
|
|
3b3764535c | ||
|
|
f2a1a65a45 | ||
|
|
b7c56108ca | ||
|
|
52478e7de0 | ||
|
|
ae9fd090de | ||
|
|
59bbd0c43c | ||
|
|
f6c9b2b6e7 | ||
|
|
0c0e33bc0e | ||
|
|
4d33ba48cc | ||
|
|
18fb23f27c | ||
|
|
92c25a8373 | ||
|
|
14f5271c20 | ||
|
|
37bda0b035 | ||
|
|
675785a4fd | ||
|
|
0572951c8a | ||
|
|
7d766ee39e | ||
|
|
1f24c913a6 | ||
|
|
2a8806ec2e | ||
|
|
ffcabe6fc4 | ||
|
|
365ad3a4f8 | ||
|
|
2cb788ceb7 | ||
|
|
257fef9734 | ||
|
|
37e0730d2a | ||
|
|
dea359ef91 | ||
|
|
64c9648dd8 | ||
|
|
6dfd13118d | ||
|
|
2f6105f9bc | ||
|
|
5c23f62cec | ||
|
|
ab4e94edab | ||
|
|
9589b7758c | ||
|
|
681cdf8eff | ||
|
|
ad6c271f9d | ||
|
|
c8414a63fb | ||
|
|
c10d9603ad | ||
|
|
91bd061110 | ||
|
|
83aa59f4dd | ||
|
|
3745a0b81d | ||
|
|
e2b093abce | ||
|
|
7e29ed1cc7 | ||
|
|
5965f0fd22 | ||
|
|
c3c46f2f74 | ||
|
|
a816aa0186 | ||
|
|
a7bad20eae | ||
|
|
089b6b1644 | ||
|
|
7509520883 | ||
|
|
9778775046 | ||
|
|
e5ae0572c5 | ||
|
|
63dac3bd9f | ||
|
|
19295eef46 | ||
|
|
0bc1369e04 | ||
|
|
ca2f7d2eed | ||
|
|
3ea535a412 | ||
|
|
a288b9df8e | ||
|
|
8b6012f8a8 | ||
|
|
1e5d7582da | ||
|
|
ad838b4827 | ||
|
|
25f603efc9 | ||
|
|
152c927929 | ||
|
|
b5bd16c6a9 | ||
|
|
14bfa83bb8 | ||
|
|
5dc8086968 | ||
|
|
7118340f12 | ||
|
|
efe7f2e4c1 | ||
|
|
fb2fc2335a | ||
|
|
345e116699 | ||
|
|
e50a1c0646 | ||
|
|
a6fd1aa06c | ||
|
|
3cdc58200a | ||
|
|
933dd81d82 | ||
|
|
a1434524a4 | ||
|
|
cdc6afefe2 | ||
|
|
738ec2a38e | ||
|
|
07886a9722 | ||
|
|
fde1b49729 | ||
|
|
58210141f4 | ||
|
|
e1f35102aa | ||
|
|
af2a8548c3 | ||
|
|
1faedd3870 | ||
|
|
f57a4171ba | ||
|
|
df9dc82de5 | ||
|
|
0573f09285 | ||
|
|
eb6fe11da1 | ||
|
|
823025f3b3 | ||
|
|
0ee3f0a5b5 | ||
|
|
6bff564129 | ||
|
|
7dabc92447 | ||
|
|
78bbe8855b | ||
|
|
5354104128 | ||
|
|
a5118e4daa | ||
|
|
b5ade5dc12 | ||
|
|
fff8c72f42 | ||
|
|
06ca5f14fc | ||
|
|
8c9d871268 | ||
|
|
5951ead1b6 | ||
|
|
4a0a0e10d2 | ||
|
|
7266eeee7a | ||
|
|
5839ce3e1a | ||
|
|
f85c6bf828 | ||
|
|
9af9e21e05 | ||
|
|
6d16ad2763 | ||
|
|
447fba1594 | ||
|
|
93f845610d | ||
|
|
aa8525385a | ||
|
|
ec783133c1 | ||
|
|
f630bddb19 | ||
|
|
71f0aab2c6 | ||
|
|
ae8de67271 | ||
|
|
6142998b87 | ||
|
|
142376f360 | ||
|
|
ae329e735f | ||
|
|
2dbab70c8c | ||
|
|
47916ecb0f | ||
|
|
635f669a32 | ||
|
|
5f49772ca6 | ||
|
|
6332552346 | ||
|
|
4fb7afcf7b | ||
|
|
ff448950ed | ||
|
|
d6319aeb92 | ||
|
|
8fc6f9c442 | ||
|
|
fdcf317963 | ||
|
|
e9ea79cc8e | ||
|
|
ad32555cc9 | ||
|
|
be8a1acca9 | ||
|
|
477615fc66 | ||
|
|
e0ed8fa0b8 | ||
|
|
dcbf285fae | ||
|
|
ceda459942 | ||
|
|
28fe44aea4 | ||
|
|
71510a8199 | ||
|
|
9cd121c8f6 | ||
|
|
03f14475ff | ||
|
|
2f105ed0a4 | ||
|
|
b4350ec77b | ||
|
|
0fa6eb4e31 | ||
|
|
704d54dd01 | ||
|
|
bee9b61831 | ||
|
|
9c6ce08db0 | ||
|
|
6c0de89de8 | ||
|
|
aa99e202b4 | ||
|
|
04728ea4bc | ||
|
|
6865cdfce3 | ||
|
|
d09b36b1d5 | ||
|
|
a5dd4f5fac | ||
|
|
9936b2967e | ||
|
|
0971613ac0 | ||
|
|
98f9bc3fc1 | ||
|
|
f5f2da11d1 | ||
|
|
f7d855c59e | ||
|
|
b6dba2cbe9 | ||
|
|
2fe743649d | ||
|
|
a015237fd2 | ||
|
|
e1f4deaacc | ||
|
|
1dceee3fa0 | ||
|
|
3a17a7c98f | ||
|
|
3298db8683 | ||
|
|
d4d9403829 | ||
|
|
426e6923ac | ||
|
|
b413f665d8 | ||
|
|
5e1a572bd8 | ||
|
|
3036a149bb | ||
|
|
0dcb4b645c | ||
|
|
d71082b491 | ||
|
|
fc4c8d6054 | ||
|
|
1a27c7eabc | ||
|
|
b2e9602aba | ||
|
|
408b959441 | ||
|
|
35fc98a188 | ||
|
|
92f6a79e1f | ||
|
|
5a9c8e304c | ||
|
|
8f94ed68a3 | ||
|
|
a4cd30451c | ||
|
|
ce1a4b94cb | ||
|
|
92a0deffcf | ||
|
|
97b6f6028b | ||
|
|
99e300a640 | ||
|
|
611e8c39ee | ||
|
|
af55287dee | ||
|
|
01f1c263a6 | ||
|
|
24ea6a9f1d | ||
|
|
46c800b8b7 | ||
|
|
f10be893ce | ||
|
|
c606c1664f | ||
|
|
90a938fe2b | ||
|
|
225932b4c9 | ||
|
|
6b850b9894 | ||
|
|
d8ef796a46 | ||
|
|
943a5d1036 | ||
|
|
e600e9c583 | ||
|
|
b25da8ceaa | ||
|
|
f7388f6492 | ||
|
|
10cdb7387d | ||
|
|
dd436dd1dd | ||
|
|
13b84635b5 | ||
|
|
144d1ea280 | ||
|
|
27534d408f | ||
|
|
b80a2921aa | ||
|
|
0a23ce870a | ||
|
|
ba1e100b42 | ||
|
|
2bc237468d | ||
|
|
c355ef8d53 | ||
|
|
3052d88164 | ||
|
|
1feb23ba99 | ||
|
|
767f650e6f | ||
|
|
59a0fd0799 | ||
|
|
00b5cfcf17 | ||
|
|
f7217944e7 | ||
|
|
9e342f253d | ||
|
|
20c48b6bb2 | ||
|
|
572004d37a | ||
|
|
4f9e7bb572 | ||
|
|
df9b692a32 | ||
|
|
0a322d5a9f | ||
|
|
59eb6d10c1 | ||
|
|
affb219d72 | ||
|
|
a57ce1dd17 | ||
|
|
4e7ace3b48 | ||
|
|
b100b3c36b | ||
|
|
b64b27e5ea | ||
|
|
81c1ba6eef | ||
|
|
46b981bb2f | ||
|
|
93ae4d1ee6 | ||
|
|
9c53d818f4 | ||
|
|
efb2a1d913 | ||
|
|
e5a2c1ab10 | ||
|
|
550c0c7625 | ||
|
|
0abc269a3e | ||
|
|
6b3cbe7882 | ||
|
|
67ed035b36 | ||
|
|
ca25105f13 | ||
|
|
ad30786f4a | ||
|
|
ff0bdcd0c2 | ||
|
|
2e01da5ec1 | ||
|
|
8fb37a0024 | ||
|
|
9412a7424c | ||
|
|
e440eb1733 | ||
|
|
f8cbb4f386 | ||
|
|
86ccaa52a5 | ||
|
|
db14d15953 | ||
|
|
cc3e5d23e4 | ||
|
|
b70d076324 | ||
|
|
cac86d1f77 | ||
|
|
22f7bb822f | ||
|
|
1b53f10091 | ||
|
|
2d697ac8db | ||
|
|
bac268a21c | ||
|
|
321e6e6679 | ||
|
|
e028700175 | ||
|
|
63a673cf1d | ||
|
|
22ef058cb6 | ||
|
|
adcdb19c88 | ||
|
|
d35fa8e8e1 | ||
|
|
9ee6419bc0 | ||
|
|
6af7bfb536 | ||
|
|
3bf0188e7f | ||
|
|
91fc0fd623 | ||
|
|
f936ec0236 | ||
|
|
d2fcf68381 | ||
|
|
a4d0c17efd | ||
|
|
ff1a721d5b | ||
|
|
c870a1bbd5 | ||
|
|
ebf332a8c9 | ||
|
|
b2d335e0da | ||
|
|
85d1fff18f | ||
|
|
2839a95198 | ||
|
|
6bb106c2cb | ||
|
|
e551fd2c1b | ||
|
|
34a11c2338 | ||
|
|
0de3a400eb | ||
|
|
e524ff965d | ||
|
|
7ba689aaeb | ||
|
|
c228e125c3 | ||
|
|
92fde83b3a | ||
|
|
3a268aef50 | ||
|
|
9486dcf6b0 | ||
|
|
f673bd8d7b | ||
|
|
299b680013 | ||
|
|
b1160af896 | ||
|
|
81e8352391 | ||
|
|
1a627d6a87 | ||
|
|
d5f00db9ea | ||
|
|
00a3e562dc | ||
|
|
36aca49fc3 | ||
|
|
0628c9161c | ||
|
|
9b28672e19 | ||
|
|
903ffef42c | ||
|
|
d764058a04 | ||
|
|
0aafe38496 | ||
|
|
d86d565b3f | ||
|
|
e7745db36e | ||
|
|
66d3e1b551 | ||
|
|
474b879b16 | ||
|
|
2f5d6e16a6 | ||
|
|
0a23b57ff8 | ||
|
|
251e1b51c5 | ||
|
|
68150b640e | ||
|
|
217d270457 | ||
|
|
143b6f0df1 | ||
|
|
81684b921e | ||
|
|
2cc6c959a5 | ||
|
|
fb9aa672c9 | ||
|
|
325e65db7f | ||
|
|
603e2b173d | ||
|
|
103b49ec45 | ||
|
|
6c78d7544f | ||
|
|
4d5fbec5a5 | ||
|
|
7cf50a15d0 | ||
|
|
adbc4e9fec | ||
|
|
4815434dd7 | ||
|
|
44f20e7ad6 | ||
|
|
b25e50bdae | ||
|
|
4aab388eff | ||
|
|
604287244f | ||
|
|
47646a4aa0 | ||
|
|
70ca5e2aef | ||
|
|
4a4a721e90 | ||
|
|
52078f7762 | ||
|
|
36377e59cb | ||
|
|
a4062b338e | ||
|
|
ec223ac2ed | ||
|
|
5587b7d469 | ||
|
|
b00577fda4 | ||
|
|
26870d134f | ||
|
|
fb2baad7cc | ||
|
|
0431a2abb1 | ||
|
|
4bae8d4cfb | ||
|
|
c2db2d3cbd | ||
|
|
05d9ec673e | ||
|
|
ae566dca98 | ||
|
|
1732cf9243 | ||
|
|
ab62c19de9 | ||
|
|
96d3a69479 | ||
|
|
2f7bb3499d | ||
|
|
8523bb1ad8 | ||
|
|
e266e1ce40 | ||
|
|
169c3d5a0f | ||
|
|
9cffbe3d49 | ||
|
|
6090439289 | ||
|
|
e6da54d9b8 | ||
|
|
0a843dc086 | ||
|
|
7b3ed2dcbf | ||
|
|
42ed6c3ded | ||
|
|
23ca011ac1 | ||
|
|
d82b3dc429 | ||
|
|
e391793c58 | ||
|
|
236cef4b56 | ||
|
|
45687513bf | ||
|
|
019ffdaf12 | ||
|
|
1a57d4fe11 | ||
|
|
3081f22e70 | ||
|
|
df847431eb | ||
|
|
5b2f1eee65 | ||
|
|
99f488d48f | ||
|
|
05929871c9 | ||
|
|
74b3daa70a | ||
|
|
5e30b0499a | ||
|
|
32bf742709 | ||
|
|
9535f399f2 | ||
|
|
85c7347899 | ||
|
|
4579d26a53 | ||
|
|
128605ab33 | ||
|
|
4dbc908619 | ||
|
|
3a84775912 | ||
|
|
290a82e61c | ||
|
|
adac7d7fb2 | ||
|
|
679fd9d60f | ||
|
|
52320ebb91 | ||
|
|
b4aa17bfbe | ||
|
|
3f8b7ec327 | ||
|
|
eab1f503a5 | ||
|
|
2bcc90a9eb | ||
|
|
a9d0aa136d | ||
|
|
bc7f2677b1 | ||
|
|
691ab3080d | ||
|
|
c5147e0c68 | ||
|
|
e9b0829860 | ||
|
|
95428ab8b0 | ||
|
|
8a595ed77a | ||
|
|
f0ce003765 | ||
|
|
775d56fe52 | ||
|
|
ac2ff29288 | ||
|
|
81cfa5891c | ||
|
|
8e1975efe4 | ||
|
|
39c09733d3 | ||
|
|
da16dfd528 | ||
|
|
e1c397993d | ||
|
|
96cbdd5c37 | ||
|
|
58ca4baf71 | ||
|
|
5245b68689 | ||
|
|
2b6811cb1b | ||
|
|
f0a8aa06bc | ||
|
|
c82496b972 | ||
|
|
c31348ea9a | ||
|
|
75d903b164 | ||
|
|
c885540749 | ||
|
|
7dd40fd2d4 | ||
|
|
bb087caddc | ||
|
|
899b54c082 | ||
|
|
5e3f8b9c2e | ||
|
|
1ccfe928f7 | ||
|
|
3016269268 | ||
|
|
952cfae4e6 | ||
|
|
df7f209ebc | ||
|
|
b09eb63e1e | ||
|
|
d464721397 | ||
|
|
551a85c1e6 | ||
|
|
f3f4bd33e5 | ||
|
|
2686761608 | ||
|
|
02a2c3224f | ||
|
|
8ec1dda9ba | ||
|
|
0308532523 | ||
|
|
10b3af2947 | ||
|
|
1f34569ddc | ||
|
|
158bfe4816 | ||
|
|
fb0941bbe9 | ||
|
|
16eefe333f | ||
|
|
65e585e122 | ||
|
|
2ba36ee04c | ||
|
|
fc05529574 | ||
|
|
010770904f | ||
|
|
07d24f487a | ||
|
|
0960e4caa4 | ||
|
|
811acdb7f5 | ||
|
|
2fce5c4d5d | ||
|
|
a7266364d1 | ||
|
|
e83b41dc01 | ||
|
|
76665dd56e | ||
|
|
2d42b478ba | ||
|
|
5797e8aeec | ||
|
|
885fa6beae | ||
|
|
65cdd5fcbe | ||
|
|
73da4844ee | ||
|
|
4302e19aba | ||
|
|
0c6f05f34a | ||
|
|
385123fd40 | ||
|
|
8040c285cd | ||
|
|
bf1ee61bf0 | ||
|
|
ada454f56f | ||
|
|
57d2ef8740 | ||
|
|
a97e0982e3 | ||
|
|
eaa2060d84 | ||
|
|
3e02c574e7 | ||
|
|
e873d55cd3 | ||
|
|
c7230ccbb0 | ||
|
|
fc71ced660 | ||
|
|
6041a9d094 | ||
|
|
599cd766e1 | ||
|
|
84e02099a2 | ||
|
|
e64c8007c0 | ||
|
|
9339823e84 | ||
|
|
1ab52cfce3 | ||
|
|
e6d4620af1 | ||
|
|
656e6db846 | ||
|
|
9ed16478f4 | ||
|
|
30474e3a2b | ||
|
|
460bd98f1b | ||
|
|
a553eba574 | ||
|
|
61f515670c | ||
|
|
789af0f8a6 | ||
|
|
86fae58c96 | ||
|
|
03ae741505 | ||
|
|
c54d3abe47 | ||
|
|
906cd975d1 | ||
|
|
6fe511eb50 | ||
|
|
839f34ec4e | ||
|
|
17d18b22c7 | ||
|
|
66a04ed730 | ||
|
|
7e14a0bc30 | ||
|
|
4f2e06407b | ||
|
|
77de0f86dc | ||
|
|
f79c998f95 | ||
|
|
3b4bc9163a | ||
|
|
e146135bd1 | ||
|
|
e9e18afb4a | ||
|
|
62c31eb202 | ||
|
|
8016e84bc7 | ||
|
|
1eacee85ae | ||
|
|
5986145282 | ||
|
|
5756be7d36 | ||
|
|
b134a69a28 | ||
|
|
83f9eacac4 | ||
|
|
baab6b951b | ||
|
|
b041fbe3ec | ||
|
|
1b7b6d4b7e | ||
|
|
903a1bec91 | ||
|
|
15c7d9b0f1 | ||
|
|
ebc3a251b7 | ||
|
|
a567f4a6de | ||
|
|
4dc49604b6 | ||
|
|
c660daf4c2 | ||
|
|
fadcf62166 | ||
|
|
c02b255766 | ||
|
|
89788fa665 | ||
|
|
a052e2ee8f | ||
|
|
158e5004b7 | ||
|
|
6f9ff3be37 | ||
|
|
f766c57743 | ||
|
|
8f53152c3e | ||
|
|
7bbc88d716 | ||
|
|
68256d2343 | ||
|
|
1a0c70acc2 | ||
|
|
f88c440c48 | ||
|
|
cfa56ba6d4 | ||
|
|
37e6297fb2 | ||
|
|
2c6b646d87 | ||
|
|
e7572094b5 | ||
|
|
ddd5e0e889 | ||
|
|
5e34823a49 | ||
|
|
fdef21a871 | ||
|
|
d40cff8a99 | ||
|
|
8927e45ded | ||
|
|
a602f73ed0 | ||
|
|
1a93df92d4 | ||
|
|
12fe28d8ab | ||
|
|
06754d6158 | ||
|
|
8f9ec07ac3 | ||
|
|
1d5087374e | ||
|
|
8356264fe0 | ||
|
|
18ecd748dd | ||
|
|
6061d0603a | ||
|
|
e324f27655 | ||
|
|
afd645fb11 | ||
|
|
e48d37ccab | ||
|
|
5b42593fbb | ||
|
|
25f3c6a548 | ||
|
|
5c04f2634a | ||
|
|
ad01610d1e | ||
|
|
697c380cd1 | ||
|
|
ce89bf3c77 | ||
|
|
81e8143a43 | ||
|
|
39c4117409 | ||
|
|
8409986ef5 | ||
|
|
2b50367d7f | ||
|
|
cd4b85b0b5 | ||
|
|
dcec02412d | ||
|
|
1dcc491fec | ||
|
|
c6419a9c61 | ||
|
|
d715f86713 | ||
|
|
5221828705 | ||
|
|
6aa4acd3db | ||
|
|
15936c29c1 | ||
|
|
41689a2d82 | ||
|
|
1f71d19004 | ||
|
|
8b70c69a0d | ||
|
|
dfe80a30dc | ||
|
|
82a7f2dc2d | ||
|
|
ce026e7ad0 | ||
|
|
58e3122dab | ||
|
|
3b55b2d1b2 | ||
|
|
e3a7164fe1 | ||
|
|
2326e61de5 | ||
|
|
32b18c9509 | ||
|
|
acf52ad8a3 | ||
|
|
08dd493f98 | ||
|
|
4188cc2949 | ||
|
|
07bbe7dfb2 | ||
|
|
0aa1b80e3e | ||
|
|
5ac390281e | ||
|
|
ac465c5a18 | ||
|
|
745cd9f501 | ||
|
|
1ef3546822 | ||
|
|
b9df028bfb | ||
|
|
e74ad2b555 | ||
|
|
71c0056c66 | ||
|
|
56b27ea785 | ||
|
|
7e8974683c | ||
|
|
2d75f59d33 | ||
|
|
a709a3bcc0 | ||
|
|
34bf5112e0 | ||
|
|
bfe18d1d28 | ||
|
|
6a76afc20d | ||
|
|
9c469c2f96 | ||
|
|
e68a1dee33 | ||
|
|
2ab42f3dd6 | ||
|
|
af34b43a8d | ||
|
|
0f71cc7864 | ||
|
|
115ca7b789 | ||
|
|
df90de3a5f | ||
|
|
42ea7a9814 | ||
|
|
809750b995 | ||
|
|
c683cbdb2d | ||
|
|
d243b73678 | ||
|
|
b9abd2f9a5 | ||
|
|
dc28d063aa | ||
|
|
34cb661c35 | ||
|
|
bb6045c1d0 | ||
|
|
f1a74b5939 | ||
|
|
6fb9038af1 | ||
|
|
27f721a1f5 | ||
|
|
5717dc294e | ||
|
|
ae0f8df11b | ||
|
|
20bbdf22c7 | ||
|
|
77460ba502 | ||
|
|
72c6a4289e | ||
|
|
f8235da4d8 | ||
|
|
29973d7a72 | ||
|
|
8d3316ccd6 | ||
|
|
2c29f831e8 | ||
|
|
9457325119 | ||
|
|
189f8afcc9 | ||
|
|
f3a34990ab | ||
|
|
9699b67510 | ||
|
|
d60633a46c | ||
|
|
ae2df33ce6 | ||
|
|
0fcf28e7e7 | ||
|
|
5fad8f74b1 | ||
|
|
e35e34d2e0 | ||
|
|
8943144b2b | ||
|
|
aa6acc6673 | ||
|
|
31a215d4d6 | ||
|
|
9be6af8481 | ||
|
|
30948de13d | ||
|
|
b97158bf7b | ||
|
|
b14a8ff2fd | ||
|
|
6646be8d94 | ||
|
|
647a2aea64 | ||
|
|
035693aa30 | ||
|
|
58e58ce51c | ||
|
|
7aff72fc7c | ||
|
|
4b7e48d3ec | ||
|
|
91086d004c | ||
|
|
0e074d3a5a | ||
|
|
5d86b8893c | ||
|
|
ea00224e7f | ||
|
|
38293efe75 | ||
|
|
3286c5e174 | ||
|
|
e0f8a28f38 | ||
|
|
bf1b00b163 | ||
|
|
2678b9003a | ||
|
|
4fa3a136ad | ||
|
|
178a6bd66e | ||
|
|
e7d3ee3bc8 | ||
|
|
57e1339230 | ||
|
|
97c9a9b0b0 | ||
|
|
4144423227 | ||
|
|
4d03514142 | ||
|
|
0bc5566976 | ||
|
|
99550b79ab | ||
|
|
925567add5 | ||
|
|
ad97731d46 | ||
|
|
40684a93a2 | ||
|
|
f3b644ceb8 | ||
|
|
901ba6e87f | ||
|
|
6e9b70a8d6 | ||
|
|
76389bd584 | ||
|
|
7bf8650d59 | ||
|
|
7cb24dd96d | ||
|
|
dee040318a | ||
|
|
b93c5a9daa | ||
|
|
baf563e46d | ||
|
|
42910ebe14 | ||
|
|
e10246f10b | ||
|
|
f524219d68 | ||
|
|
a9dfd88671 | ||
|
|
61b338f464 | ||
|
|
beac73b6c8 | ||
|
|
f9f93c77e2 | ||
|
|
6fc1b4c6c0 | ||
|
|
639898ec07 | ||
|
|
3d3790fdbc | ||
|
|
69c8968cb0 | ||
|
|
229caea5fd | ||
|
|
aa25fc7901 | ||
|
|
4aba493ee2 | ||
|
|
dd7a080e2d | ||
|
|
b9cfac5934 | ||
|
|
f8e97fcc32 | ||
|
|
d2a26d6d48 | ||
|
|
7f8f2641f6 | ||
|
|
022dbb606f | ||
|
|
fea72b190d | ||
|
|
bce4351b4f | ||
|
|
eea073f882 | ||
|
|
0352d413e3 | ||
|
|
56bf98d68a | ||
|
|
fc1d88f5bb | ||
|
|
2b109db1b1 | ||
|
|
acbe410e0b | ||
|
|
89bafea61f | ||
|
|
07b7e05caa | ||
|
|
ec072fd639 | ||
|
|
b874c1a8a8 | ||
|
|
67b03076d7 | ||
|
|
33a0c4a9ae | ||
|
|
e25914c3d3 | ||
|
|
4cc5999f05 | ||
|
|
403aa5fd3e | ||
|
|
08f203c0c2 | ||
|
|
0fbf31ec98 | ||
|
|
db9b7ca447 | ||
|
|
fc9fa2614d | ||
|
|
eecc71c77f | ||
|
|
d5f69aec10 | ||
|
|
5f898a9071 | ||
|
|
a08f21336a | ||
|
|
215125de26 | ||
|
|
dfa94eac41 | ||
|
|
5de72a74f5 | ||
|
|
a44f0a719e | ||
|
|
247d869a5c | ||
|
|
b9b6e1818f | ||
|
|
a7968ccc3c | ||
|
|
b7e0e5a356 | ||
|
|
e3aecb2aa9 | ||
|
|
2b879ab471 | ||
|
|
116ab83b95 | ||
|
|
f24ae0fc2c | ||
|
|
c5d0d4acd0 | ||
|
|
062bf737c2 | ||
|
|
06190286ec | ||
|
|
3bca856e87 | ||
|
|
b4437d9cfd | ||
|
|
b3a778b89a | ||
|
|
dcb11f7606 | ||
|
|
933ce42d5a | ||
|
|
6c1ba957bd | ||
|
|
e021286eee | ||
|
|
0ee7a66033 | ||
|
|
42c797ee97 | ||
|
|
c03699fc5b | ||
|
|
b585c6676d | ||
|
|
f5ddb0f1f8 | ||
|
|
ef97f9e738 | ||
|
|
26a03b55de | ||
|
|
e9b0100b06 | ||
|
|
b93a16abae | ||
|
|
ff2783d434 | ||
|
|
664df55525 | ||
|
|
66f93148a7 | ||
|
|
25a5a8db68 | ||
|
|
a68d91b54c | ||
|
|
e48afc9fdf | ||
|
|
acdefb394c | ||
|
|
88ec3a5751 | ||
|
|
e3af0a13da | ||
|
|
734dc2e37a | ||
|
|
06b97b91e0 | ||
|
|
6aecd8d44a | ||
|
|
8a4ac3ea10 | ||
|
|
0ca123f4bc | ||
|
|
bbf5e1fa78 | ||
|
|
7454e55693 | ||
|
|
c745fe7778 | ||
|
|
6adcebb247 | ||
|
|
4c1844e46a | ||
|
|
38f9b8f3dd | ||
|
|
c2e72c7641 | ||
|
|
7faf143a97 | ||
|
|
e53a7f65b8 | ||
|
|
21eb9df85f | ||
|
|
17cfd4924c | ||
|
|
a0bebca1e6 | ||
|
|
75cbfa2898 | ||
|
|
58a8ed1588 | ||
|
|
e032f8df59 | ||
|
|
b16e37d80a | ||
|
|
c17cc07b73 | ||
|
|
6f767a72a7 | ||
|
|
11196436e9 | ||
|
|
9afc433db4 | ||
|
|
f701e3d834 | ||
|
|
4c623ca3c5 | ||
|
|
0671f05c05 | ||
|
|
0713da7393 | ||
|
|
d980b8cfdc | ||
|
|
05955d0483 | ||
|
|
28c765bd9a | ||
|
|
8287317be7 | ||
|
|
08cc67d7c5 | ||
|
|
ec858b2d4c | ||
|
|
47ece983d2 | ||
|
|
52310b5dd9 | ||
|
|
c2a4a2778e | ||
|
|
1db5977e80 | ||
|
|
1b5dc0e434 | ||
|
|
251364d8be | ||
|
|
f07f02d866 | ||
|
|
1388103919 | ||
|
|
fe1054d58a | ||
|
|
ba6ac778fc | ||
|
|
228ffcbfce | ||
|
|
f18ab9e5cc | ||
|
|
06c82ee87d | ||
|
|
9ba5ee8043 | ||
|
|
eede4e50ca | ||
|
|
1e7b6f78ca | ||
|
|
aa10f63d9f | ||
|
|
5b984d924f | ||
|
|
a25af36e32 | ||
|
|
eb8b5e5c01 | ||
|
|
817f057927 | ||
|
|
a13c44d81a | ||
|
|
45ad8f8ffb | ||
|
|
7da9e88c0b | ||
|
|
674e63cd3e | ||
|
|
1c73c91133 | ||
|
|
b1d11d4f69 | ||
|
|
001a9310c3 | ||
|
|
4cea9023f2 | ||
|
|
8ffadfa1f1 | ||
|
|
39e0b8e40e | ||
|
|
50d7929e76 | ||
|
|
10840b22c5 | ||
|
|
acfbab5915 | ||
|
|
9b00f65798 | ||
|
|
48c324fe86 | ||
|
|
0c495e7e72 | ||
|
|
50ccfee201 | ||
|
|
f39a5f6e68 | ||
|
|
f1f2efc4f8 | ||
|
|
fa739c9594 | ||
|
|
95f0ce1816 | ||
|
|
3432529f9c | ||
|
|
a32c8fabed | ||
|
|
6a11501184 | ||
|
|
7ca228d466 | ||
|
|
09a00f7d42 | ||
|
|
eac4cd15e3 | ||
|
|
c03fd4645d | ||
|
|
911ddbe1c8 | ||
|
|
b76c7a4824 | ||
|
|
1408ac77f9 | ||
|
|
c641abc7cd | ||
|
|
7e97d10ae1 | ||
|
|
56b134facd | ||
|
|
41286650cc | ||
|
|
2aca007a59 | ||
|
|
39f5c00f7e | ||
|
|
7eab431e5d | ||
|
|
678b15e759 | ||
|
|
3c8e7c6c10 | ||
|
|
bb7433ab40 | ||
|
|
4f64513c83 | ||
|
|
4ee47b6b1b | ||
|
|
350f5ccb3c | ||
|
|
0d3f94860b | ||
|
|
3d7489563d | ||
|
|
ac1153c7cf | ||
|
|
d4c4220299 | ||
|
|
3b1672a4a7 | ||
|
|
009f81a9a6 | ||
|
|
69285f28ad | ||
|
|
3593df0e73 | ||
|
|
8b10b1dc62 | ||
|
|
0db2a81e4e | ||
|
|
9fe64008c2 | ||
|
|
077c259d5b | ||
|
|
879bd62468 | ||
|
|
e5746c19cf | ||
|
|
29814d7458 | ||
|
|
e399f9e851 | ||
|
|
e4e20c2d25 | ||
|
|
08a70664f4 | ||
|
|
9d77f8dcd2 | ||
|
|
1d76c644cb | ||
|
|
75fc35ee4b | ||
|
|
ba3102d667 | ||
|
|
a94fc22659 | ||
|
|
8a9fed64f2 | ||
|
|
4468b5a2e4 | ||
|
|
71c7e30548 | ||
|
|
940bd55079 | ||
|
|
f2aa40c772 | ||
|
|
886db1a2c3 | ||
|
|
c8979940a8 | ||
|
|
b4c06db031 | ||
|
|
82486a873a | ||
|
|
99760ba6a0 | ||
|
|
2b987e6e93 | ||
|
|
523134f24b | ||
|
|
99c228dd6d | ||
|
|
10f80f9a4f | ||
|
|
44d38a00d4 | ||
|
|
c623f70caa | ||
|
|
62a10047ca | ||
|
|
f16b783378 | ||
|
|
95c55a8ab3 | ||
|
|
a8c932ffe4 | ||
|
|
be4b75932b | ||
|
|
04d7f3a5dc | ||
|
|
eddfacd0f4 | ||
|
|
cba3c20d5c | ||
|
|
507783ed8d | ||
|
|
06c98ed229 | ||
|
|
69742839c0 | ||
|
|
bb52049bf4 | ||
|
|
20b5f0e681 | ||
|
|
3803b8f284 | ||
|
|
e3daf743f2 | ||
|
|
ae5da74bb1 | ||
|
|
05f37ec2bc | ||
|
|
cf78047830 | ||
|
|
284428a45a | ||
|
|
79f2efdfd9 | ||
|
|
07822b371f | ||
|
|
7a3a385569 | ||
|
|
0f8cb7ea6d | ||
|
|
fa7ae376e0 | ||
|
|
8c223056fe | ||
|
|
e57f78cf90 | ||
|
|
ebd79d388b | ||
|
|
10724fee04 | ||
|
|
0d46f85ead | ||
|
|
4727ba3b51 | ||
|
|
eb15a3d849 | ||
|
|
4d09bae09b | ||
|
|
56ad177d4a | ||
|
|
11902dec3c | ||
|
|
4fdbe9b9ff | ||
|
|
18037bb484 | ||
|
|
a6e7e30177 | ||
|
|
288285f22b | ||
|
|
5b69ff7e94 | ||
|
|
fa2d838e60 | ||
|
|
bc0c6be4c5 | ||
|
|
ef767728ac | ||
|
|
f56d219882 | ||
|
|
32afccb16d | ||
|
|
b4f528039f | ||
|
|
d6b470ffbe | ||
|
|
3c6b418ca8 | ||
|
|
105a38a7db | ||
|
|
e6f25b9c5e | ||
|
|
6e0b956e61 | ||
|
|
86af14ad71 | ||
|
|
a029768d24 | ||
|
|
e0b85131bd | ||
|
|
4d9c9206cf | ||
|
|
92ca8862e1 | ||
|
|
b77bc28a79 | ||
|
|
b5eca401c6 | ||
|
|
a88e6ec534 | ||
|
|
35116f9229 | ||
|
|
fe66a59618 | ||
|
|
a1f90cd39b | ||
|
|
743975db52 | ||
|
|
45dc7459b8 | ||
|
|
ff3056332e | ||
|
|
6877b663f1 | ||
|
|
3c69f81a10 | ||
|
|
d316d57e5d | ||
|
|
92eddf8eb6 | ||
|
|
109a5b4748 | ||
|
|
0c81556b90 | ||
|
|
7e4b572699 | ||
|
|
d72828b3f4 | ||
|
|
9220f4d829 | ||
|
|
66917cd2c0 | ||
|
|
d0d375aeb7 | ||
|
|
b41dde777e | ||
|
|
d3dcd39f61 | ||
|
|
4121cae1d6 | ||
|
|
341138b731 | ||
|
|
527c3996ae | ||
|
|
305b4148bd | ||
|
|
07c22ed5bc | ||
|
|
c2fba6b1cf | ||
|
|
da87059041 | ||
|
|
0e300df68c | ||
|
|
6a9c4cf8cc | ||
|
|
1bebd5488a | ||
|
|
bb354e4941 | ||
|
|
0b5053d49a | ||
|
|
b3a1e6d0a5 | ||
|
|
457459671c | ||
|
|
20f09e6c6e | ||
|
|
944e1d9698 | ||
|
|
e8f795763b | ||
|
|
1d11683ce8 | ||
|
|
cca4258887 | ||
|
|
9999321400 | ||
|
|
3de3fc00ce | ||
|
|
2debb32098 | ||
|
|
e49d7b4ec2 | ||
|
|
c009a56825 | ||
|
|
c75dada340 | ||
|
|
890b0ac301 | ||
|
|
77142eb2df | ||
|
|
1185fad75c | ||
|
|
7463652345 | ||
|
|
d63309ae51 | ||
|
|
40bac000ab | ||
|
|
dac9cee7ca | ||
|
|
ef4c0c529a | ||
|
|
9a4986f189 | ||
|
|
afa674e2ea | ||
|
|
8277c74c5b | ||
|
|
b3c615576e | ||
|
|
6610a29422 | ||
|
|
67b0f14be6 | ||
|
|
15bc5b5b5d | ||
|
|
57d594acb0 | ||
|
|
e4aa761098 | ||
|
|
40aa685aba | ||
|
|
ed766484d2 | ||
|
|
a5f844bc7d | ||
|
|
97bd9b6381 | ||
|
|
ca8f2f5734 | ||
|
|
bcaaf2bb13 | ||
|
|
23128b2e53 | ||
|
|
5fc0e4a071 | ||
|
|
d7d143b97f | ||
|
|
4fe25da30b | ||
|
|
74c8a199f7 | ||
|
|
5f4a2ec4e7 | ||
|
|
56f451b30f | ||
|
|
1683c8e963 | ||
|
|
3091a93a52 | ||
|
|
f18d310348 | ||
|
|
228bdf74a4 | ||
|
|
51a1977243 | ||
|
|
e201344ccd | ||
|
|
f630ccb134 | ||
|
|
3a100702e8 | ||
|
|
fa0745e226 | ||
|
|
e5a89946f6 | ||
|
|
276ba2cd8e | ||
|
|
05a55f4a43 | ||
|
|
37a4e8a4aa | ||
|
|
3776292278 | ||
|
|
a9bba9be2b | ||
|
|
92ee0a5227 | ||
|
|
052fd35c72 | ||
|
|
5e3357d062 | ||
|
|
9e2a55edc2 | ||
|
|
29d8efd26e | ||
|
|
4b8608906a | ||
|
|
dbfe4fd5ac | ||
|
|
266f1c3a49 | ||
|
|
6ce686ab9c | ||
|
|
ea38645493 | ||
|
|
a929aaca04 | ||
|
|
5090c07846 | ||
|
|
8eb6fc8343 | ||
|
|
7da7bec241 | ||
|
|
a69789d572 | ||
|
|
172bc81dd2 | ||
|
|
65234a5a9a | ||
|
|
495481725a | ||
|
|
88353e8748 | ||
|
|
939c46fafd | ||
|
|
3145be12c0 | ||
|
|
4cfb599165 | ||
|
|
30e834744d | ||
|
|
c95a0c86b3 | ||
|
|
2daabd000f | ||
|
|
b97fd17146 | ||
|
|
5987330e59 | ||
|
|
5903475f4a | ||
|
|
0c3dc3dea2 | ||
|
|
c2f2146872 | ||
|
|
05087a833c | ||
|
|
c6eb306691 | ||
|
|
a3545ce551 | ||
|
|
5a6aef7a1d | ||
|
|
585bbf3987 | ||
|
|
df9bd21f55 | ||
|
|
feb7cd7bbf | ||
|
|
79c05c37dd | ||
|
|
1dd3766c5f | ||
|
|
54a41b4f0a | ||
|
|
dc691daf54 | ||
|
|
768b52e517 | ||
|
|
bb0d26e116 | ||
|
|
deef167cb2 | ||
|
|
c5767a280e | ||
|
|
ce5f73a5a6 | ||
|
|
fedfc66403 | ||
|
|
27042dae4d | ||
|
|
0877c4cb29 | ||
|
|
098ea0f405 | ||
|
|
de86376724 | ||
|
|
7fc4d8a172 | ||
|
|
ab276a6a61 | ||
|
|
7e026a7072 | ||
|
|
8513b6fbd5 | ||
|
|
ee6785eff9 | ||
|
|
a341a20e2c | ||
|
|
fefadaebfa | ||
|
|
777d77db53 | ||
|
|
8d72515a30 | ||
|
|
7a262eac12 | ||
|
|
1cd0abf415 | ||
|
|
10575d80ad | ||
|
|
15cf010e44 | ||
|
|
e26e383bd7 | ||
|
|
6652f96349 | ||
|
|
3b2eacfc8e | ||
|
|
3a4cdfd7ca | ||
|
|
2a665f9d92 | ||
|
|
cdea1d5545 | ||
|
|
a577c2d859 | ||
|
|
8f74e83d83 | ||
|
|
4e9bcd0d1f | ||
|
|
bc29495dd0 | ||
|
|
5df598fd56 | ||
|
|
8ddc688c13 | ||
|
|
49dad3099a | ||
|
|
90ecc5c13b | ||
|
|
9b31e4f385 | ||
|
|
9923a07c25 | ||
|
|
35d6bfb3a8 | ||
|
|
aa4a567160 | ||
|
|
dc3ca6db4f | ||
|
|
86389a5fb3 | ||
|
|
d7140eac35 | ||
|
|
d2b81cd359 | ||
|
|
18bab4aa7d | ||
|
|
c0bbebd532 | ||
|
|
acbc2fd490 | ||
|
|
d765d11c3e | ||
|
|
79ab85c632 | ||
|
|
dea68f3cf5 | ||
|
|
50ea267664 | ||
|
|
789f11a5c4 | ||
|
|
322548f078 | ||
|
|
8ea805e4e3 | ||
|
|
3d7e4766f7 | ||
|
|
5c2166a019 | ||
|
|
f5aec1c894 | ||
|
|
b8fb8a52f1 | ||
|
|
9d32300612 | ||
|
|
51c3257df9 | ||
|
|
35180b41bc | ||
|
|
3200ba0ed0 | ||
|
|
8e742ceb91 | ||
|
|
02deea85e6 | ||
|
|
13ea678e5e | ||
|
|
2e98c16f05 | ||
|
|
ca6aa5213c | ||
|
|
54f25358eb | ||
|
|
6fce69bbac | ||
|
|
5ceb18414a | ||
|
|
716150cfd2 | ||
|
|
e10baa915d | ||
|
|
ef6ff68b0b | ||
|
|
b87a6a9fec | ||
|
|
2efe8ae0cf | ||
|
|
8b2f46f0ba | ||
|
|
3c41d4b3a4 | ||
|
|
189d95f4fa | ||
|
|
28939e7405 | ||
|
|
84be8cc045 | ||
|
|
2a7e2be675 | ||
|
|
a1057ef764 | ||
|
|
f79a0a8603 | ||
|
|
6b84f54611 | ||
|
|
1bd66297e2 | ||
|
|
ea08f39f6e | ||
|
|
818c5a9cf5 | ||
|
|
074fd14849 | ||
|
|
a783859ab2 | ||
|
|
4b84a5ec15 | ||
|
|
905db1e8ff | ||
|
|
934d7e0f02 | ||
|
|
3b9a76c1f2 | ||
|
|
dabd294eaf | ||
|
|
3d2f8a7ddb | ||
|
|
507d457900 | ||
|
|
875be1f028 | ||
|
|
71267ec333 | ||
|
|
3aed470a87 | ||
|
|
356b0ae659 | ||
|
|
d8e142d454 | ||
|
|
dd6c5292fd | ||
|
|
571c7a8069 | ||
|
|
5dbde869df | ||
|
|
432943d6ee | ||
|
|
014a821d05 | ||
|
|
d8d98e289a | ||
|
|
53a65ea810 | ||
|
|
fc0ac45f21 | ||
|
|
c2d8c9a662 | ||
|
|
2ad2a95cc6 | ||
|
|
47a8329cd0 | ||
|
|
0087f328d6 | ||
|
|
9250d90e57 | ||
|
|
2dfe9eea94 | ||
|
|
5b28594189 | ||
|
|
33c88ec9e4 | ||
|
|
6d4bb5dcbc | ||
|
|
f2d0f1e51e | ||
|
|
b6d3e76568 | ||
|
|
1515793109 | ||
|
|
26bd15ec28 | ||
|
|
b78dd69fd6 | ||
|
|
7277e30443 | ||
|
|
683c37aca1 | ||
|
|
856e5eca4c | ||
|
|
3c9963065d | ||
|
|
ad9886284b | ||
|
|
7b18ce41a1 | ||
|
|
0fb46ed60b | ||
|
|
a9875dff13 | ||
|
|
9e6427d406 | ||
|
|
92322288ca | ||
|
|
19a4c7253a | ||
|
|
8eed2329bc | ||
|
|
e07c521288 | ||
|
|
917eaa50fb | ||
|
|
9b274cb243 | ||
|
|
93cbdadff3 | ||
|
|
9f5e213402 | ||
|
|
7b60ae26fc | ||
|
|
26b552a12e | ||
|
|
4e8ca603fe | ||
|
|
8cbeecd347 | ||
|
|
ef25503d58 | ||
|
|
5c4c00bd88 | ||
|
|
52d1a103aa | ||
|
|
48888be408 | ||
|
|
8b5106adc7 | ||
|
|
6a80ce878f | ||
|
|
13a75adba9 | ||
|
|
804d4320d7 | ||
|
|
76bf89dda3 | ||
|
|
950bc05d62 | ||
|
|
f9acd6a66b | ||
|
|
d3023a0068 | ||
|
|
d7df99e960 | ||
|
|
6b8478dbe9 | ||
|
|
a297d03db5 | ||
|
|
9d3d9d1390 | ||
|
|
761ac95085 | ||
|
|
d95ca5f9e4 | ||
|
|
c410348278 | ||
|
|
d8a758211f | ||
|
|
33e60f2527 | ||
|
|
fb705eee23 | ||
|
|
635e16e934 | ||
|
|
1deb3ae67f | ||
|
|
16ff40f420 | ||
|
|
a8b5cb23fe | ||
|
|
82f88d04ad | ||
|
|
0e1091e0ea | ||
|
|
7b48f10cc9 | ||
|
|
0be34b1135 | ||
|
|
fb5e0242d0 | ||
|
|
d376035557 | ||
|
|
747b2dc7c5 | ||
|
|
fac2f1bee3 | ||
|
|
a211f6aed9 | ||
|
|
0bc494245d | ||
|
|
b31a88043e | ||
|
|
85509c6d8b | ||
|
|
2dd131cf79 | ||
|
|
51990d0b33 | ||
|
|
00a49afc30 | ||
|
|
faa0630851 | ||
|
|
aac3fc68fc | ||
|
|
9c08b96b50 | ||
|
|
15ddde1df4 | ||
|
|
f2a9de3ba8 | ||
|
|
fd725206e2 | ||
|
|
6368b9383a | ||
|
|
2b8a11b001 | ||
|
|
c9e0339a30 | ||
|
|
8d11595290 | ||
|
|
2fe9f3effa | ||
|
|
ae122ff8a2 | ||
|
|
8b941ddd33 | ||
|
|
2902ea6689 | ||
|
|
5ccbf355bd | ||
|
|
62d8f635b0 | ||
|
|
4c3aae63d3 | ||
|
|
8f94aa0c0d | ||
|
|
0370306bb6 | ||
|
|
c9176efe6f | ||
|
|
a3fd08b7ef | ||
|
|
83a9e36ef1 | ||
|
|
9668decc84 | ||
|
|
328bb47d44 | ||
|
|
c74e0b9eab | ||
|
|
20dc32413f | ||
|
|
d4e618893c | ||
|
|
8d0d934249 | ||
|
|
ef2441ad82 | ||
|
|
bb7859138c | ||
|
|
ebc4570941 | ||
|
|
d04baed38b | ||
|
|
001c81f797 | ||
|
|
3327bf4788 | ||
|
|
e0b480e232 | ||
|
|
b328d85230 | ||
|
|
3afaa5c1e6 | ||
|
|
f2c8699823 | ||
|
|
4c11315a3c | ||
|
|
0e3a347d6b | ||
|
|
dc723fadaa | ||
|
|
6396958a31 | ||
|
|
1fe57e4841 | ||
|
|
3885ae6337 | ||
|
|
39e3366b3b | ||
|
|
a5ffd47935 | ||
|
|
18a96a445b | ||
|
|
de366b976e | ||
|
|
8f6aff3a7e | ||
|
|
fb411b20cc | ||
|
|
52ce7d6935 | ||
|
|
75ee398633 | ||
|
|
53bdd946d6 | ||
|
|
79f36664ef | ||
|
|
83078a48ab | ||
|
|
6f67a812dc | ||
|
|
6ad705b40e | ||
|
|
4cb43415a1 | ||
|
|
bbb09b558c | ||
|
|
6363be81e0 | ||
|
|
c6810d7460 | ||
|
|
4c1e7e7c2f | ||
|
|
931081752a | ||
|
|
424e98e67e | ||
|
|
7cfa93f5f8 | ||
|
|
fd8e8d1475 | ||
|
|
7ed5eb22ec | ||
|
|
fa1c275904 | ||
|
|
558c72bbb7 | ||
|
|
37976455bc | ||
|
|
db6ee8f687 | ||
|
|
53e7ffa311 | ||
|
|
e0f7ff325a | ||
|
|
1fcd1e33c5 | ||
|
|
843b16c1f0 | ||
|
|
a58f3f0fe3 | ||
|
|
e69e395b25 | ||
|
|
456164fc24 | ||
|
|
9b7f61a09d | ||
|
|
2de9adb7ae | ||
|
|
c7e0cc1158 | ||
|
|
e79861c30a | ||
|
|
407f596b61 | ||
|
|
41d30fc8dc | ||
|
|
2e429c5b35 | ||
|
|
28afe3470b | ||
|
|
f623b24196 |
1396
.editorconfig
Normal file
1396
.editorconfig
Normal file
File diff suppressed because it is too large
Load Diff
4
.github/FUNDING.yml
vendored
Normal file
4
.github/FUNDING.yml
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
# Copyright 2021 Signal Messenger, LLC
|
||||||
|
# SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
|
||||||
|
custom: https://signal.org/donate/
|
||||||
0
.github/stale.yml
vendored
Normal file
0
.github/stale.yml
vendored
Normal file
18
.github/workflows/test.yml
vendored
Normal file
18
.github/workflows/test.yml
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
name: Service CI
|
||||||
|
|
||||||
|
on: [push]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: Set up JDK 17
|
||||||
|
uses: actions/setup-java@3bc31aaf88e8fc94dc1e632d48af61be5ca8721c
|
||||||
|
with:
|
||||||
|
distribution: 'temurin'
|
||||||
|
java-version: 17
|
||||||
|
cache: 'maven'
|
||||||
|
- name: Build with Maven
|
||||||
|
run: mvn -e -B verify
|
||||||
17
.gitignore
vendored
17
.gitignore
vendored
@@ -8,4 +8,21 @@ local.yml
|
|||||||
config/production.yml
|
config/production.yml
|
||||||
config/federated.yml
|
config/federated.yml
|
||||||
config/staging.yml
|
config/staging.yml
|
||||||
|
config/testing.yml
|
||||||
|
config/deploy.properties
|
||||||
|
/service/config/production.yml
|
||||||
|
/service/config/federated.yml
|
||||||
|
/service/config/staging.yml
|
||||||
|
/service/config/testing.yml
|
||||||
|
/service/config/deploy.properties
|
||||||
|
/service/dependency-reduced-pom.xml
|
||||||
.opsmanage
|
.opsmanage
|
||||||
|
put.sh
|
||||||
|
deployer-staging.properties
|
||||||
|
deployer-production.properties
|
||||||
|
deployer.log
|
||||||
|
/service/src/main/resources/org/signal/badges/Badges_*.properties
|
||||||
|
!/service/src/main/resources/org/signal/badges/Badges_en.properties
|
||||||
|
/service/src/main/resources/org/signal/subscriptions/Subscriptions_*.properties
|
||||||
|
!/service/src/main/resources/org/signal/subscriptions/Subscriptions_en.properties
|
||||||
|
/.tx/config
|
||||||
|
|||||||
11
.gitmodules
vendored
Normal file
11
.gitmodules
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
# Note that the implementation of the abusive message filter is private; internal
|
||||||
|
# developers will need to override this URL with:
|
||||||
|
#
|
||||||
|
# ```
|
||||||
|
# git config submodule.abusive-message-filter.url PRIVATE_URL
|
||||||
|
# ```
|
||||||
|
#
|
||||||
|
# External developers may safely ignore this submodule.
|
||||||
|
[submodule "abusive-message-filter"]
|
||||||
|
path = abusive-message-filter
|
||||||
|
url = REDACTED
|
||||||
Binary file not shown.
Binary file not shown.
9
.mvn/extensions.xml
Normal file
9
.mvn/extensions.xml
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
<extensions xmlns="http://maven.apache.org/EXTENSIONS/1.0.0"
|
||||||
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/EXTENSIONS/1.0.0 http://maven.apache.org/xsd/core-extensions-1.0.0.xsd">
|
||||||
|
<extension>
|
||||||
|
<groupId>fr.brouillard.oss</groupId>
|
||||||
|
<artifactId>jgitver-maven-plugin</artifactId>
|
||||||
|
<version>1.7.1</version>
|
||||||
|
</extension>
|
||||||
|
</extensions>
|
||||||
14
.mvn/jgitver.config.xml
Normal file
14
.mvn/jgitver.config.xml
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
<configuration xmlns="http://jgitver.github.io/maven/configuration/1.1.0"
|
||||||
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://jgitver.github.io/maven/configuration/1.1.0 https://jgitver.github.io/maven/configuration/jgitver-configuration-v1_1_0.xsd">
|
||||||
|
<useDirty>true</useDirty>
|
||||||
|
<useDefaultBranchingPolicy>false</useDefaultBranchingPolicy>
|
||||||
|
<branchPolicies>
|
||||||
|
<branchPolicy>
|
||||||
|
<pattern>(.*)</pattern>
|
||||||
|
<transformations>
|
||||||
|
<transformation>IGNORE</transformation>
|
||||||
|
</transformations>
|
||||||
|
</branchPolicy>
|
||||||
|
</branchPolicies>
|
||||||
|
</configuration>
|
||||||
BIN
.mvn/wrapper/maven-wrapper.jar
vendored
Normal file
BIN
.mvn/wrapper/maven-wrapper.jar
vendored
Normal file
Binary file not shown.
18
.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
18
.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
# or more contributor license agreements. See the NOTICE file
|
||||||
|
# distributed with this work for additional information
|
||||||
|
# regarding copyright ownership. The ASF licenses this file
|
||||||
|
# to you under the Apache License, Version 2.0 (the
|
||||||
|
# "License"); you may not use this file except in compliance
|
||||||
|
# with the License. You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing,
|
||||||
|
# software distributed under the License is distributed on an
|
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||||
|
# KIND, either express or implied. See the License for the
|
||||||
|
# specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.8.5/apache-maven-3.8.5-bin.zip
|
||||||
|
wrapperUrl=https://repo.maven.apache.org/maven2/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar
|
||||||
661
LICENSE
Normal file
661
LICENSE
Normal file
@@ -0,0 +1,661 @@
|
|||||||
|
GNU AFFERO GENERAL PUBLIC LICENSE
|
||||||
|
Version 3, 19 November 2007
|
||||||
|
|
||||||
|
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The GNU Affero General Public License is a free, copyleft license for
|
||||||
|
software and other kinds of works, specifically designed to ensure
|
||||||
|
cooperation with the community in the case of network server software.
|
||||||
|
|
||||||
|
The licenses for most software and other practical works are designed
|
||||||
|
to take away your freedom to share and change the works. By contrast,
|
||||||
|
our General Public Licenses are intended to guarantee your freedom to
|
||||||
|
share and change all versions of a program--to make sure it remains free
|
||||||
|
software for all its users.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
them if you wish), that you receive source code or can get it if you
|
||||||
|
want it, that you can change the software or use pieces of it in new
|
||||||
|
free programs, and that you know you can do these things.
|
||||||
|
|
||||||
|
Developers that use our General Public Licenses protect your rights
|
||||||
|
with two steps: (1) assert copyright on the software, and (2) offer
|
||||||
|
you this License which gives you legal permission to copy, distribute
|
||||||
|
and/or modify the software.
|
||||||
|
|
||||||
|
A secondary benefit of defending all users' freedom is that
|
||||||
|
improvements made in alternate versions of the program, if they
|
||||||
|
receive widespread use, become available for other developers to
|
||||||
|
incorporate. Many developers of free software are heartened and
|
||||||
|
encouraged by the resulting cooperation. However, in the case of
|
||||||
|
software used on network servers, this result may fail to come about.
|
||||||
|
The GNU General Public License permits making a modified version and
|
||||||
|
letting the public access it on a server without ever releasing its
|
||||||
|
source code to the public.
|
||||||
|
|
||||||
|
The GNU Affero General Public License is designed specifically to
|
||||||
|
ensure that, in such cases, the modified source code becomes available
|
||||||
|
to the community. It requires the operator of a network server to
|
||||||
|
provide the source code of the modified version running there to the
|
||||||
|
users of that server. Therefore, public use of a modified version, on
|
||||||
|
a publicly accessible server, gives the public access to the source
|
||||||
|
code of the modified version.
|
||||||
|
|
||||||
|
An older license, called the Affero General Public License and
|
||||||
|
published by Affero, was designed to accomplish similar goals. This is
|
||||||
|
a different license, not a version of the Affero GPL, but Affero has
|
||||||
|
released a new version of the Affero GPL which permits relicensing under
|
||||||
|
this license.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
0. Definitions.
|
||||||
|
|
||||||
|
"This License" refers to version 3 of the GNU Affero General Public License.
|
||||||
|
|
||||||
|
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||||
|
works, such as semiconductor masks.
|
||||||
|
|
||||||
|
"The Program" refers to any copyrightable work licensed under this
|
||||||
|
License. Each licensee is addressed as "you". "Licensees" and
|
||||||
|
"recipients" may be individuals or organizations.
|
||||||
|
|
||||||
|
To "modify" a work means to copy from or adapt all or part of the work
|
||||||
|
in a fashion requiring copyright permission, other than the making of an
|
||||||
|
exact copy. The resulting work is called a "modified version" of the
|
||||||
|
earlier work or a work "based on" the earlier work.
|
||||||
|
|
||||||
|
A "covered work" means either the unmodified Program or a work based
|
||||||
|
on the Program.
|
||||||
|
|
||||||
|
To "propagate" a work means to do anything with it that, without
|
||||||
|
permission, would make you directly or secondarily liable for
|
||||||
|
infringement under applicable copyright law, except executing it on a
|
||||||
|
computer or modifying a private copy. Propagation includes copying,
|
||||||
|
distribution (with or without modification), making available to the
|
||||||
|
public, and in some countries other activities as well.
|
||||||
|
|
||||||
|
To "convey" a work means any kind of propagation that enables other
|
||||||
|
parties to make or receive copies. Mere interaction with a user through
|
||||||
|
a computer network, with no transfer of a copy, is not conveying.
|
||||||
|
|
||||||
|
An interactive user interface displays "Appropriate Legal Notices"
|
||||||
|
to the extent that it includes a convenient and prominently visible
|
||||||
|
feature that (1) displays an appropriate copyright notice, and (2)
|
||||||
|
tells the user that there is no warranty for the work (except to the
|
||||||
|
extent that warranties are provided), that licensees may convey the
|
||||||
|
work under this License, and how to view a copy of this License. If
|
||||||
|
the interface presents a list of user commands or options, such as a
|
||||||
|
menu, a prominent item in the list meets this criterion.
|
||||||
|
|
||||||
|
1. Source Code.
|
||||||
|
|
||||||
|
The "source code" for a work means the preferred form of the work
|
||||||
|
for making modifications to it. "Object code" means any non-source
|
||||||
|
form of a work.
|
||||||
|
|
||||||
|
A "Standard Interface" means an interface that either is an official
|
||||||
|
standard defined by a recognized standards body, or, in the case of
|
||||||
|
interfaces specified for a particular programming language, one that
|
||||||
|
is widely used among developers working in that language.
|
||||||
|
|
||||||
|
The "System Libraries" of an executable work include anything, other
|
||||||
|
than the work as a whole, that (a) is included in the normal form of
|
||||||
|
packaging a Major Component, but which is not part of that Major
|
||||||
|
Component, and (b) serves only to enable use of the work with that
|
||||||
|
Major Component, or to implement a Standard Interface for which an
|
||||||
|
implementation is available to the public in source code form. A
|
||||||
|
"Major Component", in this context, means a major essential component
|
||||||
|
(kernel, window system, and so on) of the specific operating system
|
||||||
|
(if any) on which the executable work runs, or a compiler used to
|
||||||
|
produce the work, or an object code interpreter used to run it.
|
||||||
|
|
||||||
|
The "Corresponding Source" for a work in object code form means all
|
||||||
|
the source code needed to generate, install, and (for an executable
|
||||||
|
work) run the object code and to modify the work, including scripts to
|
||||||
|
control those activities. However, it does not include the work's
|
||||||
|
System Libraries, or general-purpose tools or generally available free
|
||||||
|
programs which are used unmodified in performing those activities but
|
||||||
|
which are not part of the work. For example, Corresponding Source
|
||||||
|
includes interface definition files associated with source files for
|
||||||
|
the work, and the source code for shared libraries and dynamically
|
||||||
|
linked subprograms that the work is specifically designed to require,
|
||||||
|
such as by intimate data communication or control flow between those
|
||||||
|
subprograms and other parts of the work.
|
||||||
|
|
||||||
|
The Corresponding Source need not include anything that users
|
||||||
|
can regenerate automatically from other parts of the Corresponding
|
||||||
|
Source.
|
||||||
|
|
||||||
|
The Corresponding Source for a work in source code form is that
|
||||||
|
same work.
|
||||||
|
|
||||||
|
2. Basic Permissions.
|
||||||
|
|
||||||
|
All rights granted under this License are granted for the term of
|
||||||
|
copyright on the Program, and are irrevocable provided the stated
|
||||||
|
conditions are met. This License explicitly affirms your unlimited
|
||||||
|
permission to run the unmodified Program. The output from running a
|
||||||
|
covered work is covered by this License only if the output, given its
|
||||||
|
content, constitutes a covered work. This License acknowledges your
|
||||||
|
rights of fair use or other equivalent, as provided by copyright law.
|
||||||
|
|
||||||
|
You may make, run and propagate covered works that you do not
|
||||||
|
convey, without conditions so long as your license otherwise remains
|
||||||
|
in force. You may convey covered works to others for the sole purpose
|
||||||
|
of having them make modifications exclusively for you, or provide you
|
||||||
|
with facilities for running those works, provided that you comply with
|
||||||
|
the terms of this License in conveying all material for which you do
|
||||||
|
not control copyright. Those thus making or running the covered works
|
||||||
|
for you must do so exclusively on your behalf, under your direction
|
||||||
|
and control, on terms that prohibit them from making any copies of
|
||||||
|
your copyrighted material outside their relationship with you.
|
||||||
|
|
||||||
|
Conveying under any other circumstances is permitted solely under
|
||||||
|
the conditions stated below. Sublicensing is not allowed; section 10
|
||||||
|
makes it unnecessary.
|
||||||
|
|
||||||
|
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||||
|
|
||||||
|
No covered work shall be deemed part of an effective technological
|
||||||
|
measure under any applicable law fulfilling obligations under article
|
||||||
|
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||||
|
similar laws prohibiting or restricting circumvention of such
|
||||||
|
measures.
|
||||||
|
|
||||||
|
When you convey a covered work, you waive any legal power to forbid
|
||||||
|
circumvention of technological measures to the extent such circumvention
|
||||||
|
is effected by exercising rights under this License with respect to
|
||||||
|
the covered work, and you disclaim any intention to limit operation or
|
||||||
|
modification of the work as a means of enforcing, against the work's
|
||||||
|
users, your or third parties' legal rights to forbid circumvention of
|
||||||
|
technological measures.
|
||||||
|
|
||||||
|
4. Conveying Verbatim Copies.
|
||||||
|
|
||||||
|
You may convey verbatim copies of the Program's source code as you
|
||||||
|
receive it, in any medium, provided that you conspicuously and
|
||||||
|
appropriately publish on each copy an appropriate copyright notice;
|
||||||
|
keep intact all notices stating that this License and any
|
||||||
|
non-permissive terms added in accord with section 7 apply to the code;
|
||||||
|
keep intact all notices of the absence of any warranty; and give all
|
||||||
|
recipients a copy of this License along with the Program.
|
||||||
|
|
||||||
|
You may charge any price or no price for each copy that you convey,
|
||||||
|
and you may offer support or warranty protection for a fee.
|
||||||
|
|
||||||
|
5. Conveying Modified Source Versions.
|
||||||
|
|
||||||
|
You may convey a work based on the Program, or the modifications to
|
||||||
|
produce it from the Program, in the form of source code under the
|
||||||
|
terms of section 4, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) The work must carry prominent notices stating that you modified
|
||||||
|
it, and giving a relevant date.
|
||||||
|
|
||||||
|
b) The work must carry prominent notices stating that it is
|
||||||
|
released under this License and any conditions added under section
|
||||||
|
7. This requirement modifies the requirement in section 4 to
|
||||||
|
"keep intact all notices".
|
||||||
|
|
||||||
|
c) You must license the entire work, as a whole, under this
|
||||||
|
License to anyone who comes into possession of a copy. This
|
||||||
|
License will therefore apply, along with any applicable section 7
|
||||||
|
additional terms, to the whole of the work, and all its parts,
|
||||||
|
regardless of how they are packaged. This License gives no
|
||||||
|
permission to license the work in any other way, but it does not
|
||||||
|
invalidate such permission if you have separately received it.
|
||||||
|
|
||||||
|
d) If the work has interactive user interfaces, each must display
|
||||||
|
Appropriate Legal Notices; however, if the Program has interactive
|
||||||
|
interfaces that do not display Appropriate Legal Notices, your
|
||||||
|
work need not make them do so.
|
||||||
|
|
||||||
|
A compilation of a covered work with other separate and independent
|
||||||
|
works, which are not by their nature extensions of the covered work,
|
||||||
|
and which are not combined with it such as to form a larger program,
|
||||||
|
in or on a volume of a storage or distribution medium, is called an
|
||||||
|
"aggregate" if the compilation and its resulting copyright are not
|
||||||
|
used to limit the access or legal rights of the compilation's users
|
||||||
|
beyond what the individual works permit. Inclusion of a covered work
|
||||||
|
in an aggregate does not cause this License to apply to the other
|
||||||
|
parts of the aggregate.
|
||||||
|
|
||||||
|
6. Conveying Non-Source Forms.
|
||||||
|
|
||||||
|
You may convey a covered work in object code form under the terms
|
||||||
|
of sections 4 and 5, provided that you also convey the
|
||||||
|
machine-readable Corresponding Source under the terms of this License,
|
||||||
|
in one of these ways:
|
||||||
|
|
||||||
|
a) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by the
|
||||||
|
Corresponding Source fixed on a durable physical medium
|
||||||
|
customarily used for software interchange.
|
||||||
|
|
||||||
|
b) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by a
|
||||||
|
written offer, valid for at least three years and valid for as
|
||||||
|
long as you offer spare parts or customer support for that product
|
||||||
|
model, to give anyone who possesses the object code either (1) a
|
||||||
|
copy of the Corresponding Source for all the software in the
|
||||||
|
product that is covered by this License, on a durable physical
|
||||||
|
medium customarily used for software interchange, for a price no
|
||||||
|
more than your reasonable cost of physically performing this
|
||||||
|
conveying of source, or (2) access to copy the
|
||||||
|
Corresponding Source from a network server at no charge.
|
||||||
|
|
||||||
|
c) Convey individual copies of the object code with a copy of the
|
||||||
|
written offer to provide the Corresponding Source. This
|
||||||
|
alternative is allowed only occasionally and noncommercially, and
|
||||||
|
only if you received the object code with such an offer, in accord
|
||||||
|
with subsection 6b.
|
||||||
|
|
||||||
|
d) Convey the object code by offering access from a designated
|
||||||
|
place (gratis or for a charge), and offer equivalent access to the
|
||||||
|
Corresponding Source in the same way through the same place at no
|
||||||
|
further charge. You need not require recipients to copy the
|
||||||
|
Corresponding Source along with the object code. If the place to
|
||||||
|
copy the object code is a network server, the Corresponding Source
|
||||||
|
may be on a different server (operated by you or a third party)
|
||||||
|
that supports equivalent copying facilities, provided you maintain
|
||||||
|
clear directions next to the object code saying where to find the
|
||||||
|
Corresponding Source. Regardless of what server hosts the
|
||||||
|
Corresponding Source, you remain obligated to ensure that it is
|
||||||
|
available for as long as needed to satisfy these requirements.
|
||||||
|
|
||||||
|
e) Convey the object code using peer-to-peer transmission, provided
|
||||||
|
you inform other peers where the object code and Corresponding
|
||||||
|
Source of the work are being offered to the general public at no
|
||||||
|
charge under subsection 6d.
|
||||||
|
|
||||||
|
A separable portion of the object code, whose source code is excluded
|
||||||
|
from the Corresponding Source as a System Library, need not be
|
||||||
|
included in conveying the object code work.
|
||||||
|
|
||||||
|
A "User Product" is either (1) a "consumer product", which means any
|
||||||
|
tangible personal property which is normally used for personal, family,
|
||||||
|
or household purposes, or (2) anything designed or sold for incorporation
|
||||||
|
into a dwelling. In determining whether a product is a consumer product,
|
||||||
|
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||||
|
product received by a particular user, "normally used" refers to a
|
||||||
|
typical or common use of that class of product, regardless of the status
|
||||||
|
of the particular user or of the way in which the particular user
|
||||||
|
actually uses, or expects or is expected to use, the product. A product
|
||||||
|
is a consumer product regardless of whether the product has substantial
|
||||||
|
commercial, industrial or non-consumer uses, unless such uses represent
|
||||||
|
the only significant mode of use of the product.
|
||||||
|
|
||||||
|
"Installation Information" for a User Product means any methods,
|
||||||
|
procedures, authorization keys, or other information required to install
|
||||||
|
and execute modified versions of a covered work in that User Product from
|
||||||
|
a modified version of its Corresponding Source. The information must
|
||||||
|
suffice to ensure that the continued functioning of the modified object
|
||||||
|
code is in no case prevented or interfered with solely because
|
||||||
|
modification has been made.
|
||||||
|
|
||||||
|
If you convey an object code work under this section in, or with, or
|
||||||
|
specifically for use in, a User Product, and the conveying occurs as
|
||||||
|
part of a transaction in which the right of possession and use of the
|
||||||
|
User Product is transferred to the recipient in perpetuity or for a
|
||||||
|
fixed term (regardless of how the transaction is characterized), the
|
||||||
|
Corresponding Source conveyed under this section must be accompanied
|
||||||
|
by the Installation Information. But this requirement does not apply
|
||||||
|
if neither you nor any third party retains the ability to install
|
||||||
|
modified object code on the User Product (for example, the work has
|
||||||
|
been installed in ROM).
|
||||||
|
|
||||||
|
The requirement to provide Installation Information does not include a
|
||||||
|
requirement to continue to provide support service, warranty, or updates
|
||||||
|
for a work that has been modified or installed by the recipient, or for
|
||||||
|
the User Product in which it has been modified or installed. Access to a
|
||||||
|
network may be denied when the modification itself materially and
|
||||||
|
adversely affects the operation of the network or violates the rules and
|
||||||
|
protocols for communication across the network.
|
||||||
|
|
||||||
|
Corresponding Source conveyed, and Installation Information provided,
|
||||||
|
in accord with this section must be in a format that is publicly
|
||||||
|
documented (and with an implementation available to the public in
|
||||||
|
source code form), and must require no special password or key for
|
||||||
|
unpacking, reading or copying.
|
||||||
|
|
||||||
|
7. Additional Terms.
|
||||||
|
|
||||||
|
"Additional permissions" are terms that supplement the terms of this
|
||||||
|
License by making exceptions from one or more of its conditions.
|
||||||
|
Additional permissions that are applicable to the entire Program shall
|
||||||
|
be treated as though they were included in this License, to the extent
|
||||||
|
that they are valid under applicable law. If additional permissions
|
||||||
|
apply only to part of the Program, that part may be used separately
|
||||||
|
under those permissions, but the entire Program remains governed by
|
||||||
|
this License without regard to the additional permissions.
|
||||||
|
|
||||||
|
When you convey a copy of a covered work, you may at your option
|
||||||
|
remove any additional permissions from that copy, or from any part of
|
||||||
|
it. (Additional permissions may be written to require their own
|
||||||
|
removal in certain cases when you modify the work.) You may place
|
||||||
|
additional permissions on material, added by you to a covered work,
|
||||||
|
for which you have or can give appropriate copyright permission.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, for material you
|
||||||
|
add to a covered work, you may (if authorized by the copyright holders of
|
||||||
|
that material) supplement the terms of this License with terms:
|
||||||
|
|
||||||
|
a) Disclaiming warranty or limiting liability differently from the
|
||||||
|
terms of sections 15 and 16 of this License; or
|
||||||
|
|
||||||
|
b) Requiring preservation of specified reasonable legal notices or
|
||||||
|
author attributions in that material or in the Appropriate Legal
|
||||||
|
Notices displayed by works containing it; or
|
||||||
|
|
||||||
|
c) Prohibiting misrepresentation of the origin of that material, or
|
||||||
|
requiring that modified versions of such material be marked in
|
||||||
|
reasonable ways as different from the original version; or
|
||||||
|
|
||||||
|
d) Limiting the use for publicity purposes of names of licensors or
|
||||||
|
authors of the material; or
|
||||||
|
|
||||||
|
e) Declining to grant rights under trademark law for use of some
|
||||||
|
trade names, trademarks, or service marks; or
|
||||||
|
|
||||||
|
f) Requiring indemnification of licensors and authors of that
|
||||||
|
material by anyone who conveys the material (or modified versions of
|
||||||
|
it) with contractual assumptions of liability to the recipient, for
|
||||||
|
any liability that these contractual assumptions directly impose on
|
||||||
|
those licensors and authors.
|
||||||
|
|
||||||
|
All other non-permissive additional terms are considered "further
|
||||||
|
restrictions" within the meaning of section 10. If the Program as you
|
||||||
|
received it, or any part of it, contains a notice stating that it is
|
||||||
|
governed by this License along with a term that is a further
|
||||||
|
restriction, you may remove that term. If a license document contains
|
||||||
|
a further restriction but permits relicensing or conveying under this
|
||||||
|
License, you may add to a covered work material governed by the terms
|
||||||
|
of that license document, provided that the further restriction does
|
||||||
|
not survive such relicensing or conveying.
|
||||||
|
|
||||||
|
If you add terms to a covered work in accord with this section, you
|
||||||
|
must place, in the relevant source files, a statement of the
|
||||||
|
additional terms that apply to those files, or a notice indicating
|
||||||
|
where to find the applicable terms.
|
||||||
|
|
||||||
|
Additional terms, permissive or non-permissive, may be stated in the
|
||||||
|
form of a separately written license, or stated as exceptions;
|
||||||
|
the above requirements apply either way.
|
||||||
|
|
||||||
|
8. Termination.
|
||||||
|
|
||||||
|
You may not propagate or modify a covered work except as expressly
|
||||||
|
provided under this License. Any attempt otherwise to propagate or
|
||||||
|
modify it is void, and will automatically terminate your rights under
|
||||||
|
this License (including any patent licenses granted under the third
|
||||||
|
paragraph of section 11).
|
||||||
|
|
||||||
|
However, if you cease all violation of this License, then your
|
||||||
|
license from a particular copyright holder is reinstated (a)
|
||||||
|
provisionally, unless and until the copyright holder explicitly and
|
||||||
|
finally terminates your license, and (b) permanently, if the copyright
|
||||||
|
holder fails to notify you of the violation by some reasonable means
|
||||||
|
prior to 60 days after the cessation.
|
||||||
|
|
||||||
|
Moreover, your license from a particular copyright holder is
|
||||||
|
reinstated permanently if the copyright holder notifies you of the
|
||||||
|
violation by some reasonable means, this is the first time you have
|
||||||
|
received notice of violation of this License (for any work) from that
|
||||||
|
copyright holder, and you cure the violation prior to 30 days after
|
||||||
|
your receipt of the notice.
|
||||||
|
|
||||||
|
Termination of your rights under this section does not terminate the
|
||||||
|
licenses of parties who have received copies or rights from you under
|
||||||
|
this License. If your rights have been terminated and not permanently
|
||||||
|
reinstated, you do not qualify to receive new licenses for the same
|
||||||
|
material under section 10.
|
||||||
|
|
||||||
|
9. Acceptance Not Required for Having Copies.
|
||||||
|
|
||||||
|
You are not required to accept this License in order to receive or
|
||||||
|
run a copy of the Program. Ancillary propagation of a covered work
|
||||||
|
occurring solely as a consequence of using peer-to-peer transmission
|
||||||
|
to receive a copy likewise does not require acceptance. However,
|
||||||
|
nothing other than this License grants you permission to propagate or
|
||||||
|
modify any covered work. These actions infringe copyright if you do
|
||||||
|
not accept this License. Therefore, by modifying or propagating a
|
||||||
|
covered work, you indicate your acceptance of this License to do so.
|
||||||
|
|
||||||
|
10. Automatic Licensing of Downstream Recipients.
|
||||||
|
|
||||||
|
Each time you convey a covered work, the recipient automatically
|
||||||
|
receives a license from the original licensors, to run, modify and
|
||||||
|
propagate that work, subject to this License. You are not responsible
|
||||||
|
for enforcing compliance by third parties with this License.
|
||||||
|
|
||||||
|
An "entity transaction" is a transaction transferring control of an
|
||||||
|
organization, or substantially all assets of one, or subdividing an
|
||||||
|
organization, or merging organizations. If propagation of a covered
|
||||||
|
work results from an entity transaction, each party to that
|
||||||
|
transaction who receives a copy of the work also receives whatever
|
||||||
|
licenses to the work the party's predecessor in interest had or could
|
||||||
|
give under the previous paragraph, plus a right to possession of the
|
||||||
|
Corresponding Source of the work from the predecessor in interest, if
|
||||||
|
the predecessor has it or can get it with reasonable efforts.
|
||||||
|
|
||||||
|
You may not impose any further restrictions on the exercise of the
|
||||||
|
rights granted or affirmed under this License. For example, you may
|
||||||
|
not impose a license fee, royalty, or other charge for exercise of
|
||||||
|
rights granted under this License, and you may not initiate litigation
|
||||||
|
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||||
|
any patent claim is infringed by making, using, selling, offering for
|
||||||
|
sale, or importing the Program or any portion of it.
|
||||||
|
|
||||||
|
11. Patents.
|
||||||
|
|
||||||
|
A "contributor" is a copyright holder who authorizes use under this
|
||||||
|
License of the Program or a work on which the Program is based. The
|
||||||
|
work thus licensed is called the contributor's "contributor version".
|
||||||
|
|
||||||
|
A contributor's "essential patent claims" are all patent claims
|
||||||
|
owned or controlled by the contributor, whether already acquired or
|
||||||
|
hereafter acquired, that would be infringed by some manner, permitted
|
||||||
|
by this License, of making, using, or selling its contributor version,
|
||||||
|
but do not include claims that would be infringed only as a
|
||||||
|
consequence of further modification of the contributor version. For
|
||||||
|
purposes of this definition, "control" includes the right to grant
|
||||||
|
patent sublicenses in a manner consistent with the requirements of
|
||||||
|
this License.
|
||||||
|
|
||||||
|
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||||
|
patent license under the contributor's essential patent claims, to
|
||||||
|
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||||
|
propagate the contents of its contributor version.
|
||||||
|
|
||||||
|
In the following three paragraphs, a "patent license" is any express
|
||||||
|
agreement or commitment, however denominated, not to enforce a patent
|
||||||
|
(such as an express permission to practice a patent or covenant not to
|
||||||
|
sue for patent infringement). To "grant" such a patent license to a
|
||||||
|
party means to make such an agreement or commitment not to enforce a
|
||||||
|
patent against the party.
|
||||||
|
|
||||||
|
If you convey a covered work, knowingly relying on a patent license,
|
||||||
|
and the Corresponding Source of the work is not available for anyone
|
||||||
|
to copy, free of charge and under the terms of this License, through a
|
||||||
|
publicly available network server or other readily accessible means,
|
||||||
|
then you must either (1) cause the Corresponding Source to be so
|
||||||
|
available, or (2) arrange to deprive yourself of the benefit of the
|
||||||
|
patent license for this particular work, or (3) arrange, in a manner
|
||||||
|
consistent with the requirements of this License, to extend the patent
|
||||||
|
license to downstream recipients. "Knowingly relying" means you have
|
||||||
|
actual knowledge that, but for the patent license, your conveying the
|
||||||
|
covered work in a country, or your recipient's use of the covered work
|
||||||
|
in a country, would infringe one or more identifiable patents in that
|
||||||
|
country that you have reason to believe are valid.
|
||||||
|
|
||||||
|
If, pursuant to or in connection with a single transaction or
|
||||||
|
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||||
|
covered work, and grant a patent license to some of the parties
|
||||||
|
receiving the covered work authorizing them to use, propagate, modify
|
||||||
|
or convey a specific copy of the covered work, then the patent license
|
||||||
|
you grant is automatically extended to all recipients of the covered
|
||||||
|
work and works based on it.
|
||||||
|
|
||||||
|
A patent license is "discriminatory" if it does not include within
|
||||||
|
the scope of its coverage, prohibits the exercise of, or is
|
||||||
|
conditioned on the non-exercise of one or more of the rights that are
|
||||||
|
specifically granted under this License. You may not convey a covered
|
||||||
|
work if you are a party to an arrangement with a third party that is
|
||||||
|
in the business of distributing software, under which you make payment
|
||||||
|
to the third party based on the extent of your activity of conveying
|
||||||
|
the work, and under which the third party grants, to any of the
|
||||||
|
parties who would receive the covered work from you, a discriminatory
|
||||||
|
patent license (a) in connection with copies of the covered work
|
||||||
|
conveyed by you (or copies made from those copies), or (b) primarily
|
||||||
|
for and in connection with specific products or compilations that
|
||||||
|
contain the covered work, unless you entered into that arrangement,
|
||||||
|
or that patent license was granted, prior to 28 March 2007.
|
||||||
|
|
||||||
|
Nothing in this License shall be construed as excluding or limiting
|
||||||
|
any implied license or other defenses to infringement that may
|
||||||
|
otherwise be available to you under applicable patent law.
|
||||||
|
|
||||||
|
12. No Surrender of Others' Freedom.
|
||||||
|
|
||||||
|
If conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot convey a
|
||||||
|
covered work so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you may
|
||||||
|
not convey it at all. For example, if you agree to terms that obligate you
|
||||||
|
to collect a royalty for further conveying from those to whom you convey
|
||||||
|
the Program, the only way you could satisfy both those terms and this
|
||||||
|
License would be to refrain entirely from conveying the Program.
|
||||||
|
|
||||||
|
13. Remote Network Interaction; Use with the GNU General Public License.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, if you modify the
|
||||||
|
Program, your modified version must prominently offer all users
|
||||||
|
interacting with it remotely through a computer network (if your version
|
||||||
|
supports such interaction) an opportunity to receive the Corresponding
|
||||||
|
Source of your version by providing access to the Corresponding Source
|
||||||
|
from a network server at no charge, through some standard or customary
|
||||||
|
means of facilitating copying of software. This Corresponding Source
|
||||||
|
shall include the Corresponding Source for any work covered by version 3
|
||||||
|
of the GNU General Public License that is incorporated pursuant to the
|
||||||
|
following paragraph.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, you have
|
||||||
|
permission to link or combine any covered work with a work licensed
|
||||||
|
under version 3 of the GNU General Public License into a single
|
||||||
|
combined work, and to convey the resulting work. The terms of this
|
||||||
|
License will continue to apply to the part which is the covered work,
|
||||||
|
but the work with which it is combined will remain governed by version
|
||||||
|
3 of the GNU General Public License.
|
||||||
|
|
||||||
|
14. Revised Versions of this License.
|
||||||
|
|
||||||
|
The Free Software Foundation may publish revised and/or new versions of
|
||||||
|
the GNU Affero General Public License from time to time. Such new versions
|
||||||
|
will be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the
|
||||||
|
Program specifies that a certain numbered version of the GNU Affero General
|
||||||
|
Public License "or any later version" applies to it, you have the
|
||||||
|
option of following the terms and conditions either of that numbered
|
||||||
|
version or of any later version published by the Free Software
|
||||||
|
Foundation. If the Program does not specify a version number of the
|
||||||
|
GNU Affero General Public License, you may choose any version ever published
|
||||||
|
by the Free Software Foundation.
|
||||||
|
|
||||||
|
If the Program specifies that a proxy can decide which future
|
||||||
|
versions of the GNU Affero General Public License can be used, that proxy's
|
||||||
|
public statement of acceptance of a version permanently authorizes you
|
||||||
|
to choose that version for the Program.
|
||||||
|
|
||||||
|
Later license versions may give you additional or different
|
||||||
|
permissions. However, no additional obligations are imposed on any
|
||||||
|
author or copyright holder as a result of your choosing to follow a
|
||||||
|
later version.
|
||||||
|
|
||||||
|
15. Disclaimer of Warranty.
|
||||||
|
|
||||||
|
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||||
|
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||||
|
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||||
|
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||||
|
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||||
|
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
16. Limitation of Liability.
|
||||||
|
|
||||||
|
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||||
|
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||||
|
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||||
|
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||||
|
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||||
|
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||||
|
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||||
|
SUCH DAMAGES.
|
||||||
|
|
||||||
|
17. Interpretation of Sections 15 and 16.
|
||||||
|
|
||||||
|
If the disclaimer of warranty and limitation of liability provided
|
||||||
|
above cannot be given local legal effect according to their terms,
|
||||||
|
reviewing courts shall apply local law that most closely approximates
|
||||||
|
an absolute waiver of all civil liability in connection with the
|
||||||
|
Program, unless a warranty or assumption of liability accompanies a
|
||||||
|
copy of the Program in return for a fee.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
state the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
This program is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Affero General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Affero General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Affero General Public License
|
||||||
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If your software can interact with users remotely through a computer
|
||||||
|
network, you should also make sure that it provides a way for users to
|
||||||
|
get its source. For example, if your program is a web application, its
|
||||||
|
interface could display a "Source" link that leads users to an archive
|
||||||
|
of the code. There are many ways you could offer source, and different
|
||||||
|
solutions will be better for different programs; see section 13 for the
|
||||||
|
specific requirements.
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or school,
|
||||||
|
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||||
|
For more information on this, and how to apply and follow the GNU AGPL, see
|
||||||
|
<https://www.gnu.org/licenses/>.
|
||||||
2
Procfile
2
Procfile
@@ -1,2 +0,0 @@
|
|||||||
web: java $JAVA_OPTS -Ddw.http.port=$PORT -Ddw.http.adminPort=$PORT -Ddw.federation.name=$FEDERATION_NAME -Ddw.federation.herokuPeers="$FEDERATED_PEERS" -Ddw.twilio.accountId=$TWILIO_ACCOUNT_SID -Ddw.twilio.accountToken=$TWILIO_ACCOUNT_TOKEN -Ddw.twilio.number=$TWILIO_NUMBER -Ddw.nexmo.apiKey=$NEXMO_KEY -Ddw.nexmo.apiSecret=$NEXMO_SECRET -Ddw.nexmo.number=$NEXMO_NUMBER -Ddw.gcm.apiKey=$GCM_KEY -Ddw.apn.certificate="$APN_CERTIFICATE" -Ddw.apn.key="$APN_KEY" -Ddw.s3.accessKey=$AWS_ACCESS_KEY -Ddw.s3.accessSecret=$AWS_SECRET_KEY -Ddw.s3.attachmentsBucket=$AWS_ATTACHMENTS_BUCKET -Ddw.memcache.servers=$MEMCACHIER_SERVERS -Ddw.memcache.user=$MEMCACHIER_USERNAME -Ddw.memcache.password=$MEMCACHIER_PASSWORD -Ddw.redis.url=$REDIS_URL -Ddw.database.driverClass=org.postgresql.Driver -Ddw.database.user=`echo $DATABASE_URL | awk -F'://' {'print $2'} | awk -F':' {'print $1'}` -Ddw.database.password=`echo $DATABASE_URL | awk -F'://' {'print $2'} | awk -F':' {'print $2'} | awk -F'@' {'print $1'}` -Ddw.database.url=jdbc:postgresql://`echo $DATABASE_URL | awk -F'@' {'print $2'}` -jar target/TextSecure-MGCM-1.0-SNAPSHOT.jar server
|
|
||||||
dir: java $JAVA_OPTS -Ddw.http.port=$PORT -Ddw.http.adminPort=$PORT -Ddw.federation.name=$FEDERATION_NAME -Ddw.federation.herokuPeers="$FEDERATED_PEERS" -Ddw.twilio.accountId=$TWILIO_ACCOUNT_SID -Ddw.twilio.accountToken=$TWILIO_ACCOUNT_TOKEN -Ddw.twilio.number=$TWILIO_NUMBER -Ddw.nexmo.apiKey=$NEXMO_KEY -Ddw.nexmo.apiSecret=$NEXMO_SECRET -Ddw.nexmo.number=$NEXMO_NUMBER -Ddw.gcm.apiKey=$GCM_KEY -Ddw.apn.certificate="$APN_CERTIFICATE" -Ddw.apn.key="$APN_KEY" -Ddw.s3.accessKey=$AWS_ACCESS_KEY -Ddw.s3.accessSecret=$AWS_SECRET_KEY -Ddw.s3.attachmentsBucket=$AWS_ATTACHMENTS_BUCKET -Ddw.memcache.servers=$MEMCACHIER_SERVERS -Ddw.memcache.user=$MEMCACHIER_USERNAME -Ddw.memcache.password=$MEMCACHIER_PASSWORD -Ddw.redis.url=$REDIS_URL -Ddw.database.driverClass=org.postgresql.Driver -Ddw.database.user=`echo $DATABASE_URL | awk -F'://' {'print $2'} | awk -F':' {'print $1'}` -Ddw.database.password=`echo $DATABASE_URL | awk -F'://' {'print $2'} | awk -F':' {'print $2'} | awk -F'@' {'print $1'}` -Ddw.database.url=jdbc:postgresql://`echo $DATABASE_URL | awk -F'@' {'print $2'}` -jar target/TextSecure-MGCM-1.0-SNAPSHOT.jar directory
|
|
||||||
37
README.md
37
README.md
@@ -1,46 +1,19 @@
|
|||||||
TextSecure-Server
|
Signal-Server
|
||||||
=================
|
=================
|
||||||
|
|
||||||
The server that handles message routing for the
|
|
||||||
[TextSecure](https://github.com/whispersystems/TextSecure/) data channel. Communication
|
|
||||||
is handled by a REST API and Push messaging (both GCM and APN).
|
|
||||||
|
|
||||||
Documentation
|
Documentation
|
||||||
-------------
|
-------------
|
||||||
|
|
||||||
Looking for protocol documentation? Check out the wiki!
|
Looking for protocol documentation? Check out the website!
|
||||||
|
|
||||||
https://github.com/WhisperSystems/TextSecure-Server/wiki/API-Protocol
|
|
||||||
|
|
||||||
|
|
||||||
Bug tracker
|
|
||||||
-----------
|
|
||||||
|
|
||||||
Have a bug? Please create an issue here on GitHub!
|
|
||||||
|
|
||||||
https://github.com/WhisperSystems/TextSecure-Server/issues
|
|
||||||
|
|
||||||
|
|
||||||
Mailing list
|
|
||||||
------------
|
|
||||||
|
|
||||||
Have a question? Ask on our mailing list!
|
|
||||||
|
|
||||||
whispersystems@lists.riseup.net
|
|
||||||
|
|
||||||
https://lists.riseup.net/www/info/whispersystems
|
|
||||||
|
|
||||||
Current BitHub Payment Per Commit:
|
|
||||||
=================
|
|
||||||

|
|
||||||
|
|
||||||
|
https://signal.org/docs/
|
||||||
|
|
||||||
Cryptography Notice
|
Cryptography Notice
|
||||||
------------
|
------------
|
||||||
|
|
||||||
This distribution includes cryptographic software. The country in which you currently reside may have restrictions on the import, possession, use, and/or re-export to another country, of encryption software.
|
This distribution includes cryptographic software. The country in which you currently reside may have restrictions on the import, possession, use, and/or re-export to another country, of encryption software.
|
||||||
BEFORE using any encryption software, please check your country's laws, regulations and policies concerning the import, possession, or use, and re-export of encryption software, to see if this is permitted.
|
BEFORE using any encryption software, please check your country's laws, regulations and policies concerning the import, possession, or use, and re-export of encryption software, to see if this is permitted.
|
||||||
See <http://www.wassenaar.org/> for more information.
|
See <https://www.wassenaar.org/> for more information.
|
||||||
|
|
||||||
The U.S. Government Department of Commerce, Bureau of Industry and Security (BIS), has classified this software as Export Commodity Control Number (ECCN) 5D002.C.1, which includes information security software using or performing cryptographic functions with asymmetric algorithms.
|
The U.S. Government Department of Commerce, Bureau of Industry and Security (BIS), has classified this software as Export Commodity Control Number (ECCN) 5D002.C.1, which includes information security software using or performing cryptographic functions with asymmetric algorithms.
|
||||||
The form and manner of this distribution makes it eligible for export under the License Exception ENC Technology Software Unrestricted (TSU) exception (see the BIS Export Administration Regulations, Section 740.13) for both object code and source code.
|
The form and manner of this distribution makes it eligible for export under the License Exception ENC Technology Software Unrestricted (TSU) exception (see the BIS Export Administration Regulations, Section 740.13) for both object code and source code.
|
||||||
@@ -48,6 +21,6 @@ The form and manner of this distribution makes it eligible for export under the
|
|||||||
License
|
License
|
||||||
---------------------
|
---------------------
|
||||||
|
|
||||||
Copyright 2013 Open Whisper Systems
|
Copyright 2013-2022 Signal Messenger, LLC
|
||||||
|
|
||||||
Licensed under the AGPLv3: https://www.gnu.org/licenses/agpl-3.0.html
|
Licensed under the AGPLv3: https://www.gnu.org/licenses/agpl-3.0.html
|
||||||
|
|||||||
1
abusive-message-filter
Submodule
1
abusive-message-filter
Submodule
Submodule abusive-message-filter added at d85b037939
@@ -1,75 +0,0 @@
|
|||||||
twilio:
|
|
||||||
accountId:
|
|
||||||
accountToken:
|
|
||||||
number:
|
|
||||||
localDomain: # The domain Twilio can call back to.
|
|
||||||
international: # Boolean specifying Twilio for international delivery
|
|
||||||
|
|
||||||
# Optional. If specified, Nexmo will be used for non-US SMS and
|
|
||||||
# voice verification if twilio.international is false. Otherwise,
|
|
||||||
# Nexmo, if specified, Nexmo will only be used as a fallback
|
|
||||||
# for failed Twilio deliveries.
|
|
||||||
nexmo:
|
|
||||||
apiKey:
|
|
||||||
apiSecret:
|
|
||||||
number:
|
|
||||||
|
|
||||||
gcm:
|
|
||||||
senderId:
|
|
||||||
apiKey:
|
|
||||||
|
|
||||||
# Optional. Only if iOS clients are supported.
|
|
||||||
apn:
|
|
||||||
# In PEM format.
|
|
||||||
certificate:
|
|
||||||
|
|
||||||
# In PEM format.
|
|
||||||
key:
|
|
||||||
|
|
||||||
s3:
|
|
||||||
accessKey:
|
|
||||||
accessSecret:
|
|
||||||
|
|
||||||
# Name of the S3 bucket (needs to have been created)
|
|
||||||
# for attachments to go. Should be configured with
|
|
||||||
# correct permissions.
|
|
||||||
attachmentsBucket:
|
|
||||||
|
|
||||||
memcache:
|
|
||||||
servers:
|
|
||||||
user:
|
|
||||||
password:
|
|
||||||
|
|
||||||
redis:
|
|
||||||
url:
|
|
||||||
|
|
||||||
federation:
|
|
||||||
name:
|
|
||||||
peers:
|
|
||||||
-
|
|
||||||
name: somepeer
|
|
||||||
url: https://foo.com
|
|
||||||
authenticationToken: foo
|
|
||||||
certificate: in pem format
|
|
||||||
|
|
||||||
# Optional address of graphite server to report metrics
|
|
||||||
graphite:
|
|
||||||
host:
|
|
||||||
port:
|
|
||||||
|
|
||||||
database:
|
|
||||||
# the name of your JDBC driver
|
|
||||||
driverClass: org.postgresql.Driver
|
|
||||||
|
|
||||||
# the username
|
|
||||||
user:
|
|
||||||
|
|
||||||
# the password
|
|
||||||
password:
|
|
||||||
|
|
||||||
# the JDBC URL
|
|
||||||
url: jdbc:postgresql://somehost:somport/somedb
|
|
||||||
|
|
||||||
# any properties specific to your JDBC driver:
|
|
||||||
properties:
|
|
||||||
charSet: UTF-8
|
|
||||||
77
event-logger/pom.xml
Normal file
77
event-logger/pom.xml
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!--
|
||||||
|
~ Copyright 2022 Signal Messenger, LLC
|
||||||
|
~ SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
-->
|
||||||
|
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||||
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<parent>
|
||||||
|
<artifactId>TextSecureServer</artifactId>
|
||||||
|
<groupId>org.whispersystems.textsecure</groupId>
|
||||||
|
<version>JGITVER</version>
|
||||||
|
</parent>
|
||||||
|
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
<artifactId>event-logger</artifactId>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.cloud</groupId>
|
||||||
|
<artifactId>google-cloud-logging</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
|
<artifactId>kotlin-stdlib</artifactId>
|
||||||
|
<version>${kotlin.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlinx</groupId>
|
||||||
|
<artifactId>kotlinx-serialization-json</artifactId>
|
||||||
|
<version>${kotlinx-serialization.version}</version>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
<build>
|
||||||
|
<sourceDirectory>${project.basedir}/src/main/kotlin</sourceDirectory>
|
||||||
|
<testSourceDirectory>${project.basedir}/src/test/kotlin</testSourceDirectory>
|
||||||
|
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
|
<artifactId>kotlin-maven-plugin</artifactId>
|
||||||
|
<version>${kotlin.version}</version>
|
||||||
|
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>compile</id>
|
||||||
|
<goals>
|
||||||
|
<goal>compile</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
|
||||||
|
<execution>
|
||||||
|
<id>test-compile</id>
|
||||||
|
<goals>
|
||||||
|
<goal>test-compile</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
<configuration>
|
||||||
|
<compilerPlugins>
|
||||||
|
<plugin>kotlinx-serialization</plugin>
|
||||||
|
</compilerPlugins>
|
||||||
|
</configuration>
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.jetbrains.kotlin</groupId>
|
||||||
|
<artifactId>kotlin-maven-serialization</artifactId>
|
||||||
|
<version>${kotlin.version}</version>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
|
||||||
|
</project>
|
||||||
40
event-logger/src/main/kotlin/events.kt
Normal file
40
event-logger/src/main/kotlin/events.kt
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2022 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.signal.event
|
||||||
|
|
||||||
|
import java.util.Collections
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
import kotlinx.serialization.json.Json
|
||||||
|
import kotlinx.serialization.modules.SerializersModule
|
||||||
|
import kotlinx.serialization.modules.polymorphic
|
||||||
|
import kotlinx.serialization.modules.subclass
|
||||||
|
|
||||||
|
val module = SerializersModule {
|
||||||
|
polymorphic(Event::class) {
|
||||||
|
subclass(RemoteConfigSetEvent::class)
|
||||||
|
subclass(RemoteConfigDeleteEvent::class)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
val jsonFormat = Json { serializersModule = module }
|
||||||
|
|
||||||
|
sealed interface Event
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class RemoteConfigSetEvent(
|
||||||
|
val token: String,
|
||||||
|
val name: String,
|
||||||
|
val percentage: Int,
|
||||||
|
val defaultValue: String? = null,
|
||||||
|
val value: String? = null,
|
||||||
|
val hashKey: String? = null,
|
||||||
|
val uuids: Collection<String> = Collections.emptyList(),
|
||||||
|
) : Event
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
data class RemoteConfigDeleteEvent(
|
||||||
|
val token: String,
|
||||||
|
val name: String,
|
||||||
|
) : Event
|
||||||
41
event-logger/src/main/kotlin/loggers.kt
Normal file
41
event-logger/src/main/kotlin/loggers.kt
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2022 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.signal.event
|
||||||
|
|
||||||
|
import com.google.cloud.logging.LogEntry
|
||||||
|
import com.google.cloud.logging.Logging
|
||||||
|
import com.google.cloud.logging.MonitoredResourceUtil
|
||||||
|
import com.google.cloud.logging.Payload.JsonPayload
|
||||||
|
import com.google.cloud.logging.Severity
|
||||||
|
import com.google.protobuf.Struct
|
||||||
|
import com.google.protobuf.util.JsonFormat
|
||||||
|
import kotlinx.serialization.encodeToString
|
||||||
|
|
||||||
|
interface AdminEventLogger {
|
||||||
|
fun logEvent(event: Event, labels: Map<String, String>?)
|
||||||
|
fun logEvent(event: Event) = logEvent(event, null)
|
||||||
|
}
|
||||||
|
|
||||||
|
class NoOpAdminEventLogger : AdminEventLogger {
|
||||||
|
override fun logEvent(event: Event, labels: Map<String, String>?) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
class GoogleCloudAdminEventLogger(private val logging: Logging, private val projectId: String, private val logName: String) : AdminEventLogger {
|
||||||
|
override fun logEvent(event: Event, labels: Map<String, String>?) {
|
||||||
|
val structBuilder = Struct.newBuilder()
|
||||||
|
JsonFormat.parser().merge(jsonFormat.encodeToString(event), structBuilder)
|
||||||
|
val struct = structBuilder.build()
|
||||||
|
|
||||||
|
val logEntryBuilder = LogEntry.newBuilder(JsonPayload.of(struct))
|
||||||
|
.setLogName(logName)
|
||||||
|
.setSeverity(Severity.NOTICE)
|
||||||
|
.setResource(MonitoredResourceUtil.getResource(projectId, "project"));
|
||||||
|
if (labels != null) {
|
||||||
|
logEntryBuilder.setLabels(labels);
|
||||||
|
}
|
||||||
|
logging.write(listOf(logEntryBuilder.build()))
|
||||||
|
}
|
||||||
|
}
|
||||||
316
mvnw
vendored
Executable file
316
mvnw
vendored
Executable file
@@ -0,0 +1,316 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# ----------------------------------------------------------------------------
|
||||||
|
# Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
# or more contributor license agreements. See the NOTICE file
|
||||||
|
# distributed with this work for additional information
|
||||||
|
# regarding copyright ownership. The ASF licenses this file
|
||||||
|
# to you under the Apache License, Version 2.0 (the
|
||||||
|
# "License"); you may not use this file except in compliance
|
||||||
|
# with the License. You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing,
|
||||||
|
# software distributed under the License is distributed on an
|
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||||
|
# KIND, either express or implied. See the License for the
|
||||||
|
# specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
# ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
# ----------------------------------------------------------------------------
|
||||||
|
# Maven Start Up Batch script
|
||||||
|
#
|
||||||
|
# Required ENV vars:
|
||||||
|
# ------------------
|
||||||
|
# JAVA_HOME - location of a JDK home dir
|
||||||
|
#
|
||||||
|
# Optional ENV vars
|
||||||
|
# -----------------
|
||||||
|
# M2_HOME - location of maven2's installed home dir
|
||||||
|
# MAVEN_OPTS - parameters passed to the Java VM when running Maven
|
||||||
|
# e.g. to debug Maven itself, use
|
||||||
|
# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
|
||||||
|
# MAVEN_SKIP_RC - flag to disable loading of mavenrc files
|
||||||
|
# ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
if [ -z "$MAVEN_SKIP_RC" ] ; then
|
||||||
|
|
||||||
|
if [ -f /usr/local/etc/mavenrc ] ; then
|
||||||
|
. /usr/local/etc/mavenrc
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -f /etc/mavenrc ] ; then
|
||||||
|
. /etc/mavenrc
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -f "$HOME/.mavenrc" ] ; then
|
||||||
|
. "$HOME/.mavenrc"
|
||||||
|
fi
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
# OS specific support. $var _must_ be set to either true or false.
|
||||||
|
cygwin=false;
|
||||||
|
darwin=false;
|
||||||
|
mingw=false
|
||||||
|
case "`uname`" in
|
||||||
|
CYGWIN*) cygwin=true ;;
|
||||||
|
MINGW*) mingw=true;;
|
||||||
|
Darwin*) darwin=true
|
||||||
|
# Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home
|
||||||
|
# See https://developer.apple.com/library/mac/qa/qa1170/_index.html
|
||||||
|
if [ -z "$JAVA_HOME" ]; then
|
||||||
|
if [ -x "/usr/libexec/java_home" ]; then
|
||||||
|
export JAVA_HOME="`/usr/libexec/java_home`"
|
||||||
|
else
|
||||||
|
export JAVA_HOME="/Library/Java/Home"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
if [ -z "$JAVA_HOME" ] ; then
|
||||||
|
if [ -r /etc/gentoo-release ] ; then
|
||||||
|
JAVA_HOME=`java-config --jre-home`
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -z "$M2_HOME" ] ; then
|
||||||
|
## resolve links - $0 may be a link to maven's home
|
||||||
|
PRG="$0"
|
||||||
|
|
||||||
|
# need this for relative symlinks
|
||||||
|
while [ -h "$PRG" ] ; do
|
||||||
|
ls=`ls -ld "$PRG"`
|
||||||
|
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||||
|
if expr "$link" : '/.*' > /dev/null; then
|
||||||
|
PRG="$link"
|
||||||
|
else
|
||||||
|
PRG="`dirname "$PRG"`/$link"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
saveddir=`pwd`
|
||||||
|
|
||||||
|
M2_HOME=`dirname "$PRG"`/..
|
||||||
|
|
||||||
|
# make it fully qualified
|
||||||
|
M2_HOME=`cd "$M2_HOME" && pwd`
|
||||||
|
|
||||||
|
cd "$saveddir"
|
||||||
|
# echo Using m2 at $M2_HOME
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Cygwin, ensure paths are in UNIX format before anything is touched
|
||||||
|
if $cygwin ; then
|
||||||
|
[ -n "$M2_HOME" ] &&
|
||||||
|
M2_HOME=`cygpath --unix "$M2_HOME"`
|
||||||
|
[ -n "$JAVA_HOME" ] &&
|
||||||
|
JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
|
||||||
|
[ -n "$CLASSPATH" ] &&
|
||||||
|
CLASSPATH=`cygpath --path --unix "$CLASSPATH"`
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Mingw, ensure paths are in UNIX format before anything is touched
|
||||||
|
if $mingw ; then
|
||||||
|
[ -n "$M2_HOME" ] &&
|
||||||
|
M2_HOME="`(cd "$M2_HOME"; pwd)`"
|
||||||
|
[ -n "$JAVA_HOME" ] &&
|
||||||
|
JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -z "$JAVA_HOME" ]; then
|
||||||
|
javaExecutable="`which javac`"
|
||||||
|
if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then
|
||||||
|
# readlink(1) is not available as standard on Solaris 10.
|
||||||
|
readLink=`which readlink`
|
||||||
|
if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then
|
||||||
|
if $darwin ; then
|
||||||
|
javaHome="`dirname \"$javaExecutable\"`"
|
||||||
|
javaExecutable="`cd \"$javaHome\" && pwd -P`/javac"
|
||||||
|
else
|
||||||
|
javaExecutable="`readlink -f \"$javaExecutable\"`"
|
||||||
|
fi
|
||||||
|
javaHome="`dirname \"$javaExecutable\"`"
|
||||||
|
javaHome=`expr "$javaHome" : '\(.*\)/bin'`
|
||||||
|
JAVA_HOME="$javaHome"
|
||||||
|
export JAVA_HOME
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -z "$JAVACMD" ] ; then
|
||||||
|
if [ -n "$JAVA_HOME" ] ; then
|
||||||
|
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||||
|
# IBM's JDK on AIX uses strange locations for the executables
|
||||||
|
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||||
|
else
|
||||||
|
JAVACMD="$JAVA_HOME/bin/java"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
JAVACMD="`\\unset -f command; \\command -v java`"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -x "$JAVACMD" ] ; then
|
||||||
|
echo "Error: JAVA_HOME is not defined correctly." >&2
|
||||||
|
echo " We cannot execute $JAVACMD" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -z "$JAVA_HOME" ] ; then
|
||||||
|
echo "Warning: JAVA_HOME environment variable is not set."
|
||||||
|
fi
|
||||||
|
|
||||||
|
CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher
|
||||||
|
|
||||||
|
# traverses directory structure from process work directory to filesystem root
|
||||||
|
# first directory with .mvn subdirectory is considered project base directory
|
||||||
|
find_maven_basedir() {
|
||||||
|
|
||||||
|
if [ -z "$1" ]
|
||||||
|
then
|
||||||
|
echo "Path not specified to find_maven_basedir"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
basedir="$1"
|
||||||
|
wdir="$1"
|
||||||
|
while [ "$wdir" != '/' ] ; do
|
||||||
|
if [ -d "$wdir"/.mvn ] ; then
|
||||||
|
basedir=$wdir
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
# workaround for JBEAP-8937 (on Solaris 10/Sparc)
|
||||||
|
if [ -d "${wdir}" ]; then
|
||||||
|
wdir=`cd "$wdir/.."; pwd`
|
||||||
|
fi
|
||||||
|
# end of workaround
|
||||||
|
done
|
||||||
|
echo "${basedir}"
|
||||||
|
}
|
||||||
|
|
||||||
|
# concatenates all lines of a file
|
||||||
|
concat_lines() {
|
||||||
|
if [ -f "$1" ]; then
|
||||||
|
echo "$(tr -s '\n' ' ' < "$1")"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
BASE_DIR=`find_maven_basedir "$(pwd)"`
|
||||||
|
if [ -z "$BASE_DIR" ]; then
|
||||||
|
exit 1;
|
||||||
|
fi
|
||||||
|
|
||||||
|
##########################################################################################
|
||||||
|
# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
|
||||||
|
# This allows using the maven wrapper in projects that prohibit checking in binary data.
|
||||||
|
##########################################################################################
|
||||||
|
if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo "Found .mvn/wrapper/maven-wrapper.jar"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..."
|
||||||
|
fi
|
||||||
|
if [ -n "$MVNW_REPOURL" ]; then
|
||||||
|
jarUrl="$MVNW_REPOURL/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar"
|
||||||
|
else
|
||||||
|
jarUrl="https://repo.maven.apache.org/maven2/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar"
|
||||||
|
fi
|
||||||
|
while IFS="=" read key value; do
|
||||||
|
case "$key" in (wrapperUrl) jarUrl="$value"; break ;;
|
||||||
|
esac
|
||||||
|
done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties"
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo "Downloading from: $jarUrl"
|
||||||
|
fi
|
||||||
|
wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar"
|
||||||
|
if $cygwin; then
|
||||||
|
wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"`
|
||||||
|
fi
|
||||||
|
|
||||||
|
if command -v wget > /dev/null; then
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo "Found wget ... using wget"
|
||||||
|
fi
|
||||||
|
if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then
|
||||||
|
wget "$jarUrl" -O "$wrapperJarPath" || rm -f "$wrapperJarPath"
|
||||||
|
else
|
||||||
|
wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath" || rm -f "$wrapperJarPath"
|
||||||
|
fi
|
||||||
|
elif command -v curl > /dev/null; then
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo "Found curl ... using curl"
|
||||||
|
fi
|
||||||
|
if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then
|
||||||
|
curl -o "$wrapperJarPath" "$jarUrl" -f
|
||||||
|
else
|
||||||
|
curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f
|
||||||
|
fi
|
||||||
|
|
||||||
|
else
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo "Falling back to using Java to download"
|
||||||
|
fi
|
||||||
|
javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java"
|
||||||
|
# For Cygwin, switch paths to Windows format before running javac
|
||||||
|
if $cygwin; then
|
||||||
|
javaClass=`cygpath --path --windows "$javaClass"`
|
||||||
|
fi
|
||||||
|
if [ -e "$javaClass" ]; then
|
||||||
|
if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo " - Compiling MavenWrapperDownloader.java ..."
|
||||||
|
fi
|
||||||
|
# Compiling the Java class
|
||||||
|
("$JAVA_HOME/bin/javac" "$javaClass")
|
||||||
|
fi
|
||||||
|
if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
|
||||||
|
# Running the downloader
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo " - Running MavenWrapperDownloader.java ..."
|
||||||
|
fi
|
||||||
|
("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR")
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
##########################################################################################
|
||||||
|
# End of extension
|
||||||
|
##########################################################################################
|
||||||
|
|
||||||
|
export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"}
|
||||||
|
if [ "$MVNW_VERBOSE" = true ]; then
|
||||||
|
echo $MAVEN_PROJECTBASEDIR
|
||||||
|
fi
|
||||||
|
MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS"
|
||||||
|
|
||||||
|
# For Cygwin, switch paths to Windows format before running java
|
||||||
|
if $cygwin; then
|
||||||
|
[ -n "$M2_HOME" ] &&
|
||||||
|
M2_HOME=`cygpath --path --windows "$M2_HOME"`
|
||||||
|
[ -n "$JAVA_HOME" ] &&
|
||||||
|
JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"`
|
||||||
|
[ -n "$CLASSPATH" ] &&
|
||||||
|
CLASSPATH=`cygpath --path --windows "$CLASSPATH"`
|
||||||
|
[ -n "$MAVEN_PROJECTBASEDIR" ] &&
|
||||||
|
MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"`
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Provide a "standardized" way to retrieve the CLI args that will
|
||||||
|
# work with both Windows and non-Windows executions.
|
||||||
|
MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@"
|
||||||
|
export MAVEN_CMD_LINE_ARGS
|
||||||
|
|
||||||
|
WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
|
||||||
|
|
||||||
|
exec "$JAVACMD" \
|
||||||
|
$MAVEN_OPTS \
|
||||||
|
$MAVEN_DEBUG_OPTS \
|
||||||
|
-classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \
|
||||||
|
"-Dmaven.home=${M2_HOME}" \
|
||||||
|
"-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \
|
||||||
|
${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@"
|
||||||
188
mvnw.cmd
vendored
Normal file
188
mvnw.cmd
vendored
Normal file
@@ -0,0 +1,188 @@
|
|||||||
|
@REM ----------------------------------------------------------------------------
|
||||||
|
@REM Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
@REM or more contributor license agreements. See the NOTICE file
|
||||||
|
@REM distributed with this work for additional information
|
||||||
|
@REM regarding copyright ownership. The ASF licenses this file
|
||||||
|
@REM to you under the Apache License, Version 2.0 (the
|
||||||
|
@REM "License"); you may not use this file except in compliance
|
||||||
|
@REM with the License. You may obtain a copy of the License at
|
||||||
|
@REM
|
||||||
|
@REM http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
@REM
|
||||||
|
@REM Unless required by applicable law or agreed to in writing,
|
||||||
|
@REM software distributed under the License is distributed on an
|
||||||
|
@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||||
|
@REM KIND, either express or implied. See the License for the
|
||||||
|
@REM specific language governing permissions and limitations
|
||||||
|
@REM under the License.
|
||||||
|
@REM ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
@REM ----------------------------------------------------------------------------
|
||||||
|
@REM Maven Start Up Batch script
|
||||||
|
@REM
|
||||||
|
@REM Required ENV vars:
|
||||||
|
@REM JAVA_HOME - location of a JDK home dir
|
||||||
|
@REM
|
||||||
|
@REM Optional ENV vars
|
||||||
|
@REM M2_HOME - location of maven2's installed home dir
|
||||||
|
@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands
|
||||||
|
@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending
|
||||||
|
@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven
|
||||||
|
@REM e.g. to debug Maven itself, use
|
||||||
|
@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
|
||||||
|
@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files
|
||||||
|
@REM ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on'
|
||||||
|
@echo off
|
||||||
|
@REM set title of command window
|
||||||
|
title %0
|
||||||
|
@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on'
|
||||||
|
@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO%
|
||||||
|
|
||||||
|
@REM set %HOME% to equivalent of $HOME
|
||||||
|
if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%")
|
||||||
|
|
||||||
|
@REM Execute a user defined script before this one
|
||||||
|
if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre
|
||||||
|
@REM check for pre script, once with legacy .bat ending and once with .cmd ending
|
||||||
|
if exist "%USERPROFILE%\mavenrc_pre.bat" call "%USERPROFILE%\mavenrc_pre.bat" %*
|
||||||
|
if exist "%USERPROFILE%\mavenrc_pre.cmd" call "%USERPROFILE%\mavenrc_pre.cmd" %*
|
||||||
|
:skipRcPre
|
||||||
|
|
||||||
|
@setlocal
|
||||||
|
|
||||||
|
set ERROR_CODE=0
|
||||||
|
|
||||||
|
@REM To isolate internal variables from possible post scripts, we use another setlocal
|
||||||
|
@setlocal
|
||||||
|
|
||||||
|
@REM ==== START VALIDATION ====
|
||||||
|
if not "%JAVA_HOME%" == "" goto OkJHome
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo Error: JAVA_HOME not found in your environment. >&2
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the >&2
|
||||||
|
echo location of your Java installation. >&2
|
||||||
|
echo.
|
||||||
|
goto error
|
||||||
|
|
||||||
|
:OkJHome
|
||||||
|
if exist "%JAVA_HOME%\bin\java.exe" goto init
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo Error: JAVA_HOME is set to an invalid directory. >&2
|
||||||
|
echo JAVA_HOME = "%JAVA_HOME%" >&2
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the >&2
|
||||||
|
echo location of your Java installation. >&2
|
||||||
|
echo.
|
||||||
|
goto error
|
||||||
|
|
||||||
|
@REM ==== END VALIDATION ====
|
||||||
|
|
||||||
|
:init
|
||||||
|
|
||||||
|
@REM Find the project base dir, i.e. the directory that contains the folder ".mvn".
|
||||||
|
@REM Fallback to current working directory if not found.
|
||||||
|
|
||||||
|
set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR%
|
||||||
|
IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir
|
||||||
|
|
||||||
|
set EXEC_DIR=%CD%
|
||||||
|
set WDIR=%EXEC_DIR%
|
||||||
|
:findBaseDir
|
||||||
|
IF EXIST "%WDIR%"\.mvn goto baseDirFound
|
||||||
|
cd ..
|
||||||
|
IF "%WDIR%"=="%CD%" goto baseDirNotFound
|
||||||
|
set WDIR=%CD%
|
||||||
|
goto findBaseDir
|
||||||
|
|
||||||
|
:baseDirFound
|
||||||
|
set MAVEN_PROJECTBASEDIR=%WDIR%
|
||||||
|
cd "%EXEC_DIR%"
|
||||||
|
goto endDetectBaseDir
|
||||||
|
|
||||||
|
:baseDirNotFound
|
||||||
|
set MAVEN_PROJECTBASEDIR=%EXEC_DIR%
|
||||||
|
cd "%EXEC_DIR%"
|
||||||
|
|
||||||
|
:endDetectBaseDir
|
||||||
|
|
||||||
|
IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig
|
||||||
|
|
||||||
|
@setlocal EnableExtensions EnableDelayedExpansion
|
||||||
|
for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a
|
||||||
|
@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS%
|
||||||
|
|
||||||
|
:endReadAdditionalConfig
|
||||||
|
|
||||||
|
SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe"
|
||||||
|
set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar"
|
||||||
|
set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
|
||||||
|
|
||||||
|
set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar"
|
||||||
|
|
||||||
|
FOR /F "usebackq tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO (
|
||||||
|
IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B
|
||||||
|
)
|
||||||
|
|
||||||
|
@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
|
||||||
|
@REM This allows using the maven wrapper in projects that prohibit checking in binary data.
|
||||||
|
if exist %WRAPPER_JAR% (
|
||||||
|
if "%MVNW_VERBOSE%" == "true" (
|
||||||
|
echo Found %WRAPPER_JAR%
|
||||||
|
)
|
||||||
|
) else (
|
||||||
|
if not "%MVNW_REPOURL%" == "" (
|
||||||
|
SET DOWNLOAD_URL="%MVNW_REPOURL%/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar"
|
||||||
|
)
|
||||||
|
if "%MVNW_VERBOSE%" == "true" (
|
||||||
|
echo Couldn't find %WRAPPER_JAR%, downloading it ...
|
||||||
|
echo Downloading from: %DOWNLOAD_URL%
|
||||||
|
)
|
||||||
|
|
||||||
|
powershell -Command "&{"^
|
||||||
|
"$webclient = new-object System.Net.WebClient;"^
|
||||||
|
"if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^
|
||||||
|
"$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^
|
||||||
|
"}"^
|
||||||
|
"[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^
|
||||||
|
"}"
|
||||||
|
if "%MVNW_VERBOSE%" == "true" (
|
||||||
|
echo Finished downloading %WRAPPER_JAR%
|
||||||
|
)
|
||||||
|
)
|
||||||
|
@REM End of extension
|
||||||
|
|
||||||
|
@REM Provide a "standardized" way to retrieve the CLI args that will
|
||||||
|
@REM work with both Windows and non-Windows executions.
|
||||||
|
set MAVEN_CMD_LINE_ARGS=%*
|
||||||
|
|
||||||
|
%MAVEN_JAVA_EXE% ^
|
||||||
|
%JVM_CONFIG_MAVEN_PROPS% ^
|
||||||
|
%MAVEN_OPTS% ^
|
||||||
|
%MAVEN_DEBUG_OPTS% ^
|
||||||
|
-classpath %WRAPPER_JAR% ^
|
||||||
|
"-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" ^
|
||||||
|
%WRAPPER_LAUNCHER% %MAVEN_CONFIG% %*
|
||||||
|
if ERRORLEVEL 1 goto error
|
||||||
|
goto end
|
||||||
|
|
||||||
|
:error
|
||||||
|
set ERROR_CODE=1
|
||||||
|
|
||||||
|
:end
|
||||||
|
@endlocal & set ERROR_CODE=%ERROR_CODE%
|
||||||
|
|
||||||
|
if not "%MAVEN_SKIP_RC%"=="" goto skipRcPost
|
||||||
|
@REM check for post script, once with legacy .bat ending and once with .cmd ending
|
||||||
|
if exist "%USERPROFILE%\mavenrc_post.bat" call "%USERPROFILE%\mavenrc_post.bat"
|
||||||
|
if exist "%USERPROFILE%\mavenrc_post.cmd" call "%USERPROFILE%\mavenrc_post.cmd"
|
||||||
|
:skipRcPost
|
||||||
|
|
||||||
|
@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on'
|
||||||
|
if "%MAVEN_BATCH_PAUSE%"=="on" pause
|
||||||
|
|
||||||
|
if "%MAVEN_TERMINATE_CMD%"=="on" exit %ERROR_CODE%
|
||||||
|
|
||||||
|
cmd /C exit /B %ERROR_CODE%
|
||||||
733
pom.xml
733
pom.xml
@@ -1,252 +1,509 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
<modelVersion>4.0.0</modelVersion>
|
<modelVersion>4.0.0</modelVersion>
|
||||||
<prerequisites>
|
<packaging>pom</packaging>
|
||||||
<maven>3.0.0</maven>
|
|
||||||
</prerequisites>
|
|
||||||
|
|
||||||
<groupId>org.whispersystems.textsecure</groupId>
|
<repositories>
|
||||||
<artifactId>TextSecureServer</artifactId>
|
<repository>
|
||||||
<version>0.40</version>
|
<id>central</id>
|
||||||
|
<name>Central Repository</name>
|
||||||
|
<url>https://repo.maven.apache.org/maven2</url>
|
||||||
|
<snapshots>
|
||||||
|
<enabled>false</enabled>
|
||||||
|
</snapshots>
|
||||||
|
</repository>
|
||||||
|
<repository>
|
||||||
|
<id>dynamodb-local-oregon</id>
|
||||||
|
<name>DynamoDB Local Release Repository</name>
|
||||||
|
<url>https://s3-us-west-2.amazonaws.com/dynamodb-local/release</url>
|
||||||
|
</repository>
|
||||||
|
</repositories>
|
||||||
|
|
||||||
<properties>
|
<pluginRepositories>
|
||||||
<dropwizard.version>0.7.1</dropwizard.version>
|
<pluginRepository>
|
||||||
<jackson.api.version>2.3.3</jackson.api.version>
|
<id>ossrh-snapshots</id>
|
||||||
<commons-codec.version>1.6</commons-codec.version>
|
<url>https://oss.sonatype.org/content/repositories/snapshots</url>
|
||||||
</properties>
|
<releases>
|
||||||
|
<enabled>false</enabled>
|
||||||
|
</releases>
|
||||||
|
<snapshots>
|
||||||
|
<enabled>true</enabled>
|
||||||
|
</snapshots>
|
||||||
|
</pluginRepository>
|
||||||
|
</pluginRepositories>
|
||||||
|
|
||||||
|
<modules>
|
||||||
|
<module>event-logger</module>
|
||||||
|
<module>redis-dispatch</module>
|
||||||
|
<module>websocket-resources</module>
|
||||||
|
<module>service</module>
|
||||||
|
</modules>
|
||||||
|
|
||||||
|
<properties>
|
||||||
|
<aws.sdk.version>1.12.287</aws.sdk.version>
|
||||||
|
<aws.sdk2.version>2.17.258</aws.sdk2.version>
|
||||||
|
<commons-codec.version>1.15</commons-codec.version>
|
||||||
|
<commons-csv.version>1.8</commons-csv.version>
|
||||||
|
<commons-io.version>2.9.0</commons-io.version>
|
||||||
|
<dropwizard.version>2.0.32</dropwizard.version>
|
||||||
|
<dropwizard-metrics-datadog.version>1.1.13</dropwizard-metrics-datadog.version>
|
||||||
|
<grpc.version>1.49.0</grpc.version>
|
||||||
|
<gson.version>2.9.0</gson.version>
|
||||||
|
<guava.version>30.1.1-jre</guava.version>
|
||||||
|
<jackson.version>2.13.3</jackson.version>
|
||||||
|
<jaxb.version>2.3.1</jaxb.version>
|
||||||
|
<jedis.version>2.9.0</jedis.version>
|
||||||
|
<kotlin.version>1.7.10</kotlin.version>
|
||||||
|
<kotlinx-serialization.version>1.4.0</kotlinx-serialization.version>
|
||||||
|
<lettuce.version>6.1.9.RELEASE</lettuce.version>
|
||||||
|
<libphonenumber.version>8.12.54</libphonenumber.version>
|
||||||
|
<logstash.logback.version>7.0.1</logstash.logback.version>
|
||||||
|
<micrometer.version>1.9.3</micrometer.version>
|
||||||
|
<mockito.version>4.7.0</mockito.version>
|
||||||
|
<netty.version>4.1.82.Final</netty.version>
|
||||||
|
<opentest4j.version>1.2.0</opentest4j.version>
|
||||||
|
<protobuf.version>3.21.7</protobuf.version>
|
||||||
|
<pushy.version>0.15.1</pushy.version>
|
||||||
|
<resilience4j.version>1.7.0</resilience4j.version>
|
||||||
|
<semver4j.version>3.1.0</semver4j.version>
|
||||||
|
<slf4j.version>1.7.30</slf4j.version>
|
||||||
|
<stripe.version>21.2.0</stripe.version>
|
||||||
|
<vavr.version>0.10.4</vavr.version>
|
||||||
|
|
||||||
|
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||||
|
</properties>
|
||||||
|
|
||||||
|
<groupId>org.whispersystems.textsecure</groupId>
|
||||||
|
<artifactId>TextSecureServer</artifactId>
|
||||||
|
<version>JGITVER</version>
|
||||||
|
|
||||||
|
<dependencyManagement>
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>io.dropwizard</groupId>
|
<groupId>com.fasterxml.jackson</groupId>
|
||||||
<artifactId>dropwizard-core</artifactId>
|
<artifactId>jackson-bom</artifactId>
|
||||||
<version>${dropwizard.version}</version>
|
<version>2.13.3</version>
|
||||||
</dependency>
|
<type>pom</type>
|
||||||
<dependency>
|
<scope>import</scope>
|
||||||
<groupId>io.dropwizard</groupId>
|
</dependency>
|
||||||
<artifactId>dropwizard-jdbi</artifactId>
|
<dependency>
|
||||||
<version>${dropwizard.version}</version>
|
<groupId>io.dropwizard</groupId>
|
||||||
</dependency>
|
<artifactId>dropwizard-dependencies</artifactId>
|
||||||
<dependency>
|
<version>${dropwizard.version}</version>
|
||||||
<groupId>io.dropwizard</groupId>
|
<type>pom</type>
|
||||||
<artifactId>dropwizard-auth</artifactId>
|
<scope>import</scope>
|
||||||
<version>${dropwizard.version}</version>
|
</dependency>
|
||||||
</dependency>
|
<dependency>
|
||||||
<dependency>
|
<groupId>io.grpc</groupId>
|
||||||
<groupId>io.dropwizard</groupId>
|
<artifactId>grpc-netty</artifactId>
|
||||||
<artifactId>dropwizard-client</artifactId>
|
<version>${grpc.version}</version>
|
||||||
<version>${dropwizard.version}</version>
|
<scope>runtime</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>io.dropwizard</groupId>
|
<groupId>io.grpc</groupId>
|
||||||
<artifactId>dropwizard-migrations</artifactId>
|
<artifactId>grpc-protobuf</artifactId>
|
||||||
<version>${dropwizard.version}</version>
|
<version>${grpc.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>io.dropwizard</groupId>
|
<groupId>io.grpc</groupId>
|
||||||
<artifactId>dropwizard-testing</artifactId>
|
<artifactId>grpc-stub</artifactId>
|
||||||
<version>${dropwizard.version}</version>
|
<version>${grpc.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<!-- Needed for gRPC with Java 9+ -->
|
||||||
<groupId>io.dropwizard</groupId>
|
<dependency>
|
||||||
<artifactId>dropwizard-metrics-graphite</artifactId>
|
<groupId>org.apache.tomcat</groupId>
|
||||||
<version>${dropwizard.version}</version>
|
<artifactId>annotations-api</artifactId>
|
||||||
</dependency>
|
<version>6.0.53</version>
|
||||||
<dependency>
|
<scope>provided</scope>
|
||||||
<groupId>com.dcsquare</groupId>
|
</dependency>
|
||||||
<artifactId>dropwizard-papertrail</artifactId>
|
<dependency>
|
||||||
<version>1.1</version>
|
<groupId>io.netty</groupId>
|
||||||
</dependency>
|
<artifactId>netty-bom</artifactId>
|
||||||
|
<version>${netty.version}</version>
|
||||||
|
<type>pom</type>
|
||||||
<dependency>
|
<scope>import</scope>
|
||||||
<groupId>com.sun.jersey</groupId>
|
</dependency>
|
||||||
<artifactId>jersey-json</artifactId>
|
<dependency>
|
||||||
<version>1.18.1</version>
|
<groupId>com.amazonaws</groupId>
|
||||||
</dependency>
|
<artifactId>aws-java-sdk-bom</artifactId>
|
||||||
<dependency>
|
<version>${aws.sdk.version}</version>
|
||||||
<groupId>com.codahale.metrics</groupId>
|
<type>pom</type>
|
||||||
<artifactId>metrics-graphite</artifactId>
|
<scope>import</scope>
|
||||||
<version>3.0.2</version>
|
</dependency>
|
||||||
</dependency>
|
<dependency>
|
||||||
<dependency>
|
<groupId>software.amazon.awssdk</groupId>
|
||||||
<groupId>org.eclipse.jetty.websocket</groupId>
|
<artifactId>bom</artifactId>
|
||||||
<artifactId>websocket-server</artifactId>
|
<version>${aws.sdk2.version}</version>
|
||||||
<version>9.0.7.v20131107</version>
|
<type>pom</type>
|
||||||
</dependency>
|
<scope>import</scope>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>bouncycastle</groupId>
|
<groupId>com.google.cloud</groupId>
|
||||||
<artifactId>bcprov-jdk16</artifactId>
|
<artifactId>libraries-bom</artifactId>
|
||||||
<version>140</version>
|
<version>26.1.0</version>
|
||||||
</dependency>
|
<type>pom</type>
|
||||||
<dependency>
|
<scope>import</scope>
|
||||||
<groupId>com.google.android.gcm</groupId>
|
</dependency>
|
||||||
<artifactId>gcm-server</artifactId>
|
<dependency>
|
||||||
<version>1.0.2</version>
|
<groupId>io.github.resilience4j</groupId>
|
||||||
</dependency>
|
<artifactId>resilience4j-bom</artifactId>
|
||||||
<dependency>
|
<version>${resilience4j.version}</version>
|
||||||
<groupId>com.notnoop.apns</groupId>
|
<type>pom</type>
|
||||||
<artifactId>apns</artifactId>
|
<scope>import</scope>
|
||||||
<version>0.2.3</version>
|
</dependency>
|
||||||
</dependency>
|
<dependency>
|
||||||
|
<groupId>io.micrometer</groupId>
|
||||||
<dependency>
|
<artifactId>micrometer-bom</artifactId>
|
||||||
<groupId>com.amazonaws</groupId>
|
<version>${micrometer.version}</version>
|
||||||
<artifactId>aws-java-sdk</artifactId>
|
<type>pom</type>
|
||||||
<version>1.4.1</version>
|
<scope>import</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.google.protobuf</groupId>
|
<groupId>com.eatthepath</groupId>
|
||||||
<artifactId>protobuf-java</artifactId>
|
<artifactId>pushy</artifactId>
|
||||||
<version>2.5.0</version>
|
<version>${pushy.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
<dependency>
|
<groupId>com.eatthepath</groupId>
|
||||||
<groupId>redis.clients</groupId>
|
<artifactId>pushy-dropwizard-metrics-listener</artifactId>
|
||||||
<artifactId>jedis</artifactId>
|
<version>${pushy.version}</version>
|
||||||
<version>2.6.1</version>
|
</dependency>
|
||||||
<type>jar</type>
|
<dependency>
|
||||||
<scope>compile</scope>
|
<groupId>com.google.guava</groupId>
|
||||||
</dependency>
|
<artifactId>guava</artifactId>
|
||||||
<dependency>
|
<version>${guava.version}</version>
|
||||||
<groupId>com.twilio.sdk</groupId>
|
</dependency>
|
||||||
<artifactId>twilio-java-sdk</artifactId>
|
<dependency>
|
||||||
<version>3.4.5</version>
|
<groupId>com.google.protobuf</groupId>
|
||||||
</dependency>
|
<artifactId>protobuf-java</artifactId>
|
||||||
|
<version>${protobuf.version}</version>
|
||||||
<dependency>
|
</dependency>
|
||||||
<groupId>postgresql</groupId>
|
<dependency>
|
||||||
<artifactId>postgresql</artifactId>
|
<groupId>com.googlecode.libphonenumber</groupId>
|
||||||
<version>9.1-901.jdbc4</version>
|
<artifactId>libphonenumber</artifactId>
|
||||||
</dependency>
|
<version>${libphonenumber.version}</version>
|
||||||
<dependency>
|
</dependency>
|
||||||
<groupId>org.igniterealtime.smack</groupId>
|
<dependency>
|
||||||
<artifactId>smack-tcp</artifactId>
|
<groupId>com.vdurmont</groupId>
|
||||||
<version>4.0.0</version>
|
<artifactId>semver4j</artifactId>
|
||||||
</dependency>
|
<version>${semver4j.version}</version>
|
||||||
<dependency>
|
</dependency>
|
||||||
<groupId>org.whispersystems</groupId>
|
<dependency>
|
||||||
<artifactId>websocket-resources</artifactId>
|
<groupId>commons-codec</groupId>
|
||||||
<version>0.2.3</version>
|
<artifactId>commons-codec</artifactId>
|
||||||
</dependency>
|
<version>${commons-codec.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>commons-io</groupId>
|
||||||
|
<artifactId>commons-io</artifactId>
|
||||||
|
<version>${commons-io.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.lettuce</groupId>
|
||||||
|
<artifactId>lettuce-core</artifactId>
|
||||||
|
<version>${lettuce.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.vavr</groupId>
|
||||||
|
<artifactId>vavr</artifactId>
|
||||||
|
<version>${vavr.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>javax.xml.bind</groupId>
|
||||||
|
<artifactId>jaxb-api</artifactId>
|
||||||
|
<version>${jaxb.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>net.logstash.logback</groupId>
|
||||||
|
<artifactId>logstash-logback-encoder</artifactId>
|
||||||
|
<version>${logstash.logback.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.commons</groupId>
|
||||||
|
<artifactId>commons-csv</artifactId>
|
||||||
|
<version>${commons-csv.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.coursera</groupId>
|
||||||
|
<artifactId>dropwizard-metrics-datadog</artifactId>
|
||||||
|
<version>${dropwizard-metrics-datadog.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.glassfish.jaxb</groupId>
|
||||||
|
<artifactId>jaxb-runtime</artifactId>
|
||||||
|
<version>${jaxb.version}</version>
|
||||||
|
<scope>runtime</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.mockito</groupId>
|
||||||
|
<artifactId>mockito-core</artifactId>
|
||||||
|
<version>${mockito.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.mockito</groupId>
|
||||||
|
<artifactId>mockito-inline</artifactId>
|
||||||
|
<version>${mockito.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.opentest4j</groupId>
|
||||||
|
<artifactId>opentest4j</artifactId>
|
||||||
|
<version>${opentest4j.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.slf4j</groupId>
|
||||||
|
<artifactId>slf4j-api</artifactId>
|
||||||
|
<version>${slf4j.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.slf4j</groupId>
|
||||||
|
<artifactId>slf4j-nop</artifactId>
|
||||||
|
<version>${slf4j.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>redis.clients</groupId>
|
||||||
|
<artifactId>jedis</artifactId>
|
||||||
|
<version>${jedis.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>commons-logging</groupId>
|
||||||
|
<artifactId>commons-logging</artifactId>
|
||||||
|
<version>1.2</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.ow2.asm</groupId>
|
||||||
|
<artifactId>asm</artifactId>
|
||||||
|
<version>9.2</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.stripe</groupId>
|
||||||
|
<artifactId>stripe-java</artifactId>
|
||||||
|
<version>${stripe.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.code.gson</groupId>
|
||||||
|
<artifactId>gson</artifactId>
|
||||||
|
<version>${gson.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.signal</groupId>
|
||||||
|
<artifactId>embedded-redis</artifactId>
|
||||||
|
<version>0.8.3</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.signal</groupId>
|
||||||
|
<artifactId>libsignal-server</artifactId>
|
||||||
|
<version>0.18.0</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.logging.log4j</groupId>
|
||||||
|
<artifactId>log4j-bom</artifactId>
|
||||||
|
<version>2.17.1</version>
|
||||||
|
<type>pom</type>
|
||||||
|
<scope>import</scope>
|
||||||
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
</dependencyManagement>
|
||||||
|
|
||||||
<dependencyManagement>
|
<dependencies>
|
||||||
<dependencies>
|
<dependency>
|
||||||
<dependency>
|
<groupId>org.hamcrest</groupId>
|
||||||
<groupId>com.fasterxml.jackson.core</groupId>
|
<artifactId>hamcrest-all</artifactId>
|
||||||
<artifactId>jackson-databind</artifactId>
|
<version>1.3</version>
|
||||||
<version>${jackson.api.version}</version>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>commons-codec</groupId>
|
<groupId>com.github.tomakehurst</groupId>
|
||||||
<artifactId>commons-codec</artifactId>
|
<artifactId>wiremock-jre8</artifactId>
|
||||||
<version>${commons-codec.version}</version>
|
<version>2.33.2</version>
|
||||||
</dependency>
|
<scope>test</scope>
|
||||||
</dependencies>
|
<exclusions>
|
||||||
</dependencyManagement>
|
<exclusion>
|
||||||
|
<groupId>org.hamcrest</groupId>
|
||||||
|
<artifactId>hamcrest-core</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>javax.xml.bind</groupId>
|
||||||
|
<artifactId>jaxb-api</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
</exclusions>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.mockito</groupId>
|
||||||
|
<artifactId>mockito-core</artifactId>
|
||||||
|
<version>${mockito.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.assertj</groupId>
|
||||||
|
<artifactId>assertj-core</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.junit.jupiter</groupId>
|
||||||
|
<artifactId>junit-jupiter-api</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
<build>
|
</dependencies>
|
||||||
<plugins>
|
|
||||||
<plugin>
|
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
|
||||||
<artifactId>maven-compiler-plugin</artifactId>
|
|
||||||
<configuration>
|
|
||||||
<source>1.7</source>
|
|
||||||
<target>1.7</target>
|
|
||||||
</configuration>
|
|
||||||
</plugin>
|
|
||||||
<plugin>
|
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
|
||||||
<artifactId>maven-source-plugin</artifactId>
|
|
||||||
<version>2.2.1</version>
|
|
||||||
<executions>
|
|
||||||
<execution>
|
|
||||||
<id>attach-sources</id>
|
|
||||||
<goals>
|
|
||||||
<goal>jar</goal>
|
|
||||||
</goals>
|
|
||||||
</execution>
|
|
||||||
</executions>
|
|
||||||
</plugin>
|
|
||||||
<plugin>
|
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
|
||||||
<artifactId>maven-jar-plugin</artifactId>
|
|
||||||
<version>2.4</version>
|
|
||||||
<configuration>
|
|
||||||
<archive>
|
|
||||||
<manifest>
|
|
||||||
<addDefaultImplementationEntries>true</addDefaultImplementationEntries>
|
|
||||||
</manifest>
|
|
||||||
</archive>
|
|
||||||
</configuration>
|
|
||||||
</plugin>
|
|
||||||
<plugin>
|
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
|
||||||
<artifactId>maven-shade-plugin</artifactId>
|
|
||||||
<version>1.6</version>
|
|
||||||
<configuration>
|
|
||||||
<createDependencyReducedPom>true</createDependencyReducedPom>
|
|
||||||
<filters>
|
|
||||||
<filter>
|
|
||||||
<artifact>*:*</artifact>
|
|
||||||
<excludes>
|
|
||||||
<exclude>META-INF/*.SF</exclude>
|
|
||||||
<exclude>META-INF/*.DSA</exclude>
|
|
||||||
<exclude>META-INF/*.RSA</exclude>
|
|
||||||
</excludes>
|
|
||||||
</filter>
|
|
||||||
</filters>
|
|
||||||
</configuration>
|
|
||||||
<executions>
|
|
||||||
<execution>
|
|
||||||
<phase>package</phase>
|
|
||||||
<goals>
|
|
||||||
<goal>shade</goal>
|
|
||||||
</goals>
|
|
||||||
<configuration>
|
|
||||||
<transformers>
|
|
||||||
<transformer implementation="org.apache.maven.plugins.shade.resource.ServicesResourceTransformer"/>
|
|
||||||
<transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
|
|
||||||
<mainClass>org.whispersystems.textsecuregcm.WhisperServerService</mainClass>
|
|
||||||
</transformer>
|
|
||||||
</transformers>
|
|
||||||
</configuration>
|
|
||||||
</execution>
|
|
||||||
</executions>
|
|
||||||
</plugin>
|
|
||||||
|
|
||||||
<plugin>
|
<profiles>
|
||||||
<artifactId>maven-assembly-plugin</artifactId>
|
<profile>
|
||||||
<version>2.4</version>
|
<id>include-abusive-message-filter</id>
|
||||||
<configuration>
|
<activation>
|
||||||
<descriptors>
|
<file>
|
||||||
<descriptor>assembly.xml</descriptor>
|
<exists>abusive-message-filter/pom.xml</exists>
|
||||||
</descriptors>
|
</file>
|
||||||
</configuration>
|
</activation>
|
||||||
<executions>
|
<modules>
|
||||||
<execution>
|
<module>abusive-message-filter</module>
|
||||||
<id>make-assembly</id> <!-- this is used for inheritance merges -->
|
</modules>
|
||||||
<phase>package</phase> <!-- bind to the packaging phase -->
|
</profile>
|
||||||
<goals>
|
|
||||||
<goal>single</goal>
|
|
||||||
</goals>
|
|
||||||
</execution>
|
|
||||||
</executions>
|
|
||||||
</plugin>
|
|
||||||
|
|
||||||
</plugins>
|
<profile>
|
||||||
</build>
|
<id>exclude-abusive-message-filter</id>
|
||||||
|
<activation>
|
||||||
|
<file>
|
||||||
|
<missing>abusive-message-filter/pom.xml</missing>
|
||||||
|
</file>
|
||||||
|
</activation>
|
||||||
|
</profile>
|
||||||
|
</profiles>
|
||||||
|
|
||||||
<repositories>
|
<build>
|
||||||
<repository>
|
<extensions>
|
||||||
<id>gcm-server-repository</id>
|
<extension>
|
||||||
<url>https://raw.github.com/whispersystems/maven/master/gcm-server/releases/</url>
|
<groupId>kr.motd.maven</groupId>
|
||||||
</repository>
|
<artifactId>os-maven-plugin</artifactId>
|
||||||
</repositories>
|
<version>1.7.0</version>
|
||||||
|
</extension>
|
||||||
|
</extensions>
|
||||||
|
<plugins>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.xolstice.maven.plugins</groupId>
|
||||||
|
<artifactId>protobuf-maven-plugin</artifactId>
|
||||||
|
<version>0.6.1</version>
|
||||||
|
<configuration>
|
||||||
|
<checkStaleness>false</checkStaleness>
|
||||||
|
<protocArtifact>com.google.protobuf:protoc:3.21.1:exe:${os.detected.classifier}</protocArtifact>
|
||||||
|
<pluginId>grpc-java</pluginId>
|
||||||
|
<pluginArtifact>io.grpc:protoc-gen-grpc-java:${grpc.version}:exe:${os.detected.classifier}</pluginArtifact>
|
||||||
|
</configuration>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<goals>
|
||||||
|
<goal>compile</goal>
|
||||||
|
<goal>compile-custom</goal>
|
||||||
|
<goal>test-compile</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-compiler-plugin</artifactId>
|
||||||
|
<version>3.8.1</version>
|
||||||
|
<configuration>
|
||||||
|
<release>17</release>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-jar-plugin</artifactId>
|
||||||
|
<version>3.2.0</version>
|
||||||
|
<configuration>
|
||||||
|
<archive>
|
||||||
|
<manifest>
|
||||||
|
<addDefaultImplementationEntries>true</addDefaultImplementationEntries>
|
||||||
|
</manifest>
|
||||||
|
</archive>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-dependency-plugin</artifactId>
|
||||||
|
<version>3.1.2</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>copy</id>
|
||||||
|
<phase>test-compile</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>copy-dependencies</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<includeScope>test</includeScope>
|
||||||
|
<includeTypes>so,dll,dylib</includeTypes>
|
||||||
|
<outputDirectory>${project.build.directory}/lib</outputDirectory>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-surefire-plugin</artifactId>
|
||||||
|
<version>3.0.0-M5</version>
|
||||||
|
<configuration>
|
||||||
|
<systemProperties>
|
||||||
|
<property>
|
||||||
|
<name>sqlite4java.library.path</name>
|
||||||
|
<value>${project.build.directory}/lib</value>
|
||||||
|
</property>
|
||||||
|
</systemProperties>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-enforcer-plugin</artifactId>
|
||||||
|
<version>3.0.0-M3</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<goals>
|
||||||
|
<goal>enforce</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<rules>
|
||||||
|
<dependencyConvergence/>
|
||||||
|
<requireMavenVersion>
|
||||||
|
<version>3.8.3</version>
|
||||||
|
</requireMavenVersion>
|
||||||
|
</rules>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-install-plugin</artifactId>
|
||||||
|
<version>3.0.0-M1</version>
|
||||||
|
<configuration>
|
||||||
|
<skip>true</skip>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-deploy-plugin</artifactId>
|
||||||
|
<version>3.0.0-M1</version>
|
||||||
|
<configuration>
|
||||||
|
<skip>true</skip>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
|
||||||
</project>
|
</project>
|
||||||
|
|||||||
@@ -1,3 +0,0 @@
|
|||||||
|
|
||||||
all:
|
|
||||||
protoc --java_out=../src/main/java/ OutgoingMessageSignal.proto PubSubMessage.proto
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
/**
|
|
||||||
* Copyright (C) 2013 - 2015 Open WhisperSystems
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU Affero General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU Affero General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
package textsecure;
|
|
||||||
|
|
||||||
option java_package = "org.whispersystems.textsecuregcm.entities";
|
|
||||||
option java_outer_classname = "MessageProtos";
|
|
||||||
|
|
||||||
message OutgoingMessageSignal {
|
|
||||||
enum Type {
|
|
||||||
UNKNOWN = 0;
|
|
||||||
CIPHERTEXT = 1;
|
|
||||||
KEY_EXCHANGE = 2;
|
|
||||||
PREKEY_BUNDLE = 3;
|
|
||||||
PLAINTEXT = 4;
|
|
||||||
RECEIPT = 5;
|
|
||||||
}
|
|
||||||
|
|
||||||
optional uint32 type = 1;
|
|
||||||
optional string source = 2;
|
|
||||||
optional uint32 sourceDevice = 7;
|
|
||||||
optional string relay = 3;
|
|
||||||
// repeated string destinations = 4;
|
|
||||||
optional uint64 timestamp = 5;
|
|
||||||
optional bytes message = 6;
|
|
||||||
}
|
|
||||||
|
|
||||||
message ProvisioningUuid {
|
|
||||||
optional string uuid = 1;
|
|
||||||
}
|
|
||||||
@@ -1,30 +0,0 @@
|
|||||||
/**
|
|
||||||
* Copyright (C) 2014 Open Whisper Systems
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU Affero General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU Affero General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
package textsecure;
|
|
||||||
|
|
||||||
option java_package = "org.whispersystems.textsecuregcm.storage";
|
|
||||||
option java_outer_classname = "StoredMessageProtos";
|
|
||||||
|
|
||||||
message StoredMessage {
|
|
||||||
enum Type {
|
|
||||||
UNKNOWN = 0;
|
|
||||||
MESSAGE = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
optional Type type = 1;
|
|
||||||
optional bytes content = 2;
|
|
||||||
}
|
|
||||||
20
redis-dispatch/pom.xml
Normal file
20
redis-dispatch/pom.xml
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||||
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<parent>
|
||||||
|
<artifactId>TextSecureServer</artifactId>
|
||||||
|
<groupId>org.whispersystems.textsecure</groupId>
|
||||||
|
<version>JGITVER</version>
|
||||||
|
</parent>
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
<artifactId>redis-dispatch</artifactId>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.slf4j</groupId>
|
||||||
|
<artifactId>slf4j-api</artifactId>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
</project>
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch;
|
||||||
|
|
||||||
|
public interface DispatchChannel {
|
||||||
|
void onDispatchMessage(String channel, byte[] message);
|
||||||
|
void onDispatchSubscribed(String channel);
|
||||||
|
void onDispatchUnsubscribed(String channel);
|
||||||
|
}
|
||||||
@@ -0,0 +1,157 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.whispersystems.dispatch.io.RedisPubSubConnectionFactory;
|
||||||
|
import org.whispersystems.dispatch.redis.PubSubConnection;
|
||||||
|
import org.whispersystems.dispatch.redis.PubSubReply;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
import java.util.concurrent.Executor;
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
|
||||||
|
@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
|
||||||
|
public class DispatchManager extends Thread {
|
||||||
|
|
||||||
|
private final Logger logger = LoggerFactory.getLogger(DispatchManager.class);
|
||||||
|
private final Executor executor = Executors.newCachedThreadPool();
|
||||||
|
private final Map<String, DispatchChannel> subscriptions = new ConcurrentHashMap<>();
|
||||||
|
|
||||||
|
private final Optional<DispatchChannel> deadLetterChannel;
|
||||||
|
private final RedisPubSubConnectionFactory redisPubSubConnectionFactory;
|
||||||
|
|
||||||
|
private PubSubConnection pubSubConnection;
|
||||||
|
private volatile boolean running;
|
||||||
|
|
||||||
|
public DispatchManager(RedisPubSubConnectionFactory redisPubSubConnectionFactory,
|
||||||
|
Optional<DispatchChannel> deadLetterChannel)
|
||||||
|
{
|
||||||
|
this.redisPubSubConnectionFactory = redisPubSubConnectionFactory;
|
||||||
|
this.deadLetterChannel = deadLetterChannel;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void start() {
|
||||||
|
this.pubSubConnection = redisPubSubConnectionFactory.connect();
|
||||||
|
this.running = true;
|
||||||
|
super.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void shutdown() {
|
||||||
|
this.running = false;
|
||||||
|
this.pubSubConnection.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
public synchronized void subscribe(String name, DispatchChannel dispatchChannel) {
|
||||||
|
Optional<DispatchChannel> previous = Optional.ofNullable(subscriptions.get(name));
|
||||||
|
subscriptions.put(name, dispatchChannel);
|
||||||
|
|
||||||
|
try {
|
||||||
|
pubSubConnection.subscribe(name);
|
||||||
|
} catch (IOException e) {
|
||||||
|
logger.warn("Subscription error", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
previous.ifPresent(channel -> dispatchUnsubscription(name, channel));
|
||||||
|
}
|
||||||
|
|
||||||
|
public synchronized void unsubscribe(String name, DispatchChannel channel) {
|
||||||
|
Optional<DispatchChannel> subscription = Optional.ofNullable(subscriptions.get(name));
|
||||||
|
|
||||||
|
if (subscription.isPresent() && subscription.get() == channel) {
|
||||||
|
subscriptions.remove(name);
|
||||||
|
|
||||||
|
try {
|
||||||
|
pubSubConnection.unsubscribe(name);
|
||||||
|
} catch (IOException e) {
|
||||||
|
logger.warn("Unsubscribe error", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
dispatchUnsubscription(name, subscription.get());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean hasSubscription(String name) {
|
||||||
|
return subscriptions.containsKey(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
while (running) {
|
||||||
|
try {
|
||||||
|
PubSubReply reply = pubSubConnection.read();
|
||||||
|
|
||||||
|
switch (reply.getType()) {
|
||||||
|
case UNSUBSCRIBE: break;
|
||||||
|
case SUBSCRIBE: dispatchSubscribe(reply); break;
|
||||||
|
case MESSAGE: dispatchMessage(reply); break;
|
||||||
|
default: throw new AssertionError("Unknown pubsub reply type! " + reply.getType());
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
logger.warn("***** PubSub Connection Error *****", e);
|
||||||
|
if (running) {
|
||||||
|
this.pubSubConnection.close();
|
||||||
|
this.pubSubConnection = redisPubSubConnectionFactory.connect();
|
||||||
|
resubscribeAll();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.warn("DispatchManager Shutting Down...");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void dispatchSubscribe(final PubSubReply reply) {
|
||||||
|
Optional<DispatchChannel> subscription = Optional.ofNullable(subscriptions.get(reply.getChannel()));
|
||||||
|
|
||||||
|
if (subscription.isPresent()) {
|
||||||
|
dispatchSubscription(reply.getChannel(), subscription.get());
|
||||||
|
} else {
|
||||||
|
logger.info("Received subscribe event for non-existing channel: " + reply.getChannel());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void dispatchMessage(PubSubReply reply) {
|
||||||
|
Optional<DispatchChannel> subscription = Optional.ofNullable(subscriptions.get(reply.getChannel()));
|
||||||
|
|
||||||
|
if (subscription.isPresent()) {
|
||||||
|
dispatchMessage(reply.getChannel(), subscription.get(), reply.getContent().get());
|
||||||
|
} else if (deadLetterChannel.isPresent()) {
|
||||||
|
dispatchMessage(reply.getChannel(), deadLetterChannel.get(), reply.getContent().get());
|
||||||
|
} else {
|
||||||
|
logger.warn("Received message for non-existing channel, with no dead letter handler: " + reply.getChannel());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void resubscribeAll() {
|
||||||
|
new Thread(() -> {
|
||||||
|
synchronized (DispatchManager.this) {
|
||||||
|
try {
|
||||||
|
for (String name : subscriptions.keySet()) {
|
||||||
|
pubSubConnection.subscribe(name);
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
logger.warn("***** RESUBSCRIPTION ERROR *****", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}).start();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void dispatchMessage(final String name, final DispatchChannel channel, final byte[] message) {
|
||||||
|
executor.execute(() -> channel.onDispatchMessage(name, message));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void dispatchSubscription(final String name, final DispatchChannel channel) {
|
||||||
|
executor.execute(() -> channel.onDispatchSubscribed(name));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void dispatchUnsubscription(final String name, final DispatchChannel channel) {
|
||||||
|
executor.execute(() -> channel.onDispatchUnsubscribed(name));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,68 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.io;
|
||||||
|
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
|
||||||
|
public class RedisInputStream {
|
||||||
|
|
||||||
|
private static final byte CR = 0x0D;
|
||||||
|
private static final byte LF = 0x0A;
|
||||||
|
|
||||||
|
private final InputStream inputStream;
|
||||||
|
|
||||||
|
public RedisInputStream(InputStream inputStream) {
|
||||||
|
this.inputStream = inputStream;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String readLine() throws IOException {
|
||||||
|
ByteArrayOutputStream baos = new ByteArrayOutputStream();
|
||||||
|
|
||||||
|
boolean foundCr = false;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
int character = inputStream.read();
|
||||||
|
|
||||||
|
if (character == -1) {
|
||||||
|
throw new IOException("Stream closed!");
|
||||||
|
}
|
||||||
|
|
||||||
|
baos.write(character);
|
||||||
|
|
||||||
|
if (foundCr && character == LF) break;
|
||||||
|
else if (character == CR) foundCr = true;
|
||||||
|
else if (foundCr) foundCr = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] data = baos.toByteArray();
|
||||||
|
return new String(data, 0, data.length-2);
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte[] readFully(int size) throws IOException {
|
||||||
|
byte[] result = new byte[size];
|
||||||
|
int offset = 0;
|
||||||
|
int remaining = result.length;
|
||||||
|
|
||||||
|
while (remaining > 0) {
|
||||||
|
int read = inputStream.read(result, offset, remaining);
|
||||||
|
|
||||||
|
if (read < 0) {
|
||||||
|
throw new IOException("Stream closed!");
|
||||||
|
}
|
||||||
|
|
||||||
|
offset += read;
|
||||||
|
remaining -= read;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void close() throws IOException {
|
||||||
|
inputStream.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.io;
|
||||||
|
|
||||||
|
import org.whispersystems.dispatch.redis.PubSubConnection;
|
||||||
|
|
||||||
|
public interface RedisPubSubConnectionFactory {
|
||||||
|
|
||||||
|
PubSubConnection connect();
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,123 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis;
|
||||||
|
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.whispersystems.dispatch.io.RedisInputStream;
|
||||||
|
import org.whispersystems.dispatch.redis.protocol.ArrayReplyHeader;
|
||||||
|
import org.whispersystems.dispatch.redis.protocol.IntReply;
|
||||||
|
import org.whispersystems.dispatch.redis.protocol.StringReplyHeader;
|
||||||
|
import org.whispersystems.dispatch.util.Util;
|
||||||
|
|
||||||
|
import java.io.BufferedInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.net.Socket;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
|
public class PubSubConnection {
|
||||||
|
|
||||||
|
private final Logger logger = LoggerFactory.getLogger(PubSubConnection.class);
|
||||||
|
|
||||||
|
private static final byte[] UNSUBSCRIBE_TYPE = {'u', 'n', 's', 'u', 'b', 's', 'c', 'r', 'i', 'b', 'e' };
|
||||||
|
private static final byte[] SUBSCRIBE_TYPE = {'s', 'u', 'b', 's', 'c', 'r', 'i', 'b', 'e' };
|
||||||
|
private static final byte[] MESSAGE_TYPE = {'m', 'e', 's', 's', 'a', 'g', 'e' };
|
||||||
|
|
||||||
|
private static final byte[] SUBSCRIBE_COMMAND = {'S', 'U', 'B', 'S', 'C', 'R', 'I', 'B', 'E', ' ' };
|
||||||
|
private static final byte[] UNSUBSCRIBE_COMMAND = {'U', 'N', 'S', 'U', 'B', 'S', 'C', 'R', 'I', 'B', 'E', ' '};
|
||||||
|
private static final byte[] CRLF = {'\r', '\n' };
|
||||||
|
|
||||||
|
private final OutputStream outputStream;
|
||||||
|
private final RedisInputStream inputStream;
|
||||||
|
private final Socket socket;
|
||||||
|
private final AtomicBoolean closed;
|
||||||
|
|
||||||
|
public PubSubConnection(Socket socket) throws IOException {
|
||||||
|
this.socket = socket;
|
||||||
|
this.outputStream = socket.getOutputStream();
|
||||||
|
this.inputStream = new RedisInputStream(new BufferedInputStream(socket.getInputStream()));
|
||||||
|
this.closed = new AtomicBoolean(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void subscribe(String channelName) throws IOException {
|
||||||
|
if (closed.get()) throw new IOException("Connection closed!");
|
||||||
|
|
||||||
|
byte[] command = Util.combine(SUBSCRIBE_COMMAND, channelName.getBytes(), CRLF);
|
||||||
|
outputStream.write(command);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void unsubscribe(String channelName) throws IOException {
|
||||||
|
if (closed.get()) throw new IOException("Connection closed!");
|
||||||
|
|
||||||
|
byte[] command = Util.combine(UNSUBSCRIBE_COMMAND, channelName.getBytes(), CRLF);
|
||||||
|
outputStream.write(command);
|
||||||
|
}
|
||||||
|
|
||||||
|
public PubSubReply read() throws IOException {
|
||||||
|
if (closed.get()) throw new IOException("Connection closed!");
|
||||||
|
|
||||||
|
ArrayReplyHeader replyHeader = new ArrayReplyHeader(inputStream.readLine());
|
||||||
|
|
||||||
|
if (replyHeader.getElementCount() != 3) {
|
||||||
|
throw new IOException("Received array reply header with strange count: " + replyHeader.getElementCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
StringReplyHeader replyTypeHeader = new StringReplyHeader(inputStream.readLine());
|
||||||
|
byte[] replyType = inputStream.readFully(replyTypeHeader.getStringLength());
|
||||||
|
inputStream.readLine();
|
||||||
|
|
||||||
|
if (Arrays.equals(SUBSCRIBE_TYPE, replyType)) return readSubscribeReply();
|
||||||
|
else if (Arrays.equals(UNSUBSCRIBE_TYPE, replyType)) return readUnsubscribeReply();
|
||||||
|
else if (Arrays.equals(MESSAGE_TYPE, replyType)) return readMessageReply();
|
||||||
|
else throw new IOException("Unknown reply type: " + new String(replyType));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void close() {
|
||||||
|
try {
|
||||||
|
this.closed.set(true);
|
||||||
|
this.inputStream.close();
|
||||||
|
this.outputStream.close();
|
||||||
|
this.socket.close();
|
||||||
|
} catch (IOException e) {
|
||||||
|
logger.warn("Exception while closing", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private PubSubReply readMessageReply() throws IOException {
|
||||||
|
StringReplyHeader channelNameHeader = new StringReplyHeader(inputStream.readLine());
|
||||||
|
byte[] channelName = inputStream.readFully(channelNameHeader.getStringLength());
|
||||||
|
inputStream.readLine();
|
||||||
|
|
||||||
|
StringReplyHeader messageHeader = new StringReplyHeader(inputStream.readLine());
|
||||||
|
byte[] message = inputStream.readFully(messageHeader.getStringLength());
|
||||||
|
inputStream.readLine();
|
||||||
|
|
||||||
|
return new PubSubReply(PubSubReply.Type.MESSAGE, new String(channelName), Optional.of(message));
|
||||||
|
}
|
||||||
|
|
||||||
|
private PubSubReply readUnsubscribeReply() throws IOException {
|
||||||
|
String channelName = readSubscriptionReply();
|
||||||
|
return new PubSubReply(PubSubReply.Type.UNSUBSCRIBE, channelName, Optional.empty());
|
||||||
|
}
|
||||||
|
|
||||||
|
private PubSubReply readSubscribeReply() throws IOException {
|
||||||
|
String channelName = readSubscriptionReply();
|
||||||
|
return new PubSubReply(PubSubReply.Type.SUBSCRIBE, channelName, Optional.empty());
|
||||||
|
}
|
||||||
|
|
||||||
|
private String readSubscriptionReply() throws IOException {
|
||||||
|
StringReplyHeader channelNameHeader = new StringReplyHeader(inputStream.readLine());
|
||||||
|
byte[] channelName = inputStream.readFully(channelNameHeader.getStringLength());
|
||||||
|
inputStream.readLine();
|
||||||
|
|
||||||
|
new IntReply(inputStream.readLine());
|
||||||
|
|
||||||
|
return new String(channelName);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
|
||||||
|
public class PubSubReply {
|
||||||
|
|
||||||
|
public enum Type {
|
||||||
|
MESSAGE,
|
||||||
|
SUBSCRIBE,
|
||||||
|
UNSUBSCRIBE
|
||||||
|
}
|
||||||
|
|
||||||
|
private final Type type;
|
||||||
|
private final String channel;
|
||||||
|
private final Optional<byte[]> content;
|
||||||
|
|
||||||
|
public PubSubReply(Type type, String channel, Optional<byte[]> content) {
|
||||||
|
this.type = type;
|
||||||
|
this.channel = channel;
|
||||||
|
this.content = content;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type getType() {
|
||||||
|
return type;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getChannel() {
|
||||||
|
return channel;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Optional<byte[]> getContent() {
|
||||||
|
return content;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis.protocol;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class ArrayReplyHeader {
|
||||||
|
|
||||||
|
private final int elementCount;
|
||||||
|
|
||||||
|
public ArrayReplyHeader(String header) throws IOException {
|
||||||
|
if (header == null || header.length() < 2 || header.charAt(0) != '*') {
|
||||||
|
throw new IOException("Invalid array reply header: " + header);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
this.elementCount = Integer.parseInt(header.substring(1));
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
throw new IOException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getElementCount() {
|
||||||
|
return elementCount;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis.protocol;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class IntReply {
|
||||||
|
|
||||||
|
private final int value;
|
||||||
|
|
||||||
|
public IntReply(String reply) throws IOException {
|
||||||
|
if (reply == null || reply.length() < 2 || reply.charAt(0) != ':') {
|
||||||
|
throw new IOException("Invalid int reply: " + reply);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
this.value = Integer.parseInt(reply.substring(1));
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
throw new IOException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getValue() {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis.protocol;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class StringReplyHeader {
|
||||||
|
|
||||||
|
private final int stringLength;
|
||||||
|
|
||||||
|
public StringReplyHeader(String header) throws IOException {
|
||||||
|
if (header == null || header.length() < 2 || header.charAt(0) != '$') {
|
||||||
|
throw new IOException("Invalid string reply header: " + header);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
this.stringLength = Integer.parseInt(header.substring(1));
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
throw new IOException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getStringLength() {
|
||||||
|
return stringLength;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.util;
|
||||||
|
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class Util {
|
||||||
|
|
||||||
|
public static byte[] combine(byte[]... elements) {
|
||||||
|
try {
|
||||||
|
int sum = 0;
|
||||||
|
|
||||||
|
for (byte[] element : elements) {
|
||||||
|
sum += element.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
ByteArrayOutputStream baos = new ByteArrayOutputStream(sum);
|
||||||
|
|
||||||
|
for (byte[] element : elements) {
|
||||||
|
baos.write(element);
|
||||||
|
}
|
||||||
|
|
||||||
|
return baos.toByteArray();
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new AssertionError(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static void sleep(long millis) {
|
||||||
|
try {
|
||||||
|
Thread.sleep(millis);
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
throw new AssertionError(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,123 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertArrayEquals;
|
||||||
|
import static org.mockito.Mockito.eq;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.timeout;
|
||||||
|
import static org.mockito.Mockito.verify;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.mockito.ArgumentCaptor;
|
||||||
|
import org.mockito.stubbing.Answer;
|
||||||
|
import org.whispersystems.dispatch.io.RedisPubSubConnectionFactory;
|
||||||
|
import org.whispersystems.dispatch.redis.PubSubConnection;
|
||||||
|
import org.whispersystems.dispatch.redis.PubSubReply;
|
||||||
|
|
||||||
|
public class DispatchManagerTest {
|
||||||
|
|
||||||
|
private PubSubConnection pubSubConnection;
|
||||||
|
private RedisPubSubConnectionFactory socketFactory;
|
||||||
|
private DispatchManager dispatchManager;
|
||||||
|
private PubSubReplyInputStream pubSubReplyInputStream;
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void setUp() throws Exception {
|
||||||
|
pubSubConnection = mock(PubSubConnection.class );
|
||||||
|
socketFactory = mock(RedisPubSubConnectionFactory.class);
|
||||||
|
pubSubReplyInputStream = new PubSubReplyInputStream();
|
||||||
|
|
||||||
|
when(socketFactory.connect()).thenReturn(pubSubConnection);
|
||||||
|
when(pubSubConnection.read()).thenAnswer((Answer<PubSubReply>) invocationOnMock -> pubSubReplyInputStream.read());
|
||||||
|
|
||||||
|
dispatchManager = new DispatchManager(socketFactory, Optional.empty());
|
||||||
|
dispatchManager.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
void tearDown() {
|
||||||
|
dispatchManager.shutdown();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testConnect() {
|
||||||
|
verify(socketFactory).connect();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSubscribe() {
|
||||||
|
DispatchChannel dispatchChannel = mock(DispatchChannel.class);
|
||||||
|
dispatchManager.subscribe("foo", dispatchChannel);
|
||||||
|
pubSubReplyInputStream.write(new PubSubReply(PubSubReply.Type.SUBSCRIBE, "foo", Optional.empty()));
|
||||||
|
|
||||||
|
verify(dispatchChannel, timeout(1000)).onDispatchSubscribed(eq("foo"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSubscribeUnsubscribe() {
|
||||||
|
DispatchChannel dispatchChannel = mock(DispatchChannel.class);
|
||||||
|
dispatchManager.subscribe("foo", dispatchChannel);
|
||||||
|
dispatchManager.unsubscribe("foo", dispatchChannel);
|
||||||
|
|
||||||
|
pubSubReplyInputStream.write(new PubSubReply(PubSubReply.Type.SUBSCRIBE, "foo", Optional.empty()));
|
||||||
|
pubSubReplyInputStream.write(new PubSubReply(PubSubReply.Type.UNSUBSCRIBE, "foo", Optional.empty()));
|
||||||
|
|
||||||
|
verify(dispatchChannel, timeout(1000)).onDispatchUnsubscribed(eq("foo"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMessages() {
|
||||||
|
DispatchChannel fooChannel = mock(DispatchChannel.class);
|
||||||
|
DispatchChannel barChannel = mock(DispatchChannel.class);
|
||||||
|
|
||||||
|
dispatchManager.subscribe("foo", fooChannel);
|
||||||
|
dispatchManager.subscribe("bar", barChannel);
|
||||||
|
|
||||||
|
pubSubReplyInputStream.write(new PubSubReply(PubSubReply.Type.SUBSCRIBE, "foo", Optional.empty()));
|
||||||
|
pubSubReplyInputStream.write(new PubSubReply(PubSubReply.Type.SUBSCRIBE, "bar", Optional.empty()));
|
||||||
|
|
||||||
|
verify(fooChannel, timeout(1000)).onDispatchSubscribed(eq("foo"));
|
||||||
|
verify(barChannel, timeout(1000)).onDispatchSubscribed(eq("bar"));
|
||||||
|
|
||||||
|
pubSubReplyInputStream.write(new PubSubReply(PubSubReply.Type.MESSAGE, "foo", Optional.of("hello".getBytes())));
|
||||||
|
pubSubReplyInputStream.write(new PubSubReply(PubSubReply.Type.MESSAGE, "bar", Optional.of("there".getBytes())));
|
||||||
|
|
||||||
|
ArgumentCaptor<byte[]> captor = ArgumentCaptor.forClass(byte[].class);
|
||||||
|
verify(fooChannel, timeout(1000)).onDispatchMessage(eq("foo"), captor.capture());
|
||||||
|
|
||||||
|
assertArrayEquals("hello".getBytes(), captor.getValue());
|
||||||
|
|
||||||
|
verify(barChannel, timeout(1000)).onDispatchMessage(eq("bar"), captor.capture());
|
||||||
|
|
||||||
|
assertArrayEquals("there".getBytes(), captor.getValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class PubSubReplyInputStream {
|
||||||
|
|
||||||
|
private final List<PubSubReply> pubSubReplyList = new LinkedList<>();
|
||||||
|
|
||||||
|
public synchronized PubSubReply read() {
|
||||||
|
try {
|
||||||
|
while (pubSubReplyList.isEmpty()) wait();
|
||||||
|
return pubSubReplyList.remove(0);
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
throw new AssertionError(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public synchronized void write(PubSubReply pubSubReply) {
|
||||||
|
pubSubReplyList.add(pubSubReply);
|
||||||
|
notifyAll();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,264 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertArrayEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertFalse;
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.ArgumentMatchers.anyInt;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.verify;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.net.Socket;
|
||||||
|
import java.security.SecureRandom;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.mockito.ArgumentCaptor;
|
||||||
|
import org.mockito.invocation.InvocationOnMock;
|
||||||
|
import org.mockito.stubbing.Answer;
|
||||||
|
|
||||||
|
class PubSubConnectionTest {
|
||||||
|
|
||||||
|
private static final String REPLY = "*3\r\n" +
|
||||||
|
"$9\r\n" +
|
||||||
|
"subscribe\r\n" +
|
||||||
|
"$5\r\n" +
|
||||||
|
"abcde\r\n" +
|
||||||
|
":1\r\n" +
|
||||||
|
"*3\r\n" +
|
||||||
|
"$9\r\n" +
|
||||||
|
"subscribe\r\n" +
|
||||||
|
"$5\r\n" +
|
||||||
|
"fghij\r\n" +
|
||||||
|
":2\r\n" +
|
||||||
|
"*3\r\n" +
|
||||||
|
"$9\r\n" +
|
||||||
|
"subscribe\r\n" +
|
||||||
|
"$5\r\n" +
|
||||||
|
"klmno\r\n" +
|
||||||
|
":2\r\n" +
|
||||||
|
"*3\r\n" +
|
||||||
|
"$7\r\n" +
|
||||||
|
"message\r\n" +
|
||||||
|
"$5\r\n" +
|
||||||
|
"abcde\r\n" +
|
||||||
|
"$10\r\n" +
|
||||||
|
"1234567890\r\n" +
|
||||||
|
"*3\r\n" +
|
||||||
|
"$7\r\n" +
|
||||||
|
"message\r\n" +
|
||||||
|
"$5\r\n" +
|
||||||
|
"klmno\r\n" +
|
||||||
|
"$10\r\n" +
|
||||||
|
"0987654321\r\n";
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testSubscribe() throws IOException {
|
||||||
|
OutputStream outputStream = mock(OutputStream.class);
|
||||||
|
Socket socket = mock(Socket.class );
|
||||||
|
when(socket.getOutputStream()).thenReturn(outputStream);
|
||||||
|
PubSubConnection connection = new PubSubConnection(socket);
|
||||||
|
|
||||||
|
connection.subscribe("foobar");
|
||||||
|
|
||||||
|
ArgumentCaptor<byte[]> captor = ArgumentCaptor.forClass(byte[].class);
|
||||||
|
verify(outputStream).write(captor.capture());
|
||||||
|
|
||||||
|
assertArrayEquals(captor.getValue(), "SUBSCRIBE foobar\r\n".getBytes());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testUnsubscribe() throws IOException {
|
||||||
|
OutputStream outputStream = mock(OutputStream.class);
|
||||||
|
Socket socket = mock(Socket.class );
|
||||||
|
when(socket.getOutputStream()).thenReturn(outputStream);
|
||||||
|
PubSubConnection connection = new PubSubConnection(socket);
|
||||||
|
|
||||||
|
connection.unsubscribe("bazbar");
|
||||||
|
|
||||||
|
ArgumentCaptor<byte[]> captor = ArgumentCaptor.forClass(byte[].class);
|
||||||
|
verify(outputStream).write(captor.capture());
|
||||||
|
|
||||||
|
assertArrayEquals(captor.getValue(), "UNSUBSCRIBE bazbar\r\n".getBytes());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTricklyResponse() throws Exception {
|
||||||
|
InputStream inputStream = mockInputStreamFor(new TrickleInputStream(REPLY.getBytes()));
|
||||||
|
OutputStream outputStream = mock(OutputStream.class);
|
||||||
|
Socket socket = mock(Socket.class );
|
||||||
|
when(socket.getOutputStream()).thenReturn(outputStream);
|
||||||
|
when(socket.getInputStream()).thenReturn(inputStream);
|
||||||
|
|
||||||
|
PubSubConnection pubSubConnection = new PubSubConnection(socket);
|
||||||
|
readResponses(pubSubConnection);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testFullResponse() throws Exception {
|
||||||
|
InputStream inputStream = mockInputStreamFor(new FullInputStream(REPLY.getBytes()));
|
||||||
|
OutputStream outputStream = mock(OutputStream.class);
|
||||||
|
Socket socket = mock(Socket.class );
|
||||||
|
when(socket.getOutputStream()).thenReturn(outputStream);
|
||||||
|
when(socket.getInputStream()).thenReturn(inputStream);
|
||||||
|
|
||||||
|
PubSubConnection pubSubConnection = new PubSubConnection(socket);
|
||||||
|
readResponses(pubSubConnection);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testRandomLengthResponse() throws Exception {
|
||||||
|
InputStream inputStream = mockInputStreamFor(new RandomInputStream(REPLY.getBytes()));
|
||||||
|
OutputStream outputStream = mock(OutputStream.class);
|
||||||
|
Socket socket = mock(Socket.class );
|
||||||
|
when(socket.getOutputStream()).thenReturn(outputStream);
|
||||||
|
when(socket.getInputStream()).thenReturn(inputStream);
|
||||||
|
|
||||||
|
PubSubConnection pubSubConnection = new PubSubConnection(socket);
|
||||||
|
readResponses(pubSubConnection);
|
||||||
|
}
|
||||||
|
|
||||||
|
private InputStream mockInputStreamFor(final MockInputStream stub) throws IOException {
|
||||||
|
InputStream result = mock(InputStream.class);
|
||||||
|
|
||||||
|
when(result.read()).thenAnswer(new Answer<Integer>() {
|
||||||
|
@Override
|
||||||
|
public Integer answer(InvocationOnMock invocationOnMock) throws Throwable {
|
||||||
|
return stub.read();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
when(result.read(any(byte[].class))).thenAnswer(new Answer<Integer>() {
|
||||||
|
@Override
|
||||||
|
public Integer answer(InvocationOnMock invocationOnMock) throws Throwable {
|
||||||
|
byte[] buffer = (byte[])invocationOnMock.getArguments()[0];
|
||||||
|
return stub.read(buffer, 0, buffer.length);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
when(result.read(any(byte[].class), anyInt(), anyInt())).thenAnswer(new Answer<Integer>() {
|
||||||
|
@Override
|
||||||
|
public Integer answer(InvocationOnMock invocationOnMock) throws Throwable {
|
||||||
|
byte[] buffer = (byte[]) invocationOnMock.getArguments()[0];
|
||||||
|
int offset = (int) invocationOnMock.getArguments()[1];
|
||||||
|
int length = (int) invocationOnMock.getArguments()[2];
|
||||||
|
|
||||||
|
return stub.read(buffer, offset, length);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void readResponses(PubSubConnection pubSubConnection) throws Exception {
|
||||||
|
PubSubReply reply = pubSubConnection.read();
|
||||||
|
|
||||||
|
assertEquals(reply.getType(), PubSubReply.Type.SUBSCRIBE);
|
||||||
|
assertEquals(reply.getChannel(), "abcde");
|
||||||
|
assertFalse(reply.getContent().isPresent());
|
||||||
|
|
||||||
|
reply = pubSubConnection.read();
|
||||||
|
|
||||||
|
assertEquals(reply.getType(), PubSubReply.Type.SUBSCRIBE);
|
||||||
|
assertEquals(reply.getChannel(), "fghij");
|
||||||
|
assertFalse(reply.getContent().isPresent());
|
||||||
|
|
||||||
|
reply = pubSubConnection.read();
|
||||||
|
|
||||||
|
assertEquals(reply.getType(), PubSubReply.Type.SUBSCRIBE);
|
||||||
|
assertEquals(reply.getChannel(), "klmno");
|
||||||
|
assertFalse(reply.getContent().isPresent());
|
||||||
|
|
||||||
|
reply = pubSubConnection.read();
|
||||||
|
|
||||||
|
assertEquals(reply.getType(), PubSubReply.Type.MESSAGE);
|
||||||
|
assertEquals(reply.getChannel(), "abcde");
|
||||||
|
assertArrayEquals(reply.getContent().get(), "1234567890".getBytes());
|
||||||
|
|
||||||
|
reply = pubSubConnection.read();
|
||||||
|
|
||||||
|
assertEquals(reply.getType(), PubSubReply.Type.MESSAGE);
|
||||||
|
assertEquals(reply.getChannel(), "klmno");
|
||||||
|
assertArrayEquals(reply.getContent().get(), "0987654321".getBytes());
|
||||||
|
}
|
||||||
|
|
||||||
|
private interface MockInputStream {
|
||||||
|
public int read();
|
||||||
|
public int read(byte[] input, int offset, int length);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TrickleInputStream implements MockInputStream {
|
||||||
|
|
||||||
|
private final byte[] data;
|
||||||
|
private int index = 0;
|
||||||
|
|
||||||
|
private TrickleInputStream(byte[] data) {
|
||||||
|
this.data = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int read() {
|
||||||
|
return data[index++];
|
||||||
|
}
|
||||||
|
|
||||||
|
public int read(byte[] input, int offset, int length) {
|
||||||
|
input[offset] = data[index++];
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class FullInputStream implements MockInputStream {
|
||||||
|
|
||||||
|
private final byte[] data;
|
||||||
|
private int index = 0;
|
||||||
|
|
||||||
|
private FullInputStream(byte[] data) {
|
||||||
|
this.data = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int read() {
|
||||||
|
return data[index++];
|
||||||
|
}
|
||||||
|
|
||||||
|
public int read(byte[] input, int offset, int length) {
|
||||||
|
int amount = Math.min(data.length - index, length);
|
||||||
|
System.arraycopy(data, index, input, offset, amount);
|
||||||
|
index += length;
|
||||||
|
|
||||||
|
return amount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class RandomInputStream implements MockInputStream {
|
||||||
|
private final byte[] data;
|
||||||
|
private int index = 0;
|
||||||
|
|
||||||
|
private RandomInputStream(byte[] data) {
|
||||||
|
this.data = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int read() {
|
||||||
|
return data[index++];
|
||||||
|
}
|
||||||
|
|
||||||
|
public int read(byte[] input, int offset, int length) {
|
||||||
|
int maxCopy = Math.min(data.length - index, length);
|
||||||
|
int randomCopy = new SecureRandom().nextInt(maxCopy) + 1;
|
||||||
|
int copyAmount = Math.min(maxCopy, randomCopy);
|
||||||
|
|
||||||
|
System.arraycopy(data, index, input, offset, copyAmount);
|
||||||
|
index += copyAmount;
|
||||||
|
|
||||||
|
return copyAmount;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis.protocol;
|
||||||
|
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
class ArrayReplyHeaderTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testNull() {
|
||||||
|
assertThrows(IOException.class, () -> new ArrayReplyHeader(null));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testBadPrefix() {
|
||||||
|
assertThrows(IOException.class, () -> new ArrayReplyHeader(":3"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testEmpty() {
|
||||||
|
assertThrows(IOException.class, () -> new ArrayReplyHeader(""));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTruncated() {
|
||||||
|
assertThrows(IOException.class, () -> new ArrayReplyHeader("*"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testBadNumber() {
|
||||||
|
assertThrows(IOException.class, () -> new ArrayReplyHeader("*ABC"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testValid() throws IOException {
|
||||||
|
assertEquals(4, new ArrayReplyHeader("*4").getElementCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis.protocol;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
class IntReplyHeaderTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testNull() {
|
||||||
|
assertThrows(IOException.class, () -> new IntReply(null));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testEmpty() {
|
||||||
|
assertThrows(IOException.class, () -> new IntReply(""));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testBadNumber() {
|
||||||
|
assertThrows(IOException.class, () -> new IntReply(":A"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testBadFormat() {
|
||||||
|
assertThrows(IOException.class, () -> new IntReply("*"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testValid() throws IOException {
|
||||||
|
assertEquals(23, new IntReply(":23").getValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.dispatch.redis.protocol;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
class StringReplyHeaderTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testNull() {
|
||||||
|
assertThrows(IOException.class, () -> new StringReplyHeader(null));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testBadNumber() {
|
||||||
|
assertThrows(IOException.class, () -> new StringReplyHeader("$100A"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testBadPrefix() {
|
||||||
|
assertThrows(IOException.class, () -> new StringReplyHeader("*"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testValid() throws IOException {
|
||||||
|
assertEquals(1000, new StringReplyHeader("$1000").getStringLength());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
<assembly xmlns="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.0"
|
<assembly xmlns="http://maven.apache.org/ASSEMBLY/2.1.0"
|
||||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
xsi:schemaLocation="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.0 http://maven.apache.org/xsd/assembly-1.1.0.xsd">
|
xsi:schemaLocation="http://maven.apache.org/ASSEMBLY/2.1.0 http://maven.apache.org/xsd/assembly-2.1.0.xsd">
|
||||||
<id>bin</id>
|
<id>bin</id>
|
||||||
<includeBaseDirectory>false</includeBaseDirectory>
|
<includeBaseDirectory>false</includeBaseDirectory>
|
||||||
<formats>
|
<formats>
|
||||||
@@ -18,8 +18,8 @@
|
|||||||
<directory>${project.build.directory}</directory>
|
<directory>${project.build.directory}</directory>
|
||||||
<outputDirectory>/</outputDirectory>
|
<outputDirectory>/</outputDirectory>
|
||||||
<includes>
|
<includes>
|
||||||
<include>${project.name}-${project.version}.jar</include>
|
<include>${parent.artifactId}-${project.version}.jar</include>
|
||||||
</includes>
|
</includes>
|
||||||
</fileSet>
|
</fileSet>
|
||||||
</fileSets>
|
</fileSets>
|
||||||
</assembly>
|
</assembly>
|
||||||
402
service/config/sample.yml
Normal file
402
service/config/sample.yml
Normal file
@@ -0,0 +1,402 @@
|
|||||||
|
# Example, relatively minimal, configuration that passes validation (see `io.dropwizard.cli.CheckCommand`)
|
||||||
|
#
|
||||||
|
# `unset` values will need to be set to work properly.
|
||||||
|
# Most other values are technically valid for a local/demonstration environment, but are probably not production-ready.
|
||||||
|
|
||||||
|
adminEventLoggingConfiguration:
|
||||||
|
credentials: |
|
||||||
|
Some credentials text
|
||||||
|
blah blah blah
|
||||||
|
projectId: some-project-id
|
||||||
|
logName: some-log-name
|
||||||
|
|
||||||
|
stripe:
|
||||||
|
apiKey: unset
|
||||||
|
idempotencyKeyGenerator: abcdefg12345678= # base64 for creating request idempotency hash
|
||||||
|
boostDescription: >
|
||||||
|
Example
|
||||||
|
|
||||||
|
dynamoDbClientConfiguration:
|
||||||
|
region: us-west-2 # AWS Region
|
||||||
|
|
||||||
|
dynamoDbTables:
|
||||||
|
accounts:
|
||||||
|
tableName: Example_Accounts
|
||||||
|
phoneNumberTableName: Example_Accounts_PhoneNumbers
|
||||||
|
phoneNumberIdentifierTableName: Example_Accounts_PhoneNumberIdentifiers
|
||||||
|
usernamesTableName: Example_Accounts_Usernames
|
||||||
|
scanPageSize: 100
|
||||||
|
deletedAccounts:
|
||||||
|
tableName: Example_DeletedAccounts
|
||||||
|
needsReconciliationIndexName: NeedsReconciliation
|
||||||
|
deletedAccountsLock:
|
||||||
|
tableName: Example_DeletedAccountsLock
|
||||||
|
issuedReceipts:
|
||||||
|
tableName: Example_IssuedReceipts
|
||||||
|
expiration: P30D # Duration of time until rows expire
|
||||||
|
generator: abcdefg12345678= # random base64-encoded binary sequence
|
||||||
|
keys:
|
||||||
|
tableName: Example_Keys
|
||||||
|
messages:
|
||||||
|
tableName: Example_Messages
|
||||||
|
expiration: P30D # Duration of time until rows expire
|
||||||
|
pendingAccounts:
|
||||||
|
tableName: Example_PendingAccounts
|
||||||
|
pendingDevices:
|
||||||
|
tableName: Example_PendingDevices
|
||||||
|
phoneNumberIdentifiers:
|
||||||
|
tableName: Example_PhoneNumberIdentifiers
|
||||||
|
profiles:
|
||||||
|
tableName: Example_Profiles
|
||||||
|
pushChallenge:
|
||||||
|
tableName: Example_PushChallenge
|
||||||
|
redeemedReceipts:
|
||||||
|
tableName: Example_RedeemedReceipts
|
||||||
|
expiration: P30D # Duration of time until rows expire
|
||||||
|
remoteConfig:
|
||||||
|
tableName: Example_RemoteConfig
|
||||||
|
reportMessage:
|
||||||
|
tableName: Example_ReportMessage
|
||||||
|
reservedUsernames:
|
||||||
|
tableName: Example_ReservedUsernames
|
||||||
|
subscriptions:
|
||||||
|
tableName: Example_Subscriptions
|
||||||
|
|
||||||
|
twilio: # Twilio gateway configuration
|
||||||
|
accountId: unset
|
||||||
|
accountToken: unset
|
||||||
|
nanpaMessagingServiceSid: unset # Twilio SID for the messaging service to use for NANPA.
|
||||||
|
messagingServiceSid: unset # Twilio SID for the message service to use for non-NANPA.
|
||||||
|
verifyServiceSid: unset # Twilio SID for a Verify service
|
||||||
|
localDomain: example.com # Domain Twilio can connect back to for calls. Should be domain of your service.
|
||||||
|
defaultClientVerificationTexts:
|
||||||
|
ios: example %1$s # Text to use for the verification message on iOS. Will be passed to String.format with the verification code as argument 1.
|
||||||
|
androidNg: example %1$s # Text to use for the verification message on android-ng client types. Will be passed to String.format with the verification code as argument 1.
|
||||||
|
android202001: example %1$s # Text to use for the verification message on android-2020-01 client types. Will be passed to String.format with the verification code as argument 1.
|
||||||
|
android202103: example %1$s # Text to use for the verification message on android-2021-03 client types. Will be passed to String.format with the verification code as argument 1.
|
||||||
|
generic: example %1$s # Text to use when the client type is unrecognized. Will be passed to String.format with the verification code as argument 1.
|
||||||
|
regionalClientVerificationTexts: # Map of country codes to custom texts
|
||||||
|
999: # example country code
|
||||||
|
ios: example %1$s # all keys from defaultClientVerificationTexts are required
|
||||||
|
androidNg: example %1$s
|
||||||
|
android202001: example %1$s
|
||||||
|
android202103: example %1$s
|
||||||
|
generic: example %1$s
|
||||||
|
androidAppHash: example # Hash appended to Android
|
||||||
|
verifyServiceFriendlyName: example # Service name used in template. Requires Twilio account rep to enable
|
||||||
|
|
||||||
|
cacheCluster: # Redis server configuration for cache cluster
|
||||||
|
configurationUri: redis://redis.example.com:6379/
|
||||||
|
|
||||||
|
clientPresenceCluster: # Redis server configuration for client presence cluster
|
||||||
|
configurationUri: redis://redis.example.com:6379/
|
||||||
|
|
||||||
|
pubsub: # Redis server configuration for pubsub cluster
|
||||||
|
url: redis://redis.example.com:6379/
|
||||||
|
replicaUrls:
|
||||||
|
- redis://redis.example.com:6379/
|
||||||
|
|
||||||
|
pushSchedulerCluster: # Redis server configuration for push scheduler cluster
|
||||||
|
configurationUri: redis://redis.example.com:6379/
|
||||||
|
|
||||||
|
rateLimitersCluster: # Redis server configuration for rate limiters cluster
|
||||||
|
configurationUri: redis://redis.example.com:6379/
|
||||||
|
|
||||||
|
directory:
|
||||||
|
client: # Configuration for interfacing with Contact Discovery Service cluster
|
||||||
|
userAuthenticationTokenSharedSecret: 00000f # hex-encoded secret shared with CDS used to generate auth tokens for Signal users
|
||||||
|
userAuthenticationTokenUserIdSecret: 00000f # hex-encoded secret shared among Signal-Servers to obscure user phone numbers from CDS
|
||||||
|
sqs:
|
||||||
|
accessKey: test # AWS SQS accessKey
|
||||||
|
accessSecret: test # AWS SQS accessSecret
|
||||||
|
queueUrls: # AWS SQS queue urls
|
||||||
|
- https://sqs.example.com/directory.fifo
|
||||||
|
server: # One or more CDS servers
|
||||||
|
- replicationName: example # CDS replication name
|
||||||
|
replicationUrl: cds.example.com # CDS replication endpoint base url
|
||||||
|
replicationPassword: example # CDS replication endpoint password
|
||||||
|
replicationCaCertificate: | # CDS replication endpoint TLS certificate trust root
|
||||||
|
-----BEGIN CERTIFICATE-----
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
AAAAAAAAAAAAAAAAAAAA
|
||||||
|
-----END CERTIFICATE-----
|
||||||
|
|
||||||
|
directoryV2:
|
||||||
|
client: # Configuration for interfacing with Contact Discovery Service v2 cluster
|
||||||
|
userAuthenticationTokenSharedSecret: abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG= # base64-encoded secret shared with CDS to generate auth tokens for Signal users
|
||||||
|
userIdTokenSharedSecret: bbcdefghijklmnopqrstuvwxyz0123456789ABCDEFG= # base64-encoded secret shared with CDS to generate auth identity tokens for Signal users
|
||||||
|
|
||||||
|
messageCache: # Redis server configuration for message store cache
|
||||||
|
persistDelayMinutes: 1
|
||||||
|
cluster:
|
||||||
|
configurationUri: redis://redis.example.com:6379/
|
||||||
|
|
||||||
|
metricsCluster:
|
||||||
|
configurationUri: redis://redis.example.com:6379/
|
||||||
|
|
||||||
|
awsAttachments: # AWS S3 configuration
|
||||||
|
accessKey: test
|
||||||
|
accessSecret: test
|
||||||
|
bucket: aws-attachments
|
||||||
|
region: us-west-2
|
||||||
|
|
||||||
|
gcpAttachments: # GCP Storage configuration
|
||||||
|
domain: example.com
|
||||||
|
email: user@example.cocm
|
||||||
|
maxSizeInBytes: 1024
|
||||||
|
pathPrefix:
|
||||||
|
rsaSigningKey: |
|
||||||
|
-----BEGIN PRIVATE KEY-----
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
AAAAAAAA
|
||||||
|
-----END PRIVATE KEY-----
|
||||||
|
|
||||||
|
accountDatabaseCrawler:
|
||||||
|
chunkSize: 10 # accounts per run
|
||||||
|
chunkIntervalMs: 60000 # time per run
|
||||||
|
|
||||||
|
apn: # Apple Push Notifications configuration
|
||||||
|
sandbox: true
|
||||||
|
bundleId: com.example.textsecuregcm
|
||||||
|
keyId: unset
|
||||||
|
teamId: unset
|
||||||
|
signingKey: |
|
||||||
|
-----BEGIN PRIVATE KEY-----
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
AAAAAAAA
|
||||||
|
-----END PRIVATE KEY-----
|
||||||
|
|
||||||
|
fcm: # FCM configuration
|
||||||
|
credentials: |
|
||||||
|
{ "json": true }
|
||||||
|
|
||||||
|
cdn:
|
||||||
|
accessKey: test # AWS Access Key ID
|
||||||
|
accessSecret: test # AWS Access Secret
|
||||||
|
bucket: cdn # S3 Bucket name
|
||||||
|
region: us-west-2 # AWS region
|
||||||
|
|
||||||
|
datadog:
|
||||||
|
apiKey: unset
|
||||||
|
environment: dev
|
||||||
|
|
||||||
|
unidentifiedDelivery:
|
||||||
|
certificate: ABCD1234
|
||||||
|
privateKey: ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789AAAAAAA
|
||||||
|
expiresDays: 7
|
||||||
|
|
||||||
|
voiceVerification:
|
||||||
|
url: https://cdn-ca.signal.org/verification/
|
||||||
|
locales:
|
||||||
|
- en
|
||||||
|
|
||||||
|
recaptcha:
|
||||||
|
projectPath: projects/example
|
||||||
|
credentialConfigurationJson: "{ }" # service account configuration for backend authentication
|
||||||
|
|
||||||
|
storageService:
|
||||||
|
uri: storage.example.com
|
||||||
|
userAuthenticationTokenSharedSecret: 00000f
|
||||||
|
storageCaCertificate: |
|
||||||
|
-----BEGIN CERTIFICATE-----
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
AAAAAAAAAAAAAAAAAAAA
|
||||||
|
-----END CERTIFICATE-----
|
||||||
|
|
||||||
|
backupService:
|
||||||
|
uri: backup.example.com
|
||||||
|
userAuthenticationTokenSharedSecret: 00000f
|
||||||
|
backupCaCertificate: |
|
||||||
|
-----BEGIN CERTIFICATE-----
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
AAAAAAAAAAAAAAAAAAAA
|
||||||
|
-----END CERTIFICATE-----
|
||||||
|
|
||||||
|
zkConfig:
|
||||||
|
serverPublic: ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
serverSecret: ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyzAA==
|
||||||
|
|
||||||
|
appConfig:
|
||||||
|
application: example
|
||||||
|
environment: example
|
||||||
|
configuration: example
|
||||||
|
|
||||||
|
remoteConfig:
|
||||||
|
authorizedTokens:
|
||||||
|
- # 1st authorized token
|
||||||
|
- # 2nd authorized token
|
||||||
|
- # ...
|
||||||
|
- # Nth authorized token
|
||||||
|
globalConfig: # keys and values that are given to clients on GET /v1/config
|
||||||
|
EXAMPLE_KEY: VALUE
|
||||||
|
|
||||||
|
paymentsService:
|
||||||
|
userAuthenticationTokenSharedSecret: 0000000f0000000f0000000f0000000f0000000f0000000f0000000f0000000f # hex-encoded 32-byte secret shared with MobileCoin services used to generate auth tokens for Signal users
|
||||||
|
fixerApiKey: unset
|
||||||
|
paymentCurrencies:
|
||||||
|
# list of symbols for supported currencies
|
||||||
|
- MOB
|
||||||
|
|
||||||
|
donation:
|
||||||
|
uri: donation.example.com # value
|
||||||
|
supportedCurrencies:
|
||||||
|
- # 1st supported currency
|
||||||
|
- # 2nd supported currency
|
||||||
|
- # ...
|
||||||
|
- # Nth supported currency
|
||||||
|
|
||||||
|
badges:
|
||||||
|
badges:
|
||||||
|
- id: TEST
|
||||||
|
category: other
|
||||||
|
sprites: # exactly 6
|
||||||
|
- sprite-1.png
|
||||||
|
- sprite-2.png
|
||||||
|
- sprite-3.png
|
||||||
|
- sprite-4.png
|
||||||
|
- sprite-5.png
|
||||||
|
- sprite-6.png
|
||||||
|
svg: example.svg
|
||||||
|
svgs:
|
||||||
|
- light: example-light.svg
|
||||||
|
dark: example-dark.svg
|
||||||
|
badgeIdsEnabledForAll:
|
||||||
|
- TEST
|
||||||
|
receiptLevels:
|
||||||
|
'1': TEST
|
||||||
|
|
||||||
|
subscription: # configuration for Stripe subscriptions
|
||||||
|
badgeGracePeriod: P15D
|
||||||
|
levels:
|
||||||
|
500:
|
||||||
|
badge: EXAMPLE
|
||||||
|
prices:
|
||||||
|
# list of ISO 4217 currency codes and amounts for the given badge level
|
||||||
|
xts:
|
||||||
|
amount: '10'
|
||||||
|
id: price_example # stripe ID
|
||||||
|
|
||||||
|
boost:
|
||||||
|
level: 1
|
||||||
|
expiration: P90D
|
||||||
|
badge: EXAMPLE
|
||||||
|
currencies:
|
||||||
|
# ISO 4217 currency codes and amounts in those currencies
|
||||||
|
xts:
|
||||||
|
- '1'
|
||||||
|
- '2'
|
||||||
|
- '4'
|
||||||
|
- '8'
|
||||||
|
- '20'
|
||||||
|
- '40'
|
||||||
|
|
||||||
|
gift:
|
||||||
|
level: 10
|
||||||
|
expiration: P90D
|
||||||
|
badge: EXAMPLE
|
||||||
|
currencies:
|
||||||
|
# ISO 4217 currency codes and amounts in those currencies
|
||||||
|
xts: '2'
|
||||||
|
|
||||||
|
registrationService:
|
||||||
|
host: registration.example.com
|
||||||
|
apiKey: EXAMPLE
|
||||||
|
registrationCaCertificate: | # Registration service TLS certificate trust root
|
||||||
|
-----BEGIN CERTIFICATE-----
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
ABCDEFGHIJKLMNOPQRSTUVWXYZ/0123456789+abcdefghijklmnopqrstuvwxyz
|
||||||
|
AAAAAAAAAAAAAAAAAAAA
|
||||||
|
-----END CERTIFICATE-----
|
||||||
609
service/pom.xml
Normal file
609
service/pom.xml
Normal file
@@ -0,0 +1,609 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||||
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<parent>
|
||||||
|
<artifactId>TextSecureServer</artifactId>
|
||||||
|
<groupId>org.whispersystems.textsecure</groupId>
|
||||||
|
<version>JGITVER</version>
|
||||||
|
</parent>
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
<artifactId>service</artifactId>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>jakarta.servlet</groupId>
|
||||||
|
<artifactId>jakarta.servlet-api</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>jakarta.validation</groupId>
|
||||||
|
<artifactId>jakarta.validation-api</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>jakarta.ws.rs</groupId>
|
||||||
|
<artifactId>jakarta.ws.rs-api</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.whispersystems.textsecure</groupId>
|
||||||
|
<artifactId>event-logger</artifactId>
|
||||||
|
<version>${project.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.whispersystems.textsecure</groupId>
|
||||||
|
<artifactId>redis-dispatch</artifactId>
|
||||||
|
<version>${project.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.whispersystems.textsecure</groupId>
|
||||||
|
<artifactId>websocket-resources</artifactId>
|
||||||
|
<version>${project.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.signal</groupId>
|
||||||
|
<artifactId>libsignal-server</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-auth</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-client</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-db</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-logging</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-metrics</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-util</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-servlets</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-lifecycle</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-jersey</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-jetty</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-validation</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-migrations</artifactId>
|
||||||
|
<scope>runtime</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.slf4j</groupId>
|
||||||
|
<artifactId>slf4j-api</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>ch.qos.logback</groupId>
|
||||||
|
<artifactId>logback-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>ch.qos.logback</groupId>
|
||||||
|
<artifactId>logback-access</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>ch.qos.logback</groupId>
|
||||||
|
<artifactId>logback-classic</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>net.logstash.logback</groupId>
|
||||||
|
<artifactId>logstash-logback-encoder</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard.metrics</groupId>
|
||||||
|
<artifactId>metrics-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard.metrics</groupId>
|
||||||
|
<artifactId>metrics-healthchecks</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard.metrics</groupId>
|
||||||
|
<artifactId>metrics-annotation</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.glassfish.jersey.core</groupId>
|
||||||
|
<artifactId>jersey-common</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.glassfish.jersey.core</groupId>
|
||||||
|
<artifactId>jersey-server</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.glassfish.jersey.core</groupId>
|
||||||
|
<artifactId>jersey-client</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.glassfish.jaxb</groupId>
|
||||||
|
<artifactId>jaxb-runtime</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.dropwizard</groupId>
|
||||||
|
<artifactId>dropwizard-testing</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
<exclusions>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>junit</groupId>
|
||||||
|
<artifactId>junit</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
</exclusions>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.eclipse.jetty.websocket</groupId>
|
||||||
|
<artifactId>websocket-api</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.eclipse.jetty</groupId>
|
||||||
|
<artifactId>jetty-servlets</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.commons</groupId>
|
||||||
|
<artifactId>commons-lang3</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>commons-codec</groupId>
|
||||||
|
<artifactId>commons-codec</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.commons</groupId>
|
||||||
|
<artifactId>commons-csv</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.firebase</groupId>
|
||||||
|
<artifactId>firebase-admin</artifactId>
|
||||||
|
<version>9.0.0</version>
|
||||||
|
|
||||||
|
<!-- firebase-admin has conflicting versions of these artifacts in its dependency tree; for firebase-admin
|
||||||
|
9.0.0, we'll need to depend directly on com.google.api-client:google-api-client:1.35.1 and
|
||||||
|
com.google.oauth-client:google-oauth-client:1.34.1 -->
|
||||||
|
<exclusions>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>com.google.api-client</groupId>
|
||||||
|
<artifactId>google-api-client</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
|
||||||
|
<exclusion>
|
||||||
|
<groupId>com.google.oauth-client</groupId>
|
||||||
|
<artifactId>google-oauth-client</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
</exclusions>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.api-client</groupId>
|
||||||
|
<artifactId>google-api-client</artifactId>
|
||||||
|
<version>1.35.1</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.oauth-client</groupId>
|
||||||
|
<artifactId>google-oauth-client</artifactId>
|
||||||
|
<version>1.34.1</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.code.findbugs</groupId>
|
||||||
|
<artifactId>jsr305</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.github.resilience4j</groupId>
|
||||||
|
<artifactId>resilience4j-circuitbreaker</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.github.resilience4j</groupId>
|
||||||
|
<artifactId>resilience4j-retry</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.grpc</groupId>
|
||||||
|
<artifactId>grpc-netty-shaded</artifactId>
|
||||||
|
<scope>runtime</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.grpc</groupId>
|
||||||
|
<artifactId>grpc-protobuf</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.grpc</groupId>
|
||||||
|
<artifactId>grpc-stub</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<!-- Needed for gRPC with Java 9+ -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.tomcat</groupId>
|
||||||
|
<artifactId>annotations-api</artifactId>
|
||||||
|
<scope>provided</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.micrometer</groupId>
|
||||||
|
<artifactId>micrometer-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.micrometer</groupId>
|
||||||
|
<artifactId>micrometer-registry-datadog</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.coursera</groupId>
|
||||||
|
<artifactId>dropwizard-metrics-datadog</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.fasterxml.jackson.core</groupId>
|
||||||
|
<artifactId>jackson-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.fasterxml.jackson.core</groupId>
|
||||||
|
<artifactId>jackson-annotations</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.fasterxml.jackson.core</groupId>
|
||||||
|
<artifactId>jackson-databind</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.fasterxml.jackson.dataformat</groupId>
|
||||||
|
<artifactId>jackson-dataformat-yaml</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.fasterxml.jackson.datatype</groupId>
|
||||||
|
<artifactId>jackson-datatype-jsr310</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.fasterxml.jackson.jaxrs</groupId>
|
||||||
|
<artifactId>jackson-jaxrs-json-provider</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>software.amazon.awssdk</groupId>
|
||||||
|
<artifactId>sts</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>software.amazon.awssdk</groupId>
|
||||||
|
<artifactId>s3</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>software.amazon.awssdk</groupId>
|
||||||
|
<artifactId>sqs</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>software.amazon.awssdk</groupId>
|
||||||
|
<artifactId>dynamodb</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>software.amazon.awssdk</groupId>
|
||||||
|
<artifactId>appconfig</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>software.amazon.awssdk</groupId>
|
||||||
|
<artifactId>appconfigdata</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.amazonaws</groupId>
|
||||||
|
<artifactId>aws-java-sdk-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.amazonaws</groupId>
|
||||||
|
<artifactId>dynamodb-lock-client</artifactId>
|
||||||
|
<version>1.1.0</version>
|
||||||
|
<exclusions>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>commons-logging</groupId>
|
||||||
|
<artifactId>commons-logging</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
</exclusions>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>redis.clients</groupId>
|
||||||
|
<artifactId>jedis</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.lettuce</groupId>
|
||||||
|
<artifactId>lettuce-core</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.eatthepath</groupId>
|
||||||
|
<artifactId>pushy</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.eatthepath</groupId>
|
||||||
|
<artifactId>pushy-dropwizard-metrics-listener</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.vdurmont</groupId>
|
||||||
|
<artifactId>semver4j</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.guava</groupId>
|
||||||
|
<artifactId>guava</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.protobuf</groupId>
|
||||||
|
<artifactId>protobuf-java</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.googlecode.libphonenumber</groupId>
|
||||||
|
<artifactId>libphonenumber</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>net.sourceforge.argparse4j</groupId>
|
||||||
|
<artifactId>argparse4j</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.glassfish.jersey.test-framework</groupId>
|
||||||
|
<artifactId>jersey-test-framework-core</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
<exclusions>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>junit</groupId>
|
||||||
|
<artifactId>junit</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
</exclusions>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.glassfish.jersey.test-framework.providers</groupId>
|
||||||
|
<artifactId>jersey-test-framework-provider-grizzly2</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
<exclusions>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>javax.servlet</groupId>
|
||||||
|
<artifactId>javax.servlet-api</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>junit</groupId>
|
||||||
|
<artifactId>junit</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
</exclusions>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.almworks.sqlite4java</groupId>
|
||||||
|
<artifactId>sqlite4java</artifactId>
|
||||||
|
<version>1.0.392</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.projectreactor</groupId>
|
||||||
|
<artifactId>reactor-core</artifactId>
|
||||||
|
<version>3.3.22.RELEASE</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.vavr</groupId>
|
||||||
|
<artifactId>vavr</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.junit.jupiter</groupId>
|
||||||
|
<artifactId>junit-jupiter-params</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.signal</groupId>
|
||||||
|
<artifactId>embedded-redis</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.fasterxml.uuid</groupId>
|
||||||
|
<artifactId>java-uuid-generator</artifactId>
|
||||||
|
<version>4.0.1</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.amazonaws</groupId>
|
||||||
|
<artifactId>DynamoDBLocal</artifactId>
|
||||||
|
<version>1.19.0</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.google.cloud</groupId>
|
||||||
|
<artifactId>google-cloud-recaptchaenterprise</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.stripe</groupId>
|
||||||
|
<artifactId>stripe-java</artifactId>
|
||||||
|
</dependency>
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
<profiles>
|
||||||
|
<profile>
|
||||||
|
<id>exclude-abusive-message-filter</id>
|
||||||
|
<build>
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-shade-plugin</artifactId>
|
||||||
|
<version>3.2.4</version>
|
||||||
|
<configuration>
|
||||||
|
<createDependencyReducedPom>true</createDependencyReducedPom>
|
||||||
|
<filters>
|
||||||
|
<filter>
|
||||||
|
<artifact>*:*</artifact>
|
||||||
|
<excludes>
|
||||||
|
<exclude>META-INF/*.SF</exclude>
|
||||||
|
<exclude>META-INF/*.DSA</exclude>
|
||||||
|
<exclude>META-INF/*.RSA</exclude>
|
||||||
|
</excludes>
|
||||||
|
</filter>
|
||||||
|
</filters>
|
||||||
|
</configuration>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<phase>package</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>shade</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<transformers>
|
||||||
|
<transformer implementation="org.apache.maven.plugins.shade.resource.ServicesResourceTransformer"/>
|
||||||
|
<transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
|
||||||
|
<mainClass>org.whispersystems.textsecuregcm.WhisperServerService</mainClass>
|
||||||
|
</transformer>
|
||||||
|
</transformers>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-assembly-plugin</artifactId>
|
||||||
|
<version>3.3.0</version>
|
||||||
|
<configuration>
|
||||||
|
<descriptors>
|
||||||
|
<descriptor>assembly.xml</descriptor>
|
||||||
|
</descriptors>
|
||||||
|
</configuration>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>make-assembly</id> <!-- this is used for inheritance merges -->
|
||||||
|
<phase>package</phase> <!-- bind to the packaging phase -->
|
||||||
|
<goals>
|
||||||
|
<goal>single</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.codehaus.mojo</groupId>
|
||||||
|
<artifactId>properties-maven-plugin</artifactId>
|
||||||
|
<version>1.0.0</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>read-deploy-configuration</id>
|
||||||
|
<phase>deploy</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>read-project-properties</goal>
|
||||||
|
</goals>
|
||||||
|
<configuration>
|
||||||
|
<files>${project.basedir}/config/deploy.properties</files>
|
||||||
|
</configuration>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.signal</groupId>
|
||||||
|
<artifactId>s3-upload-maven-plugin</artifactId>
|
||||||
|
<version>1.6-SNAPSHOT</version>
|
||||||
|
<configuration>
|
||||||
|
<source>${project.build.directory}/${project.build.finalName}-bin.tar.gz</source>
|
||||||
|
<bucketName>${deploy.bucketName}</bucketName>
|
||||||
|
<region>${deploy.bucketRegion}</region>
|
||||||
|
<destination>${project.build.finalName}-bin.tar.gz</destination>
|
||||||
|
</configuration>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>deploy-to-s3</id>
|
||||||
|
<phase>deploy</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>s3-upload</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
</profile>
|
||||||
|
</profiles>
|
||||||
|
|
||||||
|
<build>
|
||||||
|
<finalName>${project.parent.artifactId}-${project.version}</finalName>
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.codehaus.mojo</groupId>
|
||||||
|
<artifactId>templating-maven-plugin</artifactId>
|
||||||
|
<version>1.0.0</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>filter-src</id>
|
||||||
|
<goals>
|
||||||
|
<goal>filter-sources</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-jar-plugin</artifactId>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<goals>
|
||||||
|
<goal>test-jar</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.codehaus.mojo</groupId>
|
||||||
|
<artifactId>exec-maven-plugin</artifactId>
|
||||||
|
<version>3.0.0</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<id>check-all-service-config</id>
|
||||||
|
<phase>verify</phase>
|
||||||
|
<goals>
|
||||||
|
<goal>java</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
<configuration>
|
||||||
|
<mainClass>org.whispersystems.textsecuregcm.CheckServiceConfigurations</mainClass>
|
||||||
|
<classpathScope>test</classpathScope>
|
||||||
|
<arguments>
|
||||||
|
<argument>${project.basedir}/config</argument>
|
||||||
|
</arguments>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
</project>
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm;
|
||||||
|
|
||||||
|
public class WhisperServerVersion {
|
||||||
|
|
||||||
|
private static final String VERSION = "${project.version}";
|
||||||
|
|
||||||
|
public static String getServerVersion() {
|
||||||
|
return VERSION;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.signal.i18n;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.ResourceBundle;
|
||||||
|
import java.util.ResourceBundle.Control;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
|
||||||
|
public class HeaderControlledResourceBundleLookup {
|
||||||
|
|
||||||
|
private static final int MAX_LOCALES = 15;
|
||||||
|
|
||||||
|
private final ResourceBundleFactory resourceBundleFactory;
|
||||||
|
|
||||||
|
public HeaderControlledResourceBundleLookup() {
|
||||||
|
this(ResourceBundle::getBundle);
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public HeaderControlledResourceBundleLookup(
|
||||||
|
@Nonnull final ResourceBundleFactory resourceBundleFactory) {
|
||||||
|
this.resourceBundleFactory = Objects.requireNonNull(resourceBundleFactory);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nonnull
|
||||||
|
private List<Locale> getAcceptableLocales(final List<Locale> acceptableLanguages) {
|
||||||
|
return acceptableLanguages.stream().limit(MAX_LOCALES).distinct().collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nonnull
|
||||||
|
public ResourceBundle getResourceBundle(final String baseName, final List<Locale> acceptableLocales) {
|
||||||
|
final List<Locale> deduplicatedLocales = getAcceptableLocales(acceptableLocales);
|
||||||
|
final Locale desiredLocale = deduplicatedLocales.isEmpty() ? Locale.getDefault() : deduplicatedLocales.get(0);
|
||||||
|
// define a control with a fallback order as specified in the header
|
||||||
|
Control control = new Control() {
|
||||||
|
@Override
|
||||||
|
public List<String> getFormats(final String baseName) {
|
||||||
|
Objects.requireNonNull(baseName);
|
||||||
|
return Control.FORMAT_PROPERTIES;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Locale getFallbackLocale(final String baseName, final Locale locale) {
|
||||||
|
Objects.requireNonNull(baseName);
|
||||||
|
if (locale.equals(Locale.getDefault())) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
final int localeIndex = deduplicatedLocales.indexOf(locale);
|
||||||
|
if (localeIndex < 0 || localeIndex >= deduplicatedLocales.size() - 1) {
|
||||||
|
return Locale.getDefault();
|
||||||
|
}
|
||||||
|
// [0, deduplicatedLocales.size() - 2] is now the possible range for localeIndex
|
||||||
|
return deduplicatedLocales.get(localeIndex + 1);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return resourceBundleFactory.createBundle(baseName, desiredLocale, control);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.signal.i18n;
|
||||||
|
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.ResourceBundle;
|
||||||
|
|
||||||
|
public interface ResourceBundleFactory {
|
||||||
|
ResourceBundle createBundle(String baseName, Locale locale, ResourceBundle.Control control);
|
||||||
|
}
|
||||||
@@ -0,0 +1,457 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import io.dropwizard.Configuration;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import javax.validation.Valid;
|
||||||
|
import javax.validation.constraints.NotNull;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.AbusiveMessageFilterConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.AccountDatabaseCrawlerConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.AdminEventLoggingConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.ApnConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.AppConfigConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.AwsAttachmentsConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.BadgesConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.BoostConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.CdnConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.DatadogConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.DirectoryConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.DirectoryV2Configuration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.DonationConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.DynamoDbClientConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.DynamoDbTables;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.FcmConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.GcpAttachmentsConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.GiftConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.MaxDeviceConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.MessageCacheConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.PaymentsServiceConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.RateLimitsConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.RecaptchaConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.RedisClusterConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.RedisConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.RegistrationServiceConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.RemoteConfigConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.ReportMessageConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.SecureBackupServiceConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.SecureStorageServiceConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.StripeConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.SubscriptionConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.TestDeviceConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.TwilioConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.UnidentifiedDeliveryConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.UsernameConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.VoiceVerificationConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.ZkConfig;
|
||||||
|
import org.whispersystems.websocket.configuration.WebSocketConfiguration;
|
||||||
|
|
||||||
|
/** @noinspection MismatchedQueryAndUpdateOfCollection, WeakerAccess */
|
||||||
|
public class WhisperServerConfiguration extends Configuration {
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private AdminEventLoggingConfiguration adminEventLoggingConfiguration;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private StripeConfiguration stripe;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private DynamoDbClientConfiguration dynamoDbClientConfiguration;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private DynamoDbTables dynamoDbTables;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private TwilioConfiguration twilio;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private AwsAttachmentsConfiguration awsAttachments;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private GcpAttachmentsConfiguration gcpAttachments;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private CdnConfiguration cdn;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private DatadogConfiguration datadog;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private RedisClusterConfiguration cacheCluster;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private RedisConfiguration pubsub;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private RedisClusterConfiguration metricsCluster;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private DirectoryConfiguration directory;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private DirectoryV2Configuration directoryV2;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private AccountDatabaseCrawlerConfiguration accountDatabaseCrawler;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private RedisClusterConfiguration pushSchedulerCluster;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private RedisClusterConfiguration rateLimitersCluster;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private MessageCacheConfiguration messageCache;
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private RedisClusterConfiguration clientPresenceCluster;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private List<TestDeviceConfiguration> testDevices = new LinkedList<>();
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private List<MaxDeviceConfiguration> maxDevices = new LinkedList<>();
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private RateLimitsConfiguration limits = new RateLimitsConfiguration();
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private WebSocketConfiguration webSocket = new WebSocketConfiguration();
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private FcmConfiguration fcm;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private ApnConfiguration apn;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private UnidentifiedDeliveryConfiguration unidentifiedDelivery;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private VoiceVerificationConfiguration voiceVerification;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private RecaptchaConfiguration recaptcha;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private SecureStorageServiceConfiguration storageService;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private SecureBackupServiceConfiguration backupService;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private PaymentsServiceConfiguration paymentsService;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private ZkConfig zkConfig;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private RemoteConfigConfiguration remoteConfig;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private AppConfigConfiguration appConfig;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private DonationConfiguration donation;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private BadgesConfiguration badges;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
private SubscriptionConfiguration subscription;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
private BoostConfiguration boost;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
private GiftConfiguration gift;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private ReportMessageConfiguration reportMessage = new ReportMessageConfiguration();
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private UsernameConfiguration username = new UsernameConfiguration();
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@JsonProperty
|
||||||
|
private AbusiveMessageFilterConfiguration abusiveMessageFilter;
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private RegistrationServiceConfiguration registrationService;
|
||||||
|
|
||||||
|
public AdminEventLoggingConfiguration getAdminEventLoggingConfiguration() {
|
||||||
|
return adminEventLoggingConfiguration;
|
||||||
|
}
|
||||||
|
|
||||||
|
public StripeConfiguration getStripe() {
|
||||||
|
return stripe;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DynamoDbClientConfiguration getDynamoDbClientConfiguration() {
|
||||||
|
return dynamoDbClientConfiguration;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DynamoDbTables getDynamoDbTables() {
|
||||||
|
return dynamoDbTables;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RecaptchaConfiguration getRecaptchaConfiguration() {
|
||||||
|
return recaptcha;
|
||||||
|
}
|
||||||
|
|
||||||
|
public VoiceVerificationConfiguration getVoiceVerificationConfiguration() {
|
||||||
|
return voiceVerification;
|
||||||
|
}
|
||||||
|
|
||||||
|
public WebSocketConfiguration getWebSocketConfiguration() {
|
||||||
|
return webSocket;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TwilioConfiguration getTwilioConfiguration() {
|
||||||
|
return twilio;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AwsAttachmentsConfiguration getAwsAttachmentsConfiguration() {
|
||||||
|
return awsAttachments;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GcpAttachmentsConfiguration getGcpAttachmentsConfiguration() {
|
||||||
|
return gcpAttachments;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RedisClusterConfiguration getCacheClusterConfiguration() {
|
||||||
|
return cacheCluster;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RedisConfiguration getPubsubCacheConfiguration() {
|
||||||
|
return pubsub;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RedisClusterConfiguration getMetricsClusterConfiguration() {
|
||||||
|
return metricsCluster;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DirectoryConfiguration getDirectoryConfiguration() {
|
||||||
|
return directory;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DirectoryV2Configuration getDirectoryV2Configuration() {
|
||||||
|
return directoryV2;
|
||||||
|
}
|
||||||
|
|
||||||
|
public SecureStorageServiceConfiguration getSecureStorageServiceConfiguration() {
|
||||||
|
return storageService;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AccountDatabaseCrawlerConfiguration getAccountDatabaseCrawlerConfiguration() {
|
||||||
|
return accountDatabaseCrawler;
|
||||||
|
}
|
||||||
|
|
||||||
|
public MessageCacheConfiguration getMessageCacheConfiguration() {
|
||||||
|
return messageCache;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RedisClusterConfiguration getClientPresenceClusterConfiguration() {
|
||||||
|
return clientPresenceCluster;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RedisClusterConfiguration getPushSchedulerCluster() {
|
||||||
|
return pushSchedulerCluster;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RedisClusterConfiguration getRateLimitersCluster() {
|
||||||
|
return rateLimitersCluster;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RateLimitsConfiguration getLimitsConfiguration() {
|
||||||
|
return limits;
|
||||||
|
}
|
||||||
|
|
||||||
|
public FcmConfiguration getFcmConfiguration() {
|
||||||
|
return fcm;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ApnConfiguration getApnConfiguration() {
|
||||||
|
return apn;
|
||||||
|
}
|
||||||
|
|
||||||
|
public CdnConfiguration getCdnConfiguration() {
|
||||||
|
return cdn;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DatadogConfiguration getDatadogConfiguration() {
|
||||||
|
return datadog;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UnidentifiedDeliveryConfiguration getDeliveryCertificate() {
|
||||||
|
return unidentifiedDelivery;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, Integer> getTestDevices() {
|
||||||
|
Map<String, Integer> results = new HashMap<>();
|
||||||
|
|
||||||
|
for (TestDeviceConfiguration testDeviceConfiguration : testDevices) {
|
||||||
|
results.put(testDeviceConfiguration.getNumber(),
|
||||||
|
testDeviceConfiguration.getCode());
|
||||||
|
}
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, Integer> getMaxDevices() {
|
||||||
|
Map<String, Integer> results = new HashMap<>();
|
||||||
|
|
||||||
|
for (MaxDeviceConfiguration maxDeviceConfiguration : maxDevices) {
|
||||||
|
results.put(maxDeviceConfiguration.getNumber(),
|
||||||
|
maxDeviceConfiguration.getCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
public SecureBackupServiceConfiguration getSecureBackupServiceConfiguration() {
|
||||||
|
return backupService;
|
||||||
|
}
|
||||||
|
|
||||||
|
public PaymentsServiceConfiguration getPaymentsServiceConfiguration() {
|
||||||
|
return paymentsService;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ZkConfig getZkConfig() {
|
||||||
|
return zkConfig;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RemoteConfigConfiguration getRemoteConfigConfiguration() {
|
||||||
|
return remoteConfig;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AppConfigConfiguration getAppConfig() {
|
||||||
|
return appConfig;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DonationConfiguration getDonationConfiguration() {
|
||||||
|
return donation;
|
||||||
|
}
|
||||||
|
|
||||||
|
public BadgesConfiguration getBadges() {
|
||||||
|
return badges;
|
||||||
|
}
|
||||||
|
|
||||||
|
public SubscriptionConfiguration getSubscription() {
|
||||||
|
return subscription;
|
||||||
|
}
|
||||||
|
|
||||||
|
public BoostConfiguration getBoost() {
|
||||||
|
return boost;
|
||||||
|
}
|
||||||
|
|
||||||
|
public GiftConfiguration getGift() {
|
||||||
|
return gift;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ReportMessageConfiguration getReportMessageConfiguration() {
|
||||||
|
return reportMessage;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AbusiveMessageFilterConfiguration getAbusiveMessageFilterConfiguration() {
|
||||||
|
return abusiveMessageFilter;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UsernameConfiguration getUsername() {
|
||||||
|
return username;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RegistrationServiceConfiguration getRegistrationServiceConfiguration() {
|
||||||
|
return registrationService;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,801 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2022 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm;
|
||||||
|
|
||||||
|
import static com.codahale.metrics.MetricRegistry.name;
|
||||||
|
|
||||||
|
import com.amazonaws.ClientConfiguration;
|
||||||
|
import com.amazonaws.auth.InstanceProfileCredentialsProvider;
|
||||||
|
import com.amazonaws.services.dynamodbv2.AmazonDynamoDB;
|
||||||
|
import com.amazonaws.services.dynamodbv2.AmazonDynamoDBClientBuilder;
|
||||||
|
import com.codahale.metrics.SharedMetricRegistries;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonAutoDetect;
|
||||||
|
import com.fasterxml.jackson.annotation.PropertyAccessor;
|
||||||
|
import com.fasterxml.jackson.databind.DeserializationFeature;
|
||||||
|
import com.google.auth.oauth2.GoogleCredentials;
|
||||||
|
import com.google.cloud.logging.LoggingOptions;
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
import io.dropwizard.Application;
|
||||||
|
import io.dropwizard.auth.AuthFilter;
|
||||||
|
import io.dropwizard.auth.PolymorphicAuthDynamicFeature;
|
||||||
|
import io.dropwizard.auth.PolymorphicAuthValueFactoryProvider;
|
||||||
|
import io.dropwizard.auth.basic.BasicCredentialAuthFilter;
|
||||||
|
import io.dropwizard.auth.basic.BasicCredentials;
|
||||||
|
import io.dropwizard.setup.Bootstrap;
|
||||||
|
import io.dropwizard.setup.Environment;
|
||||||
|
import io.lettuce.core.metrics.MicrometerCommandLatencyRecorder;
|
||||||
|
import io.lettuce.core.metrics.MicrometerOptions;
|
||||||
|
import io.lettuce.core.resource.ClientResources;
|
||||||
|
import io.micrometer.core.instrument.Meter.Id;
|
||||||
|
import io.micrometer.core.instrument.Metrics;
|
||||||
|
import io.micrometer.core.instrument.Tags;
|
||||||
|
import io.micrometer.core.instrument.config.MeterFilter;
|
||||||
|
import io.micrometer.core.instrument.distribution.DistributionStatisticConfig;
|
||||||
|
import io.micrometer.datadog.DatadogMeterRegistry;
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.net.http.HttpClient;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.time.Clock;
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.EnumSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.ServiceLoader;
|
||||||
|
import java.util.concurrent.ArrayBlockingQueue;
|
||||||
|
import java.util.concurrent.BlockingQueue;
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.LinkedBlockingQueue;
|
||||||
|
import java.util.concurrent.ScheduledExecutorService;
|
||||||
|
import java.util.concurrent.ThreadPoolExecutor;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import javax.servlet.DispatcherType;
|
||||||
|
import javax.servlet.FilterRegistration;
|
||||||
|
import javax.servlet.ServletRegistration;
|
||||||
|
import org.eclipse.jetty.servlets.CrossOriginFilter;
|
||||||
|
import org.glassfish.jersey.server.ServerProperties;
|
||||||
|
import org.signal.event.AdminEventLogger;
|
||||||
|
import org.signal.event.GoogleCloudAdminEventLogger;
|
||||||
|
import org.signal.i18n.HeaderControlledResourceBundleLookup;
|
||||||
|
import org.signal.libsignal.zkgroup.ServerSecretParams;
|
||||||
|
import org.signal.libsignal.zkgroup.auth.ServerZkAuthOperations;
|
||||||
|
import org.signal.libsignal.zkgroup.profiles.ServerZkProfileOperations;
|
||||||
|
import org.signal.libsignal.zkgroup.receipts.ReceiptCredentialPresentation;
|
||||||
|
import org.signal.libsignal.zkgroup.receipts.ServerZkReceiptOperations;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.whispersystems.dispatch.DispatchManager;
|
||||||
|
import org.whispersystems.textsecuregcm.abuse.AbusiveMessageFilter;
|
||||||
|
import org.whispersystems.textsecuregcm.abuse.FilterAbusiveMessages;
|
||||||
|
import org.whispersystems.textsecuregcm.abuse.RateLimitChallengeListener;
|
||||||
|
import org.whispersystems.textsecuregcm.auth.AccountAuthenticator;
|
||||||
|
import org.whispersystems.textsecuregcm.auth.AuthenticatedAccount;
|
||||||
|
import org.whispersystems.textsecuregcm.auth.CertificateGenerator;
|
||||||
|
import org.whispersystems.textsecuregcm.auth.DisabledPermittedAccountAuthenticator;
|
||||||
|
import org.whispersystems.textsecuregcm.auth.DisabledPermittedAuthenticatedAccount;
|
||||||
|
import org.whispersystems.textsecuregcm.auth.ExternalServiceCredentialGenerator;
|
||||||
|
import org.whispersystems.textsecuregcm.auth.TurnTokenGenerator;
|
||||||
|
import org.whispersystems.textsecuregcm.auth.WebsocketRefreshApplicationEventListener;
|
||||||
|
import org.whispersystems.textsecuregcm.badges.ConfiguredProfileBadgeConverter;
|
||||||
|
import org.whispersystems.textsecuregcm.badges.ResourceBundleLevelTranslator;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.DirectoryServerConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.dynamic.DynamicConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.AccountController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.AttachmentControllerV2;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.AttachmentControllerV3;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.CertificateController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.ChallengeController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.DeviceController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.DirectoryController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.DirectoryV2Controller;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.DonationController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.KeepAliveController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.KeysController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.MessageController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.PaymentsController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.ProfileController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.ProvisioningController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.RemoteConfigController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.SecureBackupController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.SecureStorageController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.StickerController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.SubscriptionController;
|
||||||
|
import org.whispersystems.textsecuregcm.controllers.VoiceVerificationController;
|
||||||
|
import org.whispersystems.textsecuregcm.currency.CurrencyConversionManager;
|
||||||
|
import org.whispersystems.textsecuregcm.currency.FixerClient;
|
||||||
|
import org.whispersystems.textsecuregcm.currency.FtxClient;
|
||||||
|
import org.whispersystems.textsecuregcm.experiment.ExperimentEnrollmentManager;
|
||||||
|
import org.whispersystems.textsecuregcm.filters.ContentLengthFilter;
|
||||||
|
import org.whispersystems.textsecuregcm.filters.RemoteDeprecationFilter;
|
||||||
|
import org.whispersystems.textsecuregcm.filters.TimestampResponseFilter;
|
||||||
|
import org.whispersystems.textsecuregcm.limits.DynamicRateLimiters;
|
||||||
|
import org.whispersystems.textsecuregcm.limits.PushChallengeManager;
|
||||||
|
import org.whispersystems.textsecuregcm.limits.RateLimitChallengeManager;
|
||||||
|
import org.whispersystems.textsecuregcm.limits.RateLimitChallengeOptionManager;
|
||||||
|
import org.whispersystems.textsecuregcm.limits.RateLimiters;
|
||||||
|
import org.whispersystems.textsecuregcm.mappers.CompletionExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.mappers.DeviceLimitExceededExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.mappers.IOExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.mappers.ImpossiblePhoneNumberExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.mappers.InvalidWebsocketAddressExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.mappers.NonNormalizedPhoneNumberExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.mappers.RateLimitExceededExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.mappers.ServerRejectedExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.ApplicationShutdownMonitor;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.BufferPoolGauges;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.CpuUsageGauge;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.FileDescriptorGauge;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.FreeMemoryGauge;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.GarbageCollectionGauges;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.LettuceMetricsMeterFilter;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.MaxFileDescriptorGauge;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.MetricsApplicationEventListener;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.MetricsRequestEventListener;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.MicrometerRegistryManager;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.NetworkReceivedGauge;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.NetworkSentGauge;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.OperatingSystemMemoryGauge;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.ReportedMessageMetricsListener;
|
||||||
|
import org.whispersystems.textsecuregcm.metrics.TrafficSource;
|
||||||
|
import org.whispersystems.textsecuregcm.providers.MultiRecipientMessageProvider;
|
||||||
|
import org.whispersystems.textsecuregcm.providers.RedisClientFactory;
|
||||||
|
import org.whispersystems.textsecuregcm.providers.RedisClusterHealthCheck;
|
||||||
|
import org.whispersystems.textsecuregcm.push.APNSender;
|
||||||
|
import org.whispersystems.textsecuregcm.push.ApnPushNotificationScheduler;
|
||||||
|
import org.whispersystems.textsecuregcm.push.ClientPresenceManager;
|
||||||
|
import org.whispersystems.textsecuregcm.push.FcmSender;
|
||||||
|
import org.whispersystems.textsecuregcm.push.MessageSender;
|
||||||
|
import org.whispersystems.textsecuregcm.push.ProvisioningManager;
|
||||||
|
import org.whispersystems.textsecuregcm.push.PushLatencyManager;
|
||||||
|
import org.whispersystems.textsecuregcm.push.PushNotificationManager;
|
||||||
|
import org.whispersystems.textsecuregcm.push.ReceiptSender;
|
||||||
|
import org.whispersystems.textsecuregcm.recaptcha.RecaptchaClient;
|
||||||
|
import org.whispersystems.textsecuregcm.redis.ConnectionEventLogger;
|
||||||
|
import org.whispersystems.textsecuregcm.redis.FaultTolerantRedisCluster;
|
||||||
|
import org.whispersystems.textsecuregcm.redis.ReplicatedJedisPool;
|
||||||
|
import org.whispersystems.textsecuregcm.registration.RegistrationServiceClient;
|
||||||
|
import org.whispersystems.textsecuregcm.s3.PolicySigner;
|
||||||
|
import org.whispersystems.textsecuregcm.s3.PostPolicyGenerator;
|
||||||
|
import org.whispersystems.textsecuregcm.securebackup.SecureBackupClient;
|
||||||
|
import org.whispersystems.textsecuregcm.securestorage.SecureStorageClient;
|
||||||
|
import org.whispersystems.textsecuregcm.sms.SmsSender;
|
||||||
|
import org.whispersystems.textsecuregcm.sms.TwilioSmsSender;
|
||||||
|
import org.whispersystems.textsecuregcm.sms.TwilioVerifyExperimentEnrollmentManager;
|
||||||
|
import org.whispersystems.textsecuregcm.sqs.DirectoryQueue;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AbusiveHostRules;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountCleaner;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountDatabaseCrawler;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountDatabaseCrawlerCache;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountDatabaseCrawlerListener;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Accounts;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountsManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.ChangeNumberManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.ContactDiscoveryWriter;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.DeletedAccounts;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.DeletedAccountsDirectoryReconciler;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.DeletedAccountsManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.DeletedAccountsTableCrawler;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.DirectoryReconciler;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.DirectoryReconciliationClient;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.DynamicConfigurationManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.IssuedReceiptsManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Keys;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.MessagePersister;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.MessagesCache;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.MessagesDynamoDb;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.MessagesManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.NonNormalizedAccountCrawlerListener;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.PhoneNumberIdentifiers;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Profiles;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.ProfilesManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.ProhibitedUsernames;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.PubSubManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.PushChallengeDynamoDb;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.PushFeedbackProcessor;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.RedeemedReceiptsManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.RemoteConfigs;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.RemoteConfigsManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.ReportMessageDynamoDb;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.ReportMessageManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.StoredVerificationCodeManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.SubscriptionManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.VerificationCodeStore;
|
||||||
|
import org.whispersystems.textsecuregcm.subscriptions.StripeManager;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Constants;
|
||||||
|
import org.whispersystems.textsecuregcm.util.DynamoDbFromConfig;
|
||||||
|
import org.whispersystems.textsecuregcm.util.HostnameUtil;
|
||||||
|
import org.whispersystems.textsecuregcm.util.UsernameGenerator;
|
||||||
|
import org.whispersystems.textsecuregcm.util.logging.LoggingUnhandledExceptionMapper;
|
||||||
|
import org.whispersystems.textsecuregcm.util.logging.UncaughtExceptionHandler;
|
||||||
|
import org.whispersystems.textsecuregcm.websocket.AuthenticatedConnectListener;
|
||||||
|
import org.whispersystems.textsecuregcm.websocket.ProvisioningConnectListener;
|
||||||
|
import org.whispersystems.textsecuregcm.websocket.WebSocketAccountAuthenticator;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.AssignUsernameCommand;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.CertificateCommand;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.CheckDynamicConfigurationCommand;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.DeleteUserCommand;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.ReserveUsernameCommand;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.ServerVersionCommand;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.SetCrawlerAccelerationTask;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.SetRequestLoggingEnabledTask;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.SetUserDiscoverabilityCommand;
|
||||||
|
import org.whispersystems.textsecuregcm.workers.ZkParamsCommand;
|
||||||
|
import org.whispersystems.websocket.WebSocketResourceProviderFactory;
|
||||||
|
import org.whispersystems.websocket.setup.WebSocketEnvironment;
|
||||||
|
import software.amazon.awssdk.auth.credentials.AwsBasicCredentials;
|
||||||
|
import software.amazon.awssdk.auth.credentials.StaticCredentialsProvider;
|
||||||
|
import software.amazon.awssdk.regions.Region;
|
||||||
|
import software.amazon.awssdk.services.dynamodb.DynamoDbAsyncClient;
|
||||||
|
import software.amazon.awssdk.services.dynamodb.DynamoDbClient;
|
||||||
|
import software.amazon.awssdk.services.s3.S3Client;
|
||||||
|
|
||||||
|
public class WhisperServerService extends Application<WhisperServerConfiguration> {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(WhisperServerService.class);
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void initialize(Bootstrap<WhisperServerConfiguration> bootstrap) {
|
||||||
|
bootstrap.addCommand(new DeleteUserCommand());
|
||||||
|
bootstrap.addCommand(new CertificateCommand());
|
||||||
|
bootstrap.addCommand(new ZkParamsCommand());
|
||||||
|
bootstrap.addCommand(new ServerVersionCommand());
|
||||||
|
bootstrap.addCommand(new CheckDynamicConfigurationCommand());
|
||||||
|
bootstrap.addCommand(new SetUserDiscoverabilityCommand());
|
||||||
|
bootstrap.addCommand(new ReserveUsernameCommand());
|
||||||
|
bootstrap.addCommand(new AssignUsernameCommand());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return "whisper-server";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run(WhisperServerConfiguration config, Environment environment) throws Exception {
|
||||||
|
final Clock clock = Clock.systemUTC();
|
||||||
|
final int availableProcessors = Runtime.getRuntime().availableProcessors();
|
||||||
|
|
||||||
|
UncaughtExceptionHandler.register();
|
||||||
|
|
||||||
|
SharedMetricRegistries.add(Constants.METRICS_NAME, environment.metrics());
|
||||||
|
|
||||||
|
final DistributionStatisticConfig defaultDistributionStatisticConfig = DistributionStatisticConfig.builder()
|
||||||
|
.percentiles(.75, .95, .99, .999)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
{
|
||||||
|
final DatadogMeterRegistry datadogMeterRegistry = new DatadogMeterRegistry(
|
||||||
|
config.getDatadogConfiguration(), io.micrometer.core.instrument.Clock.SYSTEM);
|
||||||
|
|
||||||
|
datadogMeterRegistry.config().commonTags(
|
||||||
|
Tags.of(
|
||||||
|
"service", "chat",
|
||||||
|
"host", HostnameUtil.getLocalHostname(),
|
||||||
|
"version", WhisperServerVersion.getServerVersion(),
|
||||||
|
"env", config.getDatadogConfiguration().getEnvironment()))
|
||||||
|
.meterFilter(MeterFilter.denyNameStartsWith(MetricsRequestEventListener.REQUEST_COUNTER_NAME))
|
||||||
|
.meterFilter(MeterFilter.denyNameStartsWith(MetricsRequestEventListener.ANDROID_REQUEST_COUNTER_NAME))
|
||||||
|
.meterFilter(MeterFilter.denyNameStartsWith(MetricsRequestEventListener.DESKTOP_REQUEST_COUNTER_NAME))
|
||||||
|
.meterFilter(MeterFilter.denyNameStartsWith(MetricsRequestEventListener.IOS_REQUEST_COUNTER_NAME))
|
||||||
|
.meterFilter(new LettuceMetricsMeterFilter())
|
||||||
|
.meterFilter(new MeterFilter() {
|
||||||
|
@Override
|
||||||
|
public DistributionStatisticConfig configure(final Id id, final DistributionStatisticConfig config) {
|
||||||
|
return defaultDistributionStatisticConfig.merge(config);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Metrics.addRegistry(datadogMeterRegistry);
|
||||||
|
}
|
||||||
|
|
||||||
|
environment.lifecycle().manage(new MicrometerRegistryManager(Metrics.globalRegistry));
|
||||||
|
|
||||||
|
environment.getObjectMapper().configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
|
||||||
|
environment.getObjectMapper().setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.NONE);
|
||||||
|
environment.getObjectMapper().setVisibility(PropertyAccessor.FIELD, JsonAutoDetect.Visibility.ANY);
|
||||||
|
|
||||||
|
HeaderControlledResourceBundleLookup headerControlledResourceBundleLookup =
|
||||||
|
new HeaderControlledResourceBundleLookup();
|
||||||
|
ConfiguredProfileBadgeConverter profileBadgeConverter = new ConfiguredProfileBadgeConverter(
|
||||||
|
clock, config.getBadges(), headerControlledResourceBundleLookup);
|
||||||
|
ResourceBundleLevelTranslator resourceBundleLevelTranslator = new ResourceBundleLevelTranslator(
|
||||||
|
headerControlledResourceBundleLookup);
|
||||||
|
|
||||||
|
DynamoDbAsyncClient dynamoDbAsyncClient = DynamoDbFromConfig.asyncClient(
|
||||||
|
config.getDynamoDbClientConfiguration(),
|
||||||
|
software.amazon.awssdk.auth.credentials.InstanceProfileCredentialsProvider.create());
|
||||||
|
|
||||||
|
DynamoDbClient dynamoDbClient = DynamoDbFromConfig.client(
|
||||||
|
config.getDynamoDbClientConfiguration(),
|
||||||
|
software.amazon.awssdk.auth.credentials.InstanceProfileCredentialsProvider.create());
|
||||||
|
|
||||||
|
AmazonDynamoDB deletedAccountsLockDynamoDbClient = AmazonDynamoDBClientBuilder.standard()
|
||||||
|
.withRegion(config.getDynamoDbClientConfiguration().getRegion())
|
||||||
|
.withClientConfiguration(new ClientConfiguration().withClientExecutionTimeout(
|
||||||
|
((int) config.getDynamoDbClientConfiguration().getClientExecutionTimeout().toMillis()))
|
||||||
|
.withRequestTimeout(
|
||||||
|
(int) config.getDynamoDbClientConfiguration().getClientRequestTimeout().toMillis()))
|
||||||
|
.withCredentials(InstanceProfileCredentialsProvider.getInstance())
|
||||||
|
.build();
|
||||||
|
|
||||||
|
DeletedAccounts deletedAccounts = new DeletedAccounts(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getDeletedAccounts().getTableName(),
|
||||||
|
config.getDynamoDbTables().getDeletedAccounts().getNeedsReconciliationIndexName());
|
||||||
|
|
||||||
|
DynamicConfigurationManager<DynamicConfiguration> dynamicConfigurationManager =
|
||||||
|
new DynamicConfigurationManager<>(config.getAppConfig().getApplication(),
|
||||||
|
config.getAppConfig().getEnvironment(),
|
||||||
|
config.getAppConfig().getConfigurationName(),
|
||||||
|
DynamicConfiguration.class);
|
||||||
|
|
||||||
|
Accounts accounts = new Accounts(dynamicConfigurationManager,
|
||||||
|
dynamoDbClient,
|
||||||
|
dynamoDbAsyncClient,
|
||||||
|
config.getDynamoDbTables().getAccounts().getTableName(),
|
||||||
|
config.getDynamoDbTables().getAccounts().getPhoneNumberTableName(),
|
||||||
|
config.getDynamoDbTables().getAccounts().getPhoneNumberIdentifierTableName(),
|
||||||
|
config.getDynamoDbTables().getAccounts().getUsernamesTableName(),
|
||||||
|
config.getDynamoDbTables().getAccounts().getScanPageSize());
|
||||||
|
PhoneNumberIdentifiers phoneNumberIdentifiers = new PhoneNumberIdentifiers(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getPhoneNumberIdentifiers().getTableName());
|
||||||
|
ProhibitedUsernames prohibitedUsernames = new ProhibitedUsernames(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getReservedUsernames().getTableName());
|
||||||
|
Profiles profiles = new Profiles(dynamoDbClient, dynamoDbAsyncClient,
|
||||||
|
config.getDynamoDbTables().getProfiles().getTableName());
|
||||||
|
Keys keys = new Keys(dynamoDbClient, config.getDynamoDbTables().getKeys().getTableName());
|
||||||
|
MessagesDynamoDb messagesDynamoDb = new MessagesDynamoDb(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getMessages().getTableName(),
|
||||||
|
config.getDynamoDbTables().getMessages().getExpiration());
|
||||||
|
RemoteConfigs remoteConfigs = new RemoteConfigs(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getRemoteConfig().getTableName());
|
||||||
|
PushChallengeDynamoDb pushChallengeDynamoDb = new PushChallengeDynamoDb(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getPushChallenge().getTableName());
|
||||||
|
ReportMessageDynamoDb reportMessageDynamoDb = new ReportMessageDynamoDb(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getReportMessage().getTableName(),
|
||||||
|
config.getReportMessageConfiguration().getReportTtl());
|
||||||
|
VerificationCodeStore pendingAccounts = new VerificationCodeStore(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getPendingAccounts().getTableName());
|
||||||
|
VerificationCodeStore pendingDevices = new VerificationCodeStore(dynamoDbClient,
|
||||||
|
config.getDynamoDbTables().getPendingDevices().getTableName());
|
||||||
|
|
||||||
|
RedisClientFactory pubSubClientFactory = new RedisClientFactory("pubsub_cache", config.getPubsubCacheConfiguration().getUrl(), config.getPubsubCacheConfiguration().getReplicaUrls(), config.getPubsubCacheConfiguration().getCircuitBreakerConfiguration());
|
||||||
|
ReplicatedJedisPool pubsubClient = pubSubClientFactory.getRedisClientPool();
|
||||||
|
|
||||||
|
MicrometerOptions options = MicrometerOptions.builder().build();
|
||||||
|
ClientResources redisClientResources = ClientResources.builder()
|
||||||
|
.commandLatencyRecorder(new MicrometerCommandLatencyRecorder(Metrics.globalRegistry, options)).build();
|
||||||
|
ConnectionEventLogger.logConnectionEvents(redisClientResources);
|
||||||
|
|
||||||
|
FaultTolerantRedisCluster cacheCluster = new FaultTolerantRedisCluster("main_cache_cluster", config.getCacheClusterConfiguration(), redisClientResources);
|
||||||
|
FaultTolerantRedisCluster messagesCluster = new FaultTolerantRedisCluster("messages_cluster", config.getMessageCacheConfiguration().getRedisClusterConfiguration(), redisClientResources);
|
||||||
|
FaultTolerantRedisCluster clientPresenceCluster = new FaultTolerantRedisCluster("client_presence_cluster", config.getClientPresenceClusterConfiguration(), redisClientResources);
|
||||||
|
FaultTolerantRedisCluster metricsCluster = new FaultTolerantRedisCluster("metrics_cluster", config.getMetricsClusterConfiguration(), redisClientResources);
|
||||||
|
FaultTolerantRedisCluster pushSchedulerCluster = new FaultTolerantRedisCluster("push_scheduler", config.getPushSchedulerCluster(), redisClientResources);
|
||||||
|
FaultTolerantRedisCluster rateLimitersCluster = new FaultTolerantRedisCluster("rate_limiters", config.getRateLimitersCluster(), redisClientResources);
|
||||||
|
|
||||||
|
BlockingQueue<Runnable> keyspaceNotificationDispatchQueue = new ArrayBlockingQueue<>(10_000);
|
||||||
|
Metrics.gaugeCollectionSize(name(getClass(), "keyspaceNotificationDispatchQueueSize"), Collections.emptyList(), keyspaceNotificationDispatchQueue);
|
||||||
|
final ArrayBlockingQueue<Runnable> receiptSenderQueue = new ArrayBlockingQueue<>(10_000);
|
||||||
|
Metrics.gaugeCollectionSize(name(getClass(), "receiptSenderQueue"), Collections.emptyList(), receiptSenderQueue);
|
||||||
|
|
||||||
|
final BlockingQueue<Runnable> fcmSenderQueue = new LinkedBlockingQueue<>();
|
||||||
|
Metrics.gaugeCollectionSize(name(getClass(), "fcmSenderQueue"), Collections.emptyList(), fcmSenderQueue);
|
||||||
|
|
||||||
|
ScheduledExecutorService recurringJobExecutor = environment.lifecycle()
|
||||||
|
.scheduledExecutorService(name(getClass(), "recurringJob-%d")).threads(6).build();
|
||||||
|
ScheduledExecutorService websocketScheduledExecutor = environment.lifecycle().scheduledExecutorService(name(getClass(), "websocket-%d")).threads(8).build();
|
||||||
|
ExecutorService keyspaceNotificationDispatchExecutor = environment.lifecycle().executorService(name(getClass(), "keyspaceNotification-%d")).maxThreads(16).workQueue(keyspaceNotificationDispatchQueue).build();
|
||||||
|
ExecutorService apnSenderExecutor = environment.lifecycle().executorService(name(getClass(), "apnSender-%d")).maxThreads(1).minThreads(1).build();
|
||||||
|
ExecutorService fcmSenderExecutor = environment.lifecycle().executorService(name(getClass(), "fcmSender-%d")).maxThreads(32).minThreads(32).workQueue(fcmSenderQueue).build();
|
||||||
|
ExecutorService backupServiceExecutor = environment.lifecycle().executorService(name(getClass(), "backupService-%d")).maxThreads(1).minThreads(1).build();
|
||||||
|
ExecutorService storageServiceExecutor = environment.lifecycle().executorService(name(getClass(), "storageService-%d")).maxThreads(1).minThreads(1).build();
|
||||||
|
|
||||||
|
// TODO: generally speaking this is a DynamoDB I/O executor for the accounts table; we should eventually have a general executor for speaking to the accounts table, but most of the server is still synchronous so this isn't widely useful yet
|
||||||
|
ExecutorService batchIdentityCheckExecutor = environment.lifecycle().executorService(name(getClass(), "batchIdentityCheck-%d")).minThreads(32).maxThreads(32).build();
|
||||||
|
ExecutorService multiRecipientMessageExecutor = environment.lifecycle()
|
||||||
|
.executorService(name(getClass(), "multiRecipientMessage-%d")).minThreads(64).maxThreads(64).build();
|
||||||
|
ExecutorService stripeExecutor = environment.lifecycle().executorService(name(getClass(), "stripe-%d")).
|
||||||
|
maxThreads(availableProcessors). // mostly this is IO bound so tying to number of processors is tenuous at best
|
||||||
|
minThreads(availableProcessors). // mostly this is IO bound so tying to number of processors is tenuous at best
|
||||||
|
allowCoreThreadTimeOut(true).
|
||||||
|
build();
|
||||||
|
ExecutorService receiptSenderExecutor = environment.lifecycle()
|
||||||
|
.executorService(name(getClass(), "receiptSender-%d"))
|
||||||
|
.maxThreads(2)
|
||||||
|
.minThreads(2)
|
||||||
|
.workQueue(receiptSenderQueue)
|
||||||
|
.rejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy())
|
||||||
|
.build();
|
||||||
|
ExecutorService registrationCallbackExecutor = environment.lifecycle()
|
||||||
|
.executorService(name(getClass(), "registration-%d"))
|
||||||
|
.maxThreads(2)
|
||||||
|
.minThreads(2)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
final AdminEventLogger adminEventLogger = new GoogleCloudAdminEventLogger(
|
||||||
|
LoggingOptions.newBuilder().setProjectId(config.getAdminEventLoggingConfiguration().projectId())
|
||||||
|
.setCredentials(GoogleCredentials.fromStream(new ByteArrayInputStream(
|
||||||
|
config.getAdminEventLoggingConfiguration().credentials().getBytes(StandardCharsets.UTF_8))))
|
||||||
|
.build().getService(),
|
||||||
|
config.getAdminEventLoggingConfiguration().projectId(),
|
||||||
|
config.getAdminEventLoggingConfiguration().logName());
|
||||||
|
|
||||||
|
StripeManager stripeManager = new StripeManager(config.getStripe().getApiKey(), stripeExecutor,
|
||||||
|
config.getStripe().getIdempotencyKeyGenerator(), config.getStripe().getBoostDescription());
|
||||||
|
|
||||||
|
ExternalServiceCredentialGenerator directoryCredentialsGenerator = new ExternalServiceCredentialGenerator(
|
||||||
|
config.getDirectoryConfiguration().getDirectoryClientConfiguration().getUserAuthenticationTokenSharedSecret(),
|
||||||
|
config.getDirectoryConfiguration().getDirectoryClientConfiguration().getUserAuthenticationTokenUserIdSecret());
|
||||||
|
ExternalServiceCredentialGenerator directoryV2CredentialsGenerator = new ExternalServiceCredentialGenerator(
|
||||||
|
config.getDirectoryV2Configuration().getDirectoryV2ClientConfiguration().getUserAuthenticationTokenSharedSecret(),
|
||||||
|
config.getDirectoryV2Configuration().getDirectoryV2ClientConfiguration().getUserIdTokenSharedSecret(),
|
||||||
|
true, false);
|
||||||
|
|
||||||
|
dynamicConfigurationManager.start();
|
||||||
|
|
||||||
|
ExperimentEnrollmentManager experimentEnrollmentManager = new ExperimentEnrollmentManager(dynamicConfigurationManager);
|
||||||
|
|
||||||
|
TwilioVerifyExperimentEnrollmentManager verifyExperimentEnrollmentManager = new TwilioVerifyExperimentEnrollmentManager(
|
||||||
|
config.getVoiceVerificationConfiguration(), experimentEnrollmentManager);
|
||||||
|
|
||||||
|
ExternalServiceCredentialGenerator storageCredentialsGenerator = new ExternalServiceCredentialGenerator(
|
||||||
|
config.getSecureStorageServiceConfiguration().getUserAuthenticationTokenSharedSecret(), true);
|
||||||
|
ExternalServiceCredentialGenerator backupCredentialsGenerator = new ExternalServiceCredentialGenerator(
|
||||||
|
config.getSecureBackupServiceConfiguration().getUserAuthenticationTokenSharedSecret(), true);
|
||||||
|
ExternalServiceCredentialGenerator paymentsCredentialsGenerator = new ExternalServiceCredentialGenerator(
|
||||||
|
config.getPaymentsServiceConfiguration().getUserAuthenticationTokenSharedSecret(), true);
|
||||||
|
|
||||||
|
AbusiveHostRules abusiveHostRules = new AbusiveHostRules(rateLimitersCluster, dynamicConfigurationManager);
|
||||||
|
RegistrationServiceClient registrationServiceClient = new RegistrationServiceClient(config.getRegistrationServiceConfiguration().getHost(), config.getRegistrationServiceConfiguration().getPort(), config.getRegistrationServiceConfiguration().getApiKey(), config.getRegistrationServiceConfiguration().getRegistrationCaCertificate(), registrationCallbackExecutor);
|
||||||
|
SecureBackupClient secureBackupClient = new SecureBackupClient(backupCredentialsGenerator, backupServiceExecutor, config.getSecureBackupServiceConfiguration());
|
||||||
|
SecureStorageClient secureStorageClient = new SecureStorageClient(storageCredentialsGenerator, storageServiceExecutor, config.getSecureStorageServiceConfiguration());
|
||||||
|
ClientPresenceManager clientPresenceManager = new ClientPresenceManager(clientPresenceCluster, recurringJobExecutor, keyspaceNotificationDispatchExecutor);
|
||||||
|
DirectoryQueue directoryQueue = new DirectoryQueue(config.getDirectoryConfiguration().getSqsConfiguration());
|
||||||
|
StoredVerificationCodeManager pendingAccountsManager = new StoredVerificationCodeManager(pendingAccounts);
|
||||||
|
StoredVerificationCodeManager pendingDevicesManager = new StoredVerificationCodeManager(pendingDevices);
|
||||||
|
ProfilesManager profilesManager = new ProfilesManager(profiles, cacheCluster);
|
||||||
|
MessagesCache messagesCache = new MessagesCache(messagesCluster, messagesCluster, keyspaceNotificationDispatchExecutor);
|
||||||
|
PushLatencyManager pushLatencyManager = new PushLatencyManager(metricsCluster, dynamicConfigurationManager);
|
||||||
|
ReportMessageManager reportMessageManager = new ReportMessageManager(reportMessageDynamoDb, rateLimitersCluster, config.getReportMessageConfiguration().getCounterTtl());
|
||||||
|
MessagesManager messagesManager = new MessagesManager(messagesDynamoDb, messagesCache, reportMessageManager);
|
||||||
|
UsernameGenerator usernameGenerator = new UsernameGenerator(config.getUsername());
|
||||||
|
DeletedAccountsManager deletedAccountsManager = new DeletedAccountsManager(deletedAccounts,
|
||||||
|
deletedAccountsLockDynamoDbClient, config.getDynamoDbTables().getDeletedAccountsLock().getTableName());
|
||||||
|
AccountsManager accountsManager = new AccountsManager(accounts, phoneNumberIdentifiers, cacheCluster,
|
||||||
|
deletedAccountsManager, directoryQueue, keys, messagesManager, prohibitedUsernames, profilesManager,
|
||||||
|
pendingAccountsManager, secureStorageClient, secureBackupClient, clientPresenceManager, usernameGenerator,
|
||||||
|
experimentEnrollmentManager, clock);
|
||||||
|
RemoteConfigsManager remoteConfigsManager = new RemoteConfigsManager(remoteConfigs);
|
||||||
|
DispatchManager dispatchManager = new DispatchManager(pubSubClientFactory, Optional.empty());
|
||||||
|
PubSubManager pubSubManager = new PubSubManager(pubsubClient, dispatchManager);
|
||||||
|
APNSender apnSender = new APNSender(apnSenderExecutor, config.getApnConfiguration());
|
||||||
|
FcmSender fcmSender = new FcmSender(fcmSenderExecutor, config.getFcmConfiguration().credentials());
|
||||||
|
ApnPushNotificationScheduler apnPushNotificationScheduler = new ApnPushNotificationScheduler(pushSchedulerCluster, apnSender, accountsManager);
|
||||||
|
PushNotificationManager pushNotificationManager = new PushNotificationManager(accountsManager, apnSender, fcmSender, apnPushNotificationScheduler, pushLatencyManager, dynamicConfigurationManager);
|
||||||
|
RateLimiters rateLimiters = new RateLimiters(config.getLimitsConfiguration(), rateLimitersCluster);
|
||||||
|
DynamicRateLimiters dynamicRateLimiters = new DynamicRateLimiters(rateLimitersCluster, dynamicConfigurationManager);
|
||||||
|
ProvisioningManager provisioningManager = new ProvisioningManager(pubSubManager);
|
||||||
|
IssuedReceiptsManager issuedReceiptsManager = new IssuedReceiptsManager(
|
||||||
|
config.getDynamoDbTables().getIssuedReceipts().getTableName(),
|
||||||
|
config.getDynamoDbTables().getIssuedReceipts().getExpiration(),
|
||||||
|
dynamoDbAsyncClient,
|
||||||
|
config.getDynamoDbTables().getIssuedReceipts().getGenerator());
|
||||||
|
RedeemedReceiptsManager redeemedReceiptsManager = new RedeemedReceiptsManager(
|
||||||
|
clock,
|
||||||
|
config.getDynamoDbTables().getRedeemedReceipts().getTableName(),
|
||||||
|
dynamoDbAsyncClient,
|
||||||
|
config.getDynamoDbTables().getRedeemedReceipts().getExpiration());
|
||||||
|
SubscriptionManager subscriptionManager = new SubscriptionManager(
|
||||||
|
config.getDynamoDbTables().getSubscriptions().getTableName(), dynamoDbAsyncClient);
|
||||||
|
|
||||||
|
ReportedMessageMetricsListener reportedMessageMetricsListener = new ReportedMessageMetricsListener(accountsManager);
|
||||||
|
reportMessageManager.addListener(reportedMessageMetricsListener);
|
||||||
|
|
||||||
|
AccountAuthenticator accountAuthenticator = new AccountAuthenticator(accountsManager);
|
||||||
|
DisabledPermittedAccountAuthenticator disabledPermittedAccountAuthenticator = new DisabledPermittedAccountAuthenticator(accountsManager);
|
||||||
|
|
||||||
|
TwilioSmsSender twilioSmsSender = new TwilioSmsSender(config.getTwilioConfiguration(), dynamicConfigurationManager);
|
||||||
|
SmsSender smsSender = new SmsSender(twilioSmsSender);
|
||||||
|
MessageSender messageSender = new MessageSender(clientPresenceManager, messagesManager, pushNotificationManager, pushLatencyManager);
|
||||||
|
ReceiptSender receiptSender = new ReceiptSender(accountsManager, messageSender, receiptSenderExecutor);
|
||||||
|
TurnTokenGenerator turnTokenGenerator = new TurnTokenGenerator(dynamicConfigurationManager);
|
||||||
|
RecaptchaClient recaptchaClient = new RecaptchaClient(
|
||||||
|
config.getRecaptchaConfiguration().getProjectPath(),
|
||||||
|
config.getRecaptchaConfiguration().getCredentialConfigurationJson(),
|
||||||
|
dynamicConfigurationManager);
|
||||||
|
PushChallengeManager pushChallengeManager = new PushChallengeManager(pushNotificationManager, pushChallengeDynamoDb);
|
||||||
|
RateLimitChallengeManager rateLimitChallengeManager = new RateLimitChallengeManager(pushChallengeManager,
|
||||||
|
recaptchaClient, dynamicRateLimiters);
|
||||||
|
RateLimitChallengeOptionManager rateLimitChallengeOptionManager =
|
||||||
|
new RateLimitChallengeOptionManager(dynamicRateLimiters, dynamicConfigurationManager);
|
||||||
|
|
||||||
|
MessagePersister messagePersister = new MessagePersister(messagesCache, messagesManager, accountsManager, dynamicConfigurationManager, Duration.ofMinutes(config.getMessageCacheConfiguration().getPersistDelayMinutes()));
|
||||||
|
ChangeNumberManager changeNumberManager = new ChangeNumberManager(messageSender, accountsManager);
|
||||||
|
|
||||||
|
final List<AccountDatabaseCrawlerListener> directoryReconciliationAccountDatabaseCrawlerListeners = new ArrayList<>();
|
||||||
|
final List<DeletedAccountsDirectoryReconciler> deletedAccountsDirectoryReconcilers = new ArrayList<>();
|
||||||
|
for (DirectoryServerConfiguration directoryServerConfiguration : config.getDirectoryConfiguration()
|
||||||
|
.getDirectoryServerConfiguration()) {
|
||||||
|
final DirectoryReconciliationClient directoryReconciliationClient = new DirectoryReconciliationClient(
|
||||||
|
directoryServerConfiguration);
|
||||||
|
final DirectoryReconciler directoryReconciler = new DirectoryReconciler(
|
||||||
|
directoryServerConfiguration.getReplicationName(), directoryReconciliationClient,
|
||||||
|
dynamicConfigurationManager);
|
||||||
|
// reconcilers are read-only
|
||||||
|
directoryReconciliationAccountDatabaseCrawlerListeners.add(directoryReconciler);
|
||||||
|
|
||||||
|
final DeletedAccountsDirectoryReconciler deletedAccountsDirectoryReconciler = new DeletedAccountsDirectoryReconciler(
|
||||||
|
directoryServerConfiguration.getReplicationName(), directoryReconciliationClient);
|
||||||
|
deletedAccountsDirectoryReconcilers.add(deletedAccountsDirectoryReconciler);
|
||||||
|
}
|
||||||
|
|
||||||
|
AccountDatabaseCrawlerCache directoryReconciliationAccountDatabaseCrawlerCache = new AccountDatabaseCrawlerCache(
|
||||||
|
cacheCluster, AccountDatabaseCrawlerCache.DIRECTORY_RECONCILER_PREFIX);
|
||||||
|
AccountDatabaseCrawler directoryReconciliationAccountDatabaseCrawler = new AccountDatabaseCrawler(
|
||||||
|
"Reconciliation crawler",
|
||||||
|
accountsManager,
|
||||||
|
directoryReconciliationAccountDatabaseCrawlerCache, directoryReconciliationAccountDatabaseCrawlerListeners,
|
||||||
|
config.getAccountDatabaseCrawlerConfiguration().getChunkSize(),
|
||||||
|
config.getAccountDatabaseCrawlerConfiguration().getChunkIntervalMs()
|
||||||
|
);
|
||||||
|
|
||||||
|
AccountDatabaseCrawlerCache accountCleanerAccountDatabaseCrawlerCache =
|
||||||
|
new AccountDatabaseCrawlerCache(cacheCluster, AccountDatabaseCrawlerCache.ACCOUNT_CLEANER_PREFIX);
|
||||||
|
AccountDatabaseCrawler accountCleanerAccountDatabaseCrawler = new AccountDatabaseCrawler("Account cleaner crawler",
|
||||||
|
accountsManager,
|
||||||
|
accountCleanerAccountDatabaseCrawlerCache, List.of(new AccountCleaner(accountsManager)),
|
||||||
|
config.getAccountDatabaseCrawlerConfiguration().getChunkSize(),
|
||||||
|
config.getAccountDatabaseCrawlerConfiguration().getChunkIntervalMs()
|
||||||
|
);
|
||||||
|
|
||||||
|
// TODO listeners must be ordered so that ones that directly update accounts come last, so that read-only ones are not working with stale data
|
||||||
|
final List<AccountDatabaseCrawlerListener> accountDatabaseCrawlerListeners = List.of(
|
||||||
|
new NonNormalizedAccountCrawlerListener(accountsManager, metricsCluster),
|
||||||
|
new ContactDiscoveryWriter(accountsManager),
|
||||||
|
// PushFeedbackProcessor may update device properties
|
||||||
|
new PushFeedbackProcessor(accountsManager));
|
||||||
|
|
||||||
|
AccountDatabaseCrawlerCache accountDatabaseCrawlerCache = new AccountDatabaseCrawlerCache(cacheCluster,
|
||||||
|
AccountDatabaseCrawlerCache.GENERAL_PURPOSE_PREFIX);
|
||||||
|
AccountDatabaseCrawler accountDatabaseCrawler = new AccountDatabaseCrawler("General-purpose account crawler",
|
||||||
|
accountsManager,
|
||||||
|
accountDatabaseCrawlerCache, accountDatabaseCrawlerListeners,
|
||||||
|
config.getAccountDatabaseCrawlerConfiguration().getChunkSize(),
|
||||||
|
config.getAccountDatabaseCrawlerConfiguration().getChunkIntervalMs()
|
||||||
|
);
|
||||||
|
|
||||||
|
DeletedAccountsTableCrawler deletedAccountsTableCrawler = new DeletedAccountsTableCrawler(deletedAccountsManager, deletedAccountsDirectoryReconcilers, cacheCluster, recurringJobExecutor);
|
||||||
|
|
||||||
|
HttpClient currencyClient = HttpClient.newBuilder().version(HttpClient.Version.HTTP_2).connectTimeout(Duration.ofSeconds(10)).build();
|
||||||
|
FixerClient fixerClient = new FixerClient(currencyClient, config.getPaymentsServiceConfiguration().getFixerApiKey());
|
||||||
|
FtxClient ftxClient = new FtxClient(currencyClient);
|
||||||
|
CurrencyConversionManager currencyManager = new CurrencyConversionManager(fixerClient, ftxClient, config.getPaymentsServiceConfiguration().getPaymentCurrencies());
|
||||||
|
|
||||||
|
environment.lifecycle().manage(apnSender);
|
||||||
|
environment.lifecycle().manage(apnPushNotificationScheduler);
|
||||||
|
environment.lifecycle().manage(pubSubManager);
|
||||||
|
environment.lifecycle().manage(accountDatabaseCrawler);
|
||||||
|
environment.lifecycle().manage(directoryReconciliationAccountDatabaseCrawler);
|
||||||
|
environment.lifecycle().manage(accountCleanerAccountDatabaseCrawler);
|
||||||
|
environment.lifecycle().manage(deletedAccountsTableCrawler);
|
||||||
|
environment.lifecycle().manage(messagesCache);
|
||||||
|
environment.lifecycle().manage(messagePersister);
|
||||||
|
environment.lifecycle().manage(clientPresenceManager);
|
||||||
|
environment.lifecycle().manage(currencyManager);
|
||||||
|
environment.lifecycle().manage(directoryQueue);
|
||||||
|
environment.lifecycle().manage(registrationServiceClient);
|
||||||
|
|
||||||
|
StaticCredentialsProvider cdnCredentialsProvider = StaticCredentialsProvider
|
||||||
|
.create(AwsBasicCredentials.create(
|
||||||
|
config.getCdnConfiguration().getAccessKey(),
|
||||||
|
config.getCdnConfiguration().getAccessSecret()));
|
||||||
|
S3Client cdnS3Client = S3Client.builder()
|
||||||
|
.credentialsProvider(cdnCredentialsProvider)
|
||||||
|
.region(Region.of(config.getCdnConfiguration().getRegion()))
|
||||||
|
.build();
|
||||||
|
PostPolicyGenerator profileCdnPolicyGenerator = new PostPolicyGenerator(config.getCdnConfiguration().getRegion(),
|
||||||
|
config.getCdnConfiguration().getBucket(), config.getCdnConfiguration().getAccessKey());
|
||||||
|
PolicySigner profileCdnPolicySigner = new PolicySigner(config.getCdnConfiguration().getAccessSecret(),
|
||||||
|
config.getCdnConfiguration().getRegion());
|
||||||
|
|
||||||
|
ServerSecretParams zkSecretParams = new ServerSecretParams(config.getZkConfig().getServerSecret());
|
||||||
|
ServerZkProfileOperations zkProfileOperations = new ServerZkProfileOperations(zkSecretParams);
|
||||||
|
ServerZkAuthOperations zkAuthOperations = new ServerZkAuthOperations(zkSecretParams);
|
||||||
|
ServerZkReceiptOperations zkReceiptOperations = new ServerZkReceiptOperations(zkSecretParams);
|
||||||
|
|
||||||
|
AuthFilter<BasicCredentials, AuthenticatedAccount> accountAuthFilter = new BasicCredentialAuthFilter.Builder<AuthenticatedAccount>().setAuthenticator(
|
||||||
|
accountAuthenticator).buildAuthFilter();
|
||||||
|
AuthFilter<BasicCredentials, DisabledPermittedAuthenticatedAccount> disabledPermittedAccountAuthFilter = new BasicCredentialAuthFilter.Builder<DisabledPermittedAuthenticatedAccount>().setAuthenticator(
|
||||||
|
disabledPermittedAccountAuthenticator).buildAuthFilter();
|
||||||
|
|
||||||
|
environment.servlets()
|
||||||
|
.addFilter("RemoteDeprecationFilter", new RemoteDeprecationFilter(dynamicConfigurationManager))
|
||||||
|
.addMappingForUrlPatterns(EnumSet.of(DispatcherType.REQUEST), false, "/*");
|
||||||
|
|
||||||
|
environment.jersey().register(new ContentLengthFilter(TrafficSource.HTTP));
|
||||||
|
environment.jersey().register(MultiRecipientMessageProvider.class);
|
||||||
|
environment.jersey().register(new MetricsApplicationEventListener(TrafficSource.HTTP));
|
||||||
|
environment.jersey()
|
||||||
|
.register(new PolymorphicAuthDynamicFeature<>(ImmutableMap.of(AuthenticatedAccount.class, accountAuthFilter,
|
||||||
|
DisabledPermittedAuthenticatedAccount.class, disabledPermittedAccountAuthFilter)));
|
||||||
|
environment.jersey().register(new PolymorphicAuthValueFactoryProvider.Binder<>(
|
||||||
|
ImmutableSet.of(AuthenticatedAccount.class, DisabledPermittedAuthenticatedAccount.class)));
|
||||||
|
environment.jersey().register(new WebsocketRefreshApplicationEventListener(accountsManager, clientPresenceManager));
|
||||||
|
environment.jersey().register(new TimestampResponseFilter());
|
||||||
|
environment.jersey().register(new VoiceVerificationController(config.getVoiceVerificationConfiguration().getUrl(),
|
||||||
|
config.getVoiceVerificationConfiguration().getLocales()));
|
||||||
|
|
||||||
|
///
|
||||||
|
WebSocketEnvironment<AuthenticatedAccount> webSocketEnvironment = new WebSocketEnvironment<>(environment,
|
||||||
|
config.getWebSocketConfiguration(), 90000);
|
||||||
|
webSocketEnvironment.setAuthenticator(new WebSocketAccountAuthenticator(accountAuthenticator));
|
||||||
|
webSocketEnvironment.setConnectListener(
|
||||||
|
new AuthenticatedConnectListener(receiptSender, messagesManager, pushNotificationManager,
|
||||||
|
clientPresenceManager, websocketScheduledExecutor));
|
||||||
|
webSocketEnvironment.jersey().register(new WebsocketRefreshApplicationEventListener(accountsManager, clientPresenceManager));
|
||||||
|
webSocketEnvironment.jersey().register(new ContentLengthFilter(TrafficSource.WEBSOCKET));
|
||||||
|
webSocketEnvironment.jersey().register(MultiRecipientMessageProvider.class);
|
||||||
|
webSocketEnvironment.jersey().register(new MetricsApplicationEventListener(TrafficSource.WEBSOCKET));
|
||||||
|
webSocketEnvironment.jersey().register(new KeepAliveController(clientPresenceManager));
|
||||||
|
|
||||||
|
// these should be common, but use @Auth DisabledPermittedAccount, which isn’t supported yet on websocket
|
||||||
|
environment.jersey().register(
|
||||||
|
new AccountController(pendingAccountsManager, accountsManager, abusiveHostRules, rateLimiters,
|
||||||
|
smsSender, registrationServiceClient, dynamicConfigurationManager, turnTokenGenerator, config.getTestDevices(),
|
||||||
|
recaptchaClient, pushNotificationManager, verifyExperimentEnrollmentManager,
|
||||||
|
changeNumberManager, backupCredentialsGenerator, experimentEnrollmentManager));
|
||||||
|
environment.jersey().register(new KeysController(rateLimiters, keys, accountsManager));
|
||||||
|
|
||||||
|
final List<Object> commonControllers = Lists.newArrayList(
|
||||||
|
new AttachmentControllerV2(rateLimiters, config.getAwsAttachmentsConfiguration().getAccessKey(), config.getAwsAttachmentsConfiguration().getAccessSecret(), config.getAwsAttachmentsConfiguration().getRegion(), config.getAwsAttachmentsConfiguration().getBucket()),
|
||||||
|
new AttachmentControllerV3(rateLimiters, config.getGcpAttachmentsConfiguration().getDomain(), config.getGcpAttachmentsConfiguration().getEmail(), config.getGcpAttachmentsConfiguration().getMaxSizeInBytes(), config.getGcpAttachmentsConfiguration().getPathPrefix(), config.getGcpAttachmentsConfiguration().getRsaSigningKey()),
|
||||||
|
new CertificateController(new CertificateGenerator(config.getDeliveryCertificate().getCertificate(), config.getDeliveryCertificate().getPrivateKey(), config.getDeliveryCertificate().getExpiresDays()), zkAuthOperations, clock),
|
||||||
|
new ChallengeController(rateLimitChallengeManager),
|
||||||
|
new DeviceController(pendingDevicesManager, accountsManager, messagesManager, keys, rateLimiters, config.getMaxDevices()),
|
||||||
|
new DirectoryController(directoryCredentialsGenerator),
|
||||||
|
new DirectoryV2Controller(directoryV2CredentialsGenerator),
|
||||||
|
new DonationController(clock, zkReceiptOperations, redeemedReceiptsManager, accountsManager, config.getBadges(),
|
||||||
|
ReceiptCredentialPresentation::new, stripeExecutor, config.getDonationConfiguration(), config.getStripe()),
|
||||||
|
new MessageController(rateLimiters, messageSender, receiptSender, accountsManager, deletedAccountsManager, messagesManager, pushNotificationManager, reportMessageManager, multiRecipientMessageExecutor),
|
||||||
|
new PaymentsController(currencyManager, paymentsCredentialsGenerator),
|
||||||
|
new ProfileController(clock, rateLimiters, accountsManager, profilesManager, dynamicConfigurationManager, profileBadgeConverter, config.getBadges(), cdnS3Client, profileCdnPolicyGenerator, profileCdnPolicySigner, config.getCdnConfiguration().getBucket(), zkProfileOperations, batchIdentityCheckExecutor),
|
||||||
|
new ProvisioningController(rateLimiters, provisioningManager),
|
||||||
|
new RemoteConfigController(remoteConfigsManager, adminEventLogger, config.getRemoteConfigConfiguration().getAuthorizedTokens(), config.getRemoteConfigConfiguration().getGlobalConfig()),
|
||||||
|
new SecureBackupController(backupCredentialsGenerator),
|
||||||
|
new SecureStorageController(storageCredentialsGenerator),
|
||||||
|
new StickerController(rateLimiters, config.getCdnConfiguration().getAccessKey(),
|
||||||
|
config.getCdnConfiguration().getAccessSecret(), config.getCdnConfiguration().getRegion(),
|
||||||
|
config.getCdnConfiguration().getBucket())
|
||||||
|
);
|
||||||
|
if (config.getSubscription() != null && config.getBoost() != null) {
|
||||||
|
commonControllers.add(new SubscriptionController(clock, config.getSubscription(), config.getBoost(),
|
||||||
|
config.getGift(), subscriptionManager, stripeManager, zkReceiptOperations, issuedReceiptsManager,
|
||||||
|
profileBadgeConverter, resourceBundleLevelTranslator));
|
||||||
|
}
|
||||||
|
|
||||||
|
for (Object controller : commonControllers) {
|
||||||
|
environment.jersey().register(controller);
|
||||||
|
webSocketEnvironment.jersey().register(controller);
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean registeredAbusiveMessageFilter = false;
|
||||||
|
|
||||||
|
for (final AbusiveMessageFilter filter : ServiceLoader.load(AbusiveMessageFilter.class)) {
|
||||||
|
if (filter.getClass().isAnnotationPresent(FilterAbusiveMessages.class)) {
|
||||||
|
try {
|
||||||
|
filter.configure(config.getAbusiveMessageFilterConfiguration().getEnvironment());
|
||||||
|
|
||||||
|
environment.lifecycle().manage(filter);
|
||||||
|
environment.jersey().register(filter);
|
||||||
|
webSocketEnvironment.jersey().register(filter);
|
||||||
|
|
||||||
|
log.info("Registered abusive message filter: {}", filter.getClass().getName());
|
||||||
|
registeredAbusiveMessageFilter = true;
|
||||||
|
} catch (final Exception e) {
|
||||||
|
log.warn("Failed to register abusive message filter: {}", filter.getClass().getName(), e);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.warn("Abusive message filter {} not annotated with @FilterAbusiveMessages and will not be installed",
|
||||||
|
filter.getClass().getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filter instanceof RateLimitChallengeListener) {
|
||||||
|
log.info("Registered rate limit challenge listener: {}", filter.getClass().getName());
|
||||||
|
rateLimitChallengeManager.addListener((RateLimitChallengeListener) filter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!registeredAbusiveMessageFilter) {
|
||||||
|
log.warn("No abusive message filters installed");
|
||||||
|
}
|
||||||
|
|
||||||
|
WebSocketEnvironment<AuthenticatedAccount> provisioningEnvironment = new WebSocketEnvironment<>(environment,
|
||||||
|
webSocketEnvironment.getRequestLog(), 60000);
|
||||||
|
provisioningEnvironment.jersey().register(new WebsocketRefreshApplicationEventListener(accountsManager, clientPresenceManager));
|
||||||
|
provisioningEnvironment.setConnectListener(new ProvisioningConnectListener(pubSubManager));
|
||||||
|
provisioningEnvironment.jersey().register(new MetricsApplicationEventListener(TrafficSource.WEBSOCKET));
|
||||||
|
provisioningEnvironment.jersey().register(new KeepAliveController(clientPresenceManager));
|
||||||
|
|
||||||
|
registerCorsFilter(environment);
|
||||||
|
registerExceptionMappers(environment, webSocketEnvironment, provisioningEnvironment);
|
||||||
|
|
||||||
|
environment.jersey().property(ServerProperties.UNWRAP_COMPLETION_STAGE_IN_WRITER_ENABLE, Boolean.TRUE);
|
||||||
|
webSocketEnvironment.jersey().property(ServerProperties.UNWRAP_COMPLETION_STAGE_IN_WRITER_ENABLE, Boolean.TRUE);
|
||||||
|
provisioningEnvironment.jersey().property(ServerProperties.UNWRAP_COMPLETION_STAGE_IN_WRITER_ENABLE, Boolean.TRUE);
|
||||||
|
|
||||||
|
WebSocketResourceProviderFactory<AuthenticatedAccount> webSocketServlet = new WebSocketResourceProviderFactory<>(
|
||||||
|
webSocketEnvironment, AuthenticatedAccount.class, config.getWebSocketConfiguration());
|
||||||
|
WebSocketResourceProviderFactory<AuthenticatedAccount> provisioningServlet = new WebSocketResourceProviderFactory<>(
|
||||||
|
provisioningEnvironment, AuthenticatedAccount.class, config.getWebSocketConfiguration());
|
||||||
|
|
||||||
|
ServletRegistration.Dynamic websocket = environment.servlets().addServlet("WebSocket", webSocketServlet);
|
||||||
|
ServletRegistration.Dynamic provisioning = environment.servlets().addServlet("Provisioning", provisioningServlet);
|
||||||
|
|
||||||
|
websocket.addMapping("/v1/websocket/");
|
||||||
|
websocket.setAsyncSupported(true);
|
||||||
|
|
||||||
|
provisioning.addMapping("/v1/websocket/provisioning/");
|
||||||
|
provisioning.setAsyncSupported(true);
|
||||||
|
|
||||||
|
environment.admin().addTask(new SetRequestLoggingEnabledTask());
|
||||||
|
environment.admin().addTask(new SetCrawlerAccelerationTask(accountDatabaseCrawlerCache));
|
||||||
|
|
||||||
|
environment.healthChecks().register("cacheCluster", new RedisClusterHealthCheck(cacheCluster));
|
||||||
|
|
||||||
|
environment.lifecycle().manage(new ApplicationShutdownMonitor(Metrics.globalRegistry));
|
||||||
|
|
||||||
|
environment.metrics().register(name(CpuUsageGauge.class, "cpu"), new CpuUsageGauge(3, TimeUnit.SECONDS));
|
||||||
|
environment.metrics().register(name(FreeMemoryGauge.class, "free_memory"), new FreeMemoryGauge());
|
||||||
|
environment.metrics().register(name(NetworkSentGauge.class, "bytes_sent"), new NetworkSentGauge());
|
||||||
|
environment.metrics().register(name(NetworkReceivedGauge.class, "bytes_received"), new NetworkReceivedGauge());
|
||||||
|
environment.metrics().register(name(FileDescriptorGauge.class, "fd_count"), new FileDescriptorGauge());
|
||||||
|
environment.metrics().register(name(MaxFileDescriptorGauge.class, "max_fd_count"), new MaxFileDescriptorGauge());
|
||||||
|
environment.metrics()
|
||||||
|
.register(name(OperatingSystemMemoryGauge.class, "buffers"), new OperatingSystemMemoryGauge("Buffers"));
|
||||||
|
environment.metrics()
|
||||||
|
.register(name(OperatingSystemMemoryGauge.class, "cached"), new OperatingSystemMemoryGauge("Cached"));
|
||||||
|
|
||||||
|
BufferPoolGauges.registerMetrics();
|
||||||
|
GarbageCollectionGauges.registerMetrics();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void registerExceptionMappers(Environment environment,
|
||||||
|
WebSocketEnvironment<AuthenticatedAccount> webSocketEnvironment,
|
||||||
|
WebSocketEnvironment<AuthenticatedAccount> provisioningEnvironment) {
|
||||||
|
|
||||||
|
List.of(
|
||||||
|
new LoggingUnhandledExceptionMapper(),
|
||||||
|
new CompletionExceptionMapper(),
|
||||||
|
new IOExceptionMapper(),
|
||||||
|
new RateLimitExceededExceptionMapper(),
|
||||||
|
new InvalidWebsocketAddressExceptionMapper(),
|
||||||
|
new DeviceLimitExceededExceptionMapper(),
|
||||||
|
new ServerRejectedExceptionMapper(),
|
||||||
|
new ImpossiblePhoneNumberExceptionMapper(),
|
||||||
|
new NonNormalizedPhoneNumberExceptionMapper()
|
||||||
|
).forEach(exceptionMapper -> {
|
||||||
|
environment.jersey().register(exceptionMapper);
|
||||||
|
webSocketEnvironment.jersey().register(exceptionMapper);
|
||||||
|
provisioningEnvironment.jersey().register(exceptionMapper);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void registerCorsFilter(Environment environment) {
|
||||||
|
FilterRegistration.Dynamic filter = environment.servlets().addFilter("CORS", CrossOriginFilter.class);
|
||||||
|
filter.addMappingForUrlPatterns(EnumSet.allOf(DispatcherType.class), true, "/*");
|
||||||
|
filter.setInitParameter("allowedOrigins", "*");
|
||||||
|
filter.setInitParameter("allowedHeaders", "Content-Type,Authorization,X-Requested-With,Content-Length,Accept,Origin,X-Signal-Agent");
|
||||||
|
filter.setInitParameter("allowedMethods", "GET,PUT,POST,DELETE,OPTIONS");
|
||||||
|
filter.setInitParameter("preflightMaxAge", "5184000");
|
||||||
|
filter.setInitParameter("allowCredentials", "true");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void main(String[] args) throws Exception {
|
||||||
|
new WhisperServerService().run(args);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.abuse;
|
||||||
|
|
||||||
|
import io.dropwizard.lifecycle.Managed;
|
||||||
|
import javax.ws.rs.container.ContainerRequestFilter;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An abusive message filter is a {@link ContainerRequestFilter} that filters requests to message-sending endpoints to
|
||||||
|
* detect and respond to patterns of abusive behavior.
|
||||||
|
* <p/>
|
||||||
|
* Abusive message filters are managed components that are generally loaded dynamically via a
|
||||||
|
* {@link java.util.ServiceLoader}. Their {@link #configure(String)} method will be called prior to be adding to the
|
||||||
|
* server's pool of {@link Managed} objects.
|
||||||
|
* <p/>
|
||||||
|
* Abusive message filters must be annotated with {@link FilterAbusiveMessages}, a name binding annotation that
|
||||||
|
* restricts the endpoints to which the filter may apply.
|
||||||
|
*/
|
||||||
|
public interface AbusiveMessageFilter extends ContainerRequestFilter, Managed {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configures this abusive message filter. This method will be called before the filter is added to the server's pool
|
||||||
|
* of managed objects and before the server processes any requests.
|
||||||
|
*
|
||||||
|
* @param environmentName the name of the environment in which this filter is running (e.g. "staging" or "production")
|
||||||
|
* @throws IOException if the filter could not read its configuration source for any reason
|
||||||
|
*/
|
||||||
|
void configure(String environmentName) throws IOException;
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.abuse;
|
||||||
|
|
||||||
|
import javax.ws.rs.NameBinding;
|
||||||
|
import java.lang.annotation.ElementType;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.RetentionPolicy;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A name-binding annotation that associates {@link AbusiveMessageFilter}s with resource methods.
|
||||||
|
*/
|
||||||
|
@NameBinding
|
||||||
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
@Target({ElementType.TYPE, ElementType.METHOD})
|
||||||
|
public @interface FilterAbusiveMessages {
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.abuse;
|
||||||
|
|
||||||
|
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public interface RateLimitChallengeListener {
|
||||||
|
|
||||||
|
void handleRateLimitChallengeAnswered(Account account);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configures this rate limit challenge listener. This method will be called before the service begins processing any
|
||||||
|
* challenges.
|
||||||
|
*
|
||||||
|
* @param environmentName the name of the environment in which this listener is running (e.g. "staging" or "production")
|
||||||
|
* @throws IOException if the listener could not read its configuration source for any reason
|
||||||
|
*/
|
||||||
|
void configure(String environmentName) throws IOException;
|
||||||
|
}
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.abuse;
|
||||||
|
|
||||||
|
public enum RateLimitChallengeType {
|
||||||
|
|
||||||
|
PUSH_CHALLENGE,
|
||||||
|
RECAPTCHA
|
||||||
|
}
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Device;
|
||||||
|
|
||||||
|
public interface AccountAndAuthenticatedDeviceHolder {
|
||||||
|
|
||||||
|
Account getAccount();
|
||||||
|
|
||||||
|
Device getAuthenticatedDevice();
|
||||||
|
}
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import io.dropwizard.auth.Authenticator;
|
||||||
|
import io.dropwizard.auth.basic.BasicCredentials;
|
||||||
|
import java.util.Optional;
|
||||||
|
import org.whispersystems.textsecuregcm.experiment.ExperimentEnrollmentManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountsManager;
|
||||||
|
|
||||||
|
public class AccountAuthenticator extends BaseAccountAuthenticator implements
|
||||||
|
Authenticator<BasicCredentials, AuthenticatedAccount> {
|
||||||
|
|
||||||
|
public AccountAuthenticator(AccountsManager accountsManager) {
|
||||||
|
super(accountsManager);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Optional<AuthenticatedAccount> authenticate(BasicCredentials basicCredentials) {
|
||||||
|
return super.authenticate(basicCredentials, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.util.Base64;
|
||||||
|
import javax.ws.rs.WebApplicationException;
|
||||||
|
import javax.ws.rs.core.Response;
|
||||||
|
|
||||||
|
public class Anonymous {
|
||||||
|
|
||||||
|
private final byte[] unidentifiedSenderAccessKey;
|
||||||
|
|
||||||
|
public Anonymous(String header) {
|
||||||
|
try {
|
||||||
|
this.unidentifiedSenderAccessKey = Base64.getDecoder().decode(header);
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
throw new WebApplicationException(e, Response.Status.UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte[] getAccessKey() {
|
||||||
|
return unidentifiedSenderAccessKey;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,100 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.UUID;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import org.glassfish.jersey.server.ContainerRequest;
|
||||||
|
import org.glassfish.jersey.server.monitoring.RequestEvent;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountsManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Device;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Pair;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This {@link WebsocketRefreshRequirementProvider} observes intra-request changes in {@link Account#isEnabled()} and
|
||||||
|
* {@link Device#isEnabled()}.
|
||||||
|
* <p>
|
||||||
|
* If a change in {@link Account#isEnabled()} or any associated {@link Device#isEnabled()} is observed, then any active
|
||||||
|
* WebSocket connections for the account must be closed in order for clients to get a refreshed
|
||||||
|
* {@link io.dropwizard.auth.Auth} object with a current device list.
|
||||||
|
*
|
||||||
|
* @see AuthenticatedAccount
|
||||||
|
* @see DisabledPermittedAuthenticatedAccount
|
||||||
|
*/
|
||||||
|
public class AuthEnablementRefreshRequirementProvider implements WebsocketRefreshRequirementProvider {
|
||||||
|
|
||||||
|
private final AccountsManager accountsManager;
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(AuthEnablementRefreshRequirementProvider.class);
|
||||||
|
|
||||||
|
private static final String ACCOUNT_UUID = AuthEnablementRefreshRequirementProvider.class.getName() + ".accountUuid";
|
||||||
|
private static final String DEVICES_ENABLED = AuthEnablementRefreshRequirementProvider.class.getName() + ".devicesEnabled";
|
||||||
|
|
||||||
|
public AuthEnablementRefreshRequirementProvider(final AccountsManager accountsManager) {
|
||||||
|
this.accountsManager = accountsManager;
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
static Map<Long, Boolean> buildDevicesEnabledMap(final Account account) {
|
||||||
|
return account.getDevices().stream().collect(Collectors.toMap(Device::getId, Device::isEnabled));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void handleRequestFiltered(final RequestEvent requestEvent) {
|
||||||
|
if (requestEvent.getUriInfo().getMatchedResourceMethod().getInvocable().getHandlingMethod().getAnnotation(ChangesDeviceEnabledState.class) != null) {
|
||||||
|
// The authenticated principal, if any, will be available after filters have run.
|
||||||
|
// Now that the account is known, capture a snapshot of `isEnabled` for the account's devices before carrying out
|
||||||
|
// the request’s business logic.
|
||||||
|
ContainerRequestUtil.getAuthenticatedAccount(requestEvent.getContainerRequest()).ifPresent(account ->
|
||||||
|
setAccount(requestEvent.getContainerRequest(), account));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setAccount(final ContainerRequest containerRequest, final Account account) {
|
||||||
|
containerRequest.setProperty(ACCOUNT_UUID, account.getUuid());
|
||||||
|
containerRequest.setProperty(DEVICES_ENABLED, buildDevicesEnabledMap(account));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Pair<UUID, Long>> handleRequestFinished(final RequestEvent requestEvent) {
|
||||||
|
// Now that the request is finished, check whether `isEnabled` changed for any of the devices. If the value did
|
||||||
|
// change or if a devices was added or removed, all devices must disconnect and reauthenticate.
|
||||||
|
if (requestEvent.getContainerRequest().getProperty(DEVICES_ENABLED) != null) {
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked") final Map<Long, Boolean> initialDevicesEnabled =
|
||||||
|
(Map<Long, Boolean>) requestEvent.getContainerRequest().getProperty(DEVICES_ENABLED);
|
||||||
|
|
||||||
|
return accountsManager.getByAccountIdentifier((UUID) requestEvent.getContainerRequest().getProperty(ACCOUNT_UUID)).map(account -> {
|
||||||
|
final Set<Long> deviceIdsToDisplace;
|
||||||
|
final Map<Long, Boolean> currentDevicesEnabled = buildDevicesEnabledMap(account);
|
||||||
|
|
||||||
|
if (!initialDevicesEnabled.equals(currentDevicesEnabled)) {
|
||||||
|
deviceIdsToDisplace = new HashSet<>(initialDevicesEnabled.keySet());
|
||||||
|
deviceIdsToDisplace.addAll(currentDevicesEnabled.keySet());
|
||||||
|
} else {
|
||||||
|
deviceIdsToDisplace = Collections.emptySet();
|
||||||
|
}
|
||||||
|
|
||||||
|
return deviceIdsToDisplace.stream()
|
||||||
|
.map(deviceId -> new Pair<>(account.getUuid(), deviceId))
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
}).orElseGet(() -> {
|
||||||
|
logger.error("Request had account, but it is no longer present");
|
||||||
|
return Collections.emptyList();
|
||||||
|
});
|
||||||
|
} else
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.security.Principal;
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
import javax.security.auth.Subject;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Device;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Pair;
|
||||||
|
|
||||||
|
public class AuthenticatedAccount implements Principal, AccountAndAuthenticatedDeviceHolder {
|
||||||
|
|
||||||
|
private final Supplier<Pair<Account, Device>> accountAndDevice;
|
||||||
|
|
||||||
|
public AuthenticatedAccount(final Supplier<Pair<Account, Device>> accountAndDevice) {
|
||||||
|
this.accountAndDevice = accountAndDevice;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Account getAccount() {
|
||||||
|
return accountAndDevice.get().first();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Device getAuthenticatedDevice() {
|
||||||
|
return accountAndDevice.get().second();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Principal implementation
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean implies(final Subject subject) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,85 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import org.apache.commons.codec.binary.Hex;
|
||||||
|
import org.signal.libsignal.protocol.kdf.HKDF;
|
||||||
|
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.security.MessageDigest;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
|
import java.security.SecureRandom;
|
||||||
|
|
||||||
|
public class AuthenticationCredentials {
|
||||||
|
private static final String V2_PREFIX = "2.";
|
||||||
|
|
||||||
|
private final String hashedAuthenticationToken;
|
||||||
|
private final String salt;
|
||||||
|
|
||||||
|
public enum Version {
|
||||||
|
V1,
|
||||||
|
V2,
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Version CURRENT_VERSION = Version.V2;
|
||||||
|
|
||||||
|
public AuthenticationCredentials(String hashedAuthenticationToken, String salt) {
|
||||||
|
this.hashedAuthenticationToken = hashedAuthenticationToken;
|
||||||
|
this.salt = salt;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AuthenticationCredentials(String authenticationToken) {
|
||||||
|
this.salt = String.valueOf(Math.abs(new SecureRandom().nextInt()));
|
||||||
|
this.hashedAuthenticationToken = getV2HashedValue(salt, authenticationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public AuthenticationCredentials v1ForTesting(String authenticationToken) {
|
||||||
|
String salt = String.valueOf(Math.abs(new SecureRandom().nextInt()));
|
||||||
|
return new AuthenticationCredentials(getV1HashedValue(salt, authenticationToken), salt);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Version getVersion() {
|
||||||
|
if (this.hashedAuthenticationToken.startsWith(V2_PREFIX)) {
|
||||||
|
return Version.V2;
|
||||||
|
}
|
||||||
|
return Version.V1;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getHashedAuthenticationToken() {
|
||||||
|
return hashedAuthenticationToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSalt() {
|
||||||
|
return salt;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean verify(String authenticationToken) {
|
||||||
|
final String theirValue = switch (getVersion()) {
|
||||||
|
case V1 -> getV1HashedValue(salt, authenticationToken);
|
||||||
|
case V2 -> getV2HashedValue(salt, authenticationToken);
|
||||||
|
};
|
||||||
|
return MessageDigest.isEqual(theirValue.getBytes(StandardCharsets.UTF_8), this.hashedAuthenticationToken.getBytes(StandardCharsets.UTF_8));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String getV1HashedValue(String salt, String token) {
|
||||||
|
try {
|
||||||
|
return new String(Hex.encodeHex(MessageDigest.getInstance("SHA1").digest((salt + token).getBytes(StandardCharsets.UTF_8))));
|
||||||
|
} catch (NoSuchAlgorithmException e) {
|
||||||
|
throw new AssertionError(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final byte[] AUTH_TOKEN_HKDF_INFO = "authtoken".getBytes(StandardCharsets.UTF_8);
|
||||||
|
private static String getV2HashedValue(String salt, String token) {
|
||||||
|
byte[] secret = HKDF.deriveSecrets(
|
||||||
|
token.getBytes(StandardCharsets.UTF_8), // key
|
||||||
|
salt.getBytes(StandardCharsets.UTF_8), // salt
|
||||||
|
AUTH_TOKEN_HKDF_INFO,
|
||||||
|
32);
|
||||||
|
return V2_PREFIX + Hex.encodeHexString(secret);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,152 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import static com.codahale.metrics.MetricRegistry.name;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import io.dropwizard.auth.basic.BasicCredentials;
|
||||||
|
import io.micrometer.core.instrument.Metrics;
|
||||||
|
import io.micrometer.core.instrument.Tags;
|
||||||
|
import java.time.Clock;
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.time.temporal.ChronoUnit;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.UUID;
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountsManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Device;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.RefreshingAccountAndDeviceSupplier;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Pair;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Util;
|
||||||
|
|
||||||
|
public class BaseAccountAuthenticator {
|
||||||
|
|
||||||
|
private static final String AUTHENTICATION_COUNTER_NAME = name(BaseAccountAuthenticator.class, "authentication");
|
||||||
|
private static final String AUTHENTICATION_SUCCEEDED_TAG_NAME = "succeeded";
|
||||||
|
private static final String AUTHENTICATION_FAILURE_REASON_TAG_NAME = "reason";
|
||||||
|
private static final String AUTHENTICATION_ENABLED_REQUIRED_TAG_NAME = "enabledRequired";
|
||||||
|
|
||||||
|
private static final String DAYS_SINCE_LAST_SEEN_DISTRIBUTION_NAME = name(BaseAccountAuthenticator.class, "daysSinceLastSeen");
|
||||||
|
private static final String IS_PRIMARY_DEVICE_TAG = "isPrimary";
|
||||||
|
|
||||||
|
private final AccountsManager accountsManager;
|
||||||
|
private final Clock clock;
|
||||||
|
|
||||||
|
public BaseAccountAuthenticator(AccountsManager accountsManager) {
|
||||||
|
this(accountsManager, Clock.systemUTC());
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public BaseAccountAuthenticator(AccountsManager accountsManager, Clock clock) {
|
||||||
|
this.accountsManager = accountsManager;
|
||||||
|
this.clock = clock;
|
||||||
|
}
|
||||||
|
|
||||||
|
static Pair<String, Long> getIdentifierAndDeviceId(final String basicUsername) {
|
||||||
|
final String identifier;
|
||||||
|
final long deviceId;
|
||||||
|
|
||||||
|
final int deviceIdSeparatorIndex = basicUsername.indexOf('.');
|
||||||
|
|
||||||
|
if (deviceIdSeparatorIndex == -1) {
|
||||||
|
identifier = basicUsername;
|
||||||
|
deviceId = Device.MASTER_ID;
|
||||||
|
} else {
|
||||||
|
identifier = basicUsername.substring(0, deviceIdSeparatorIndex);
|
||||||
|
deviceId = Long.parseLong(basicUsername.substring(deviceIdSeparatorIndex + 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Pair<>(identifier, deviceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Optional<AuthenticatedAccount> authenticate(BasicCredentials basicCredentials, boolean enabledRequired) {
|
||||||
|
boolean succeeded = false;
|
||||||
|
String failureReason = null;
|
||||||
|
|
||||||
|
try {
|
||||||
|
final UUID accountUuid;
|
||||||
|
final long deviceId;
|
||||||
|
{
|
||||||
|
final Pair<String, Long> identifierAndDeviceId = getIdentifierAndDeviceId(basicCredentials.getUsername());
|
||||||
|
|
||||||
|
accountUuid = UUID.fromString(identifierAndDeviceId.first());
|
||||||
|
deviceId = identifierAndDeviceId.second();
|
||||||
|
}
|
||||||
|
|
||||||
|
Optional<Account> account = accountsManager.getByAccountIdentifier(accountUuid);
|
||||||
|
|
||||||
|
if (account.isEmpty()) {
|
||||||
|
failureReason = "noSuchAccount";
|
||||||
|
return Optional.empty();
|
||||||
|
}
|
||||||
|
|
||||||
|
Optional<Device> device = account.get().getDevice(deviceId);
|
||||||
|
|
||||||
|
if (device.isEmpty()) {
|
||||||
|
failureReason = "noSuchDevice";
|
||||||
|
return Optional.empty();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (enabledRequired) {
|
||||||
|
if (!device.get().isEnabled()) {
|
||||||
|
failureReason = "deviceDisabled";
|
||||||
|
return Optional.empty();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!account.get().isEnabled()) {
|
||||||
|
failureReason = "accountDisabled";
|
||||||
|
return Optional.empty();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
AuthenticationCredentials deviceAuthenticationCredentials = device.get().getAuthenticationCredentials();
|
||||||
|
if (deviceAuthenticationCredentials.verify(basicCredentials.getPassword())) {
|
||||||
|
succeeded = true;
|
||||||
|
Account authenticatedAccount = updateLastSeen(account.get(), device.get());
|
||||||
|
if (deviceAuthenticationCredentials.getVersion() != AuthenticationCredentials.CURRENT_VERSION) {
|
||||||
|
authenticatedAccount = accountsManager.updateDeviceAuthentication(
|
||||||
|
authenticatedAccount,
|
||||||
|
device.get(),
|
||||||
|
new AuthenticationCredentials(basicCredentials.getPassword())); // new credentials have current version
|
||||||
|
}
|
||||||
|
return Optional.of(new AuthenticatedAccount(
|
||||||
|
new RefreshingAccountAndDeviceSupplier(authenticatedAccount, device.get().getId(), accountsManager)));
|
||||||
|
}
|
||||||
|
|
||||||
|
return Optional.empty();
|
||||||
|
} catch (IllegalArgumentException | InvalidAuthorizationHeaderException iae) {
|
||||||
|
failureReason = "invalidHeader";
|
||||||
|
return Optional.empty();
|
||||||
|
} finally {
|
||||||
|
Tags tags = Tags.of(
|
||||||
|
AUTHENTICATION_SUCCEEDED_TAG_NAME, String.valueOf(succeeded),
|
||||||
|
AUTHENTICATION_ENABLED_REQUIRED_TAG_NAME, String.valueOf(enabledRequired));
|
||||||
|
|
||||||
|
if (StringUtils.isNotBlank(failureReason)) {
|
||||||
|
tags = tags.and(AUTHENTICATION_FAILURE_REASON_TAG_NAME, failureReason);
|
||||||
|
}
|
||||||
|
|
||||||
|
Metrics.counter(AUTHENTICATION_COUNTER_NAME, tags).increment();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public Account updateLastSeen(Account account, Device device) {
|
||||||
|
final long lastSeenOffsetSeconds = Math.abs(account.getUuid().getLeastSignificantBits()) % ChronoUnit.DAYS.getDuration().toSeconds();
|
||||||
|
final long todayInMillisWithOffset = Util.todayInMillisGivenOffsetFromNow(clock, Duration.ofSeconds(lastSeenOffsetSeconds).negated());
|
||||||
|
|
||||||
|
if (device.getLastSeen() < todayInMillisWithOffset) {
|
||||||
|
Metrics.summary(DAYS_SINCE_LAST_SEEN_DISTRIBUTION_NAME, IS_PRIMARY_DEVICE_TAG, String.valueOf(device.isMaster()))
|
||||||
|
.record(Duration.ofMillis(todayInMillisWithOffset - device.getLastSeen()).toDays());
|
||||||
|
|
||||||
|
return accountsManager.updateDeviceLastSeen(account, device, Util.todayInMillis(clock));
|
||||||
|
}
|
||||||
|
|
||||||
|
return account;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,94 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.util.Base64;
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Pair;
|
||||||
|
|
||||||
|
public class BasicAuthorizationHeader {
|
||||||
|
|
||||||
|
private final String username;
|
||||||
|
private final long deviceId;
|
||||||
|
private final String password;
|
||||||
|
|
||||||
|
private BasicAuthorizationHeader(final String username, final long deviceId, final String password) {
|
||||||
|
this.username = username;
|
||||||
|
this.deviceId = deviceId;
|
||||||
|
this.password = password;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static BasicAuthorizationHeader fromString(final String header) throws InvalidAuthorizationHeaderException {
|
||||||
|
try {
|
||||||
|
if (StringUtils.isBlank(header)) {
|
||||||
|
throw new InvalidAuthorizationHeaderException("Blank header");
|
||||||
|
}
|
||||||
|
|
||||||
|
final int spaceIndex = header.indexOf(' ');
|
||||||
|
|
||||||
|
if (spaceIndex == -1) {
|
||||||
|
throw new InvalidAuthorizationHeaderException("Invalid authorization header: " + header);
|
||||||
|
}
|
||||||
|
|
||||||
|
final String authorizationType = header.substring(0, spaceIndex);
|
||||||
|
|
||||||
|
if (!"Basic".equals(authorizationType)) {
|
||||||
|
throw new InvalidAuthorizationHeaderException("Unsupported authorization method: " + authorizationType);
|
||||||
|
}
|
||||||
|
|
||||||
|
final String credentials;
|
||||||
|
|
||||||
|
try {
|
||||||
|
credentials = new String(Base64.getDecoder().decode(header.substring(spaceIndex + 1)));
|
||||||
|
} catch (final IndexOutOfBoundsException e) {
|
||||||
|
throw new InvalidAuthorizationHeaderException("Missing credentials");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (StringUtils.isEmpty(credentials)) {
|
||||||
|
throw new InvalidAuthorizationHeaderException("Bad decoded value: " + credentials);
|
||||||
|
}
|
||||||
|
|
||||||
|
final int credentialSeparatorIndex = credentials.indexOf(':');
|
||||||
|
|
||||||
|
if (credentialSeparatorIndex == -1) {
|
||||||
|
throw new InvalidAuthorizationHeaderException("Badly-formatted credentials: " + credentials);
|
||||||
|
}
|
||||||
|
|
||||||
|
final String usernameComponent = credentials.substring(0, credentialSeparatorIndex);
|
||||||
|
|
||||||
|
final String username;
|
||||||
|
final long deviceId;
|
||||||
|
{
|
||||||
|
final Pair<String, Long> identifierAndDeviceId =
|
||||||
|
BaseAccountAuthenticator.getIdentifierAndDeviceId(usernameComponent);
|
||||||
|
|
||||||
|
username = identifierAndDeviceId.first();
|
||||||
|
deviceId = identifierAndDeviceId.second();
|
||||||
|
}
|
||||||
|
|
||||||
|
final String password = credentials.substring(credentialSeparatorIndex + 1);
|
||||||
|
|
||||||
|
if (StringUtils.isAnyBlank(username, password)) {
|
||||||
|
throw new InvalidAuthorizationHeaderException("Username or password were blank");
|
||||||
|
}
|
||||||
|
|
||||||
|
return new BasicAuthorizationHeader(username, deviceId, password);
|
||||||
|
} catch (final IllegalArgumentException | IndexOutOfBoundsException e) {
|
||||||
|
throw new InvalidAuthorizationHeaderException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getUsername() {
|
||||||
|
return username;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getDeviceId() {
|
||||||
|
return deviceId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getPassword() {
|
||||||
|
return password;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import com.google.protobuf.ByteString;
|
||||||
|
import com.google.protobuf.InvalidProtocolBufferException;
|
||||||
|
import java.security.InvalidKeyException;
|
||||||
|
import java.util.Base64;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import org.signal.libsignal.protocol.ecc.Curve;
|
||||||
|
import org.signal.libsignal.protocol.ecc.ECPrivateKey;
|
||||||
|
import org.whispersystems.textsecuregcm.entities.MessageProtos.SenderCertificate;
|
||||||
|
import org.whispersystems.textsecuregcm.entities.MessageProtos.ServerCertificate;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Device;
|
||||||
|
|
||||||
|
public class CertificateGenerator {
|
||||||
|
|
||||||
|
private final ECPrivateKey privateKey;
|
||||||
|
private final int expiresDays;
|
||||||
|
private final ServerCertificate serverCertificate;
|
||||||
|
|
||||||
|
public CertificateGenerator(byte[] serverCertificate, ECPrivateKey privateKey, int expiresDays)
|
||||||
|
throws InvalidProtocolBufferException
|
||||||
|
{
|
||||||
|
this.privateKey = privateKey;
|
||||||
|
this.expiresDays = expiresDays;
|
||||||
|
this.serverCertificate = ServerCertificate.parseFrom(serverCertificate);
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte[] createFor(Account account, Device device, boolean includeE164) throws InvalidKeyException {
|
||||||
|
SenderCertificate.Certificate.Builder builder = SenderCertificate.Certificate.newBuilder()
|
||||||
|
.setSenderDevice(Math.toIntExact(device.getId()))
|
||||||
|
.setExpires(System.currentTimeMillis() + TimeUnit.DAYS.toMillis(expiresDays))
|
||||||
|
.setIdentityKey(ByteString.copyFrom(Base64.getDecoder().decode(account.getIdentityKey())))
|
||||||
|
.setSigner(serverCertificate)
|
||||||
|
.setSenderUuid(account.getUuid().toString());
|
||||||
|
|
||||||
|
if (includeE164) {
|
||||||
|
builder.setSender(account.getNumber());
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] certificate = builder.build().toByteArray();
|
||||||
|
byte[] signature;
|
||||||
|
try {
|
||||||
|
signature = Curve.calculateSignature(privateKey, certificate);
|
||||||
|
} catch (org.signal.libsignal.protocol.InvalidKeyException e) {
|
||||||
|
throw new InvalidKeyException(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return SenderCertificate.newBuilder()
|
||||||
|
.setCertificate(ByteString.copyFrom(certificate))
|
||||||
|
.setSignature(ByteString.copyFrom(signature))
|
||||||
|
.build()
|
||||||
|
.toByteArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.lang.annotation.ElementType;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.RetentionPolicy;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Indicates that an endpoint may change the "enabled" state of one or more devices associated with an account, and that
|
||||||
|
* any websockets associated with the account may need to be refreshed after a call to that endpoint.
|
||||||
|
*/
|
||||||
|
@Target(ElementType.METHOD)
|
||||||
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
public @interface ChangesDeviceEnabledState {
|
||||||
|
}
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.util.Base64;
|
||||||
|
import javax.ws.rs.WebApplicationException;
|
||||||
|
import javax.ws.rs.core.Response;
|
||||||
|
import javax.ws.rs.core.Response.Status;
|
||||||
|
|
||||||
|
public class CombinedUnidentifiedSenderAccessKeys {
|
||||||
|
private final byte[] combinedUnidentifiedSenderAccessKeys;
|
||||||
|
|
||||||
|
public CombinedUnidentifiedSenderAccessKeys(String header) {
|
||||||
|
try {
|
||||||
|
this.combinedUnidentifiedSenderAccessKeys = Base64.getDecoder().decode(header);
|
||||||
|
if (this.combinedUnidentifiedSenderAccessKeys == null || this.combinedUnidentifiedSenderAccessKeys.length != 16) {
|
||||||
|
throw new WebApplicationException("Invalid combined unidentified sender access keys", Status.UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
throw new WebApplicationException(e, Response.Status.UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte[] getAccessKeys() {
|
||||||
|
return combinedUnidentifiedSenderAccessKeys;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import org.glassfish.jersey.server.ContainerRequest;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import javax.ws.rs.core.SecurityContext;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
class ContainerRequestUtil {
|
||||||
|
|
||||||
|
static Optional<Account> getAuthenticatedAccount(final ContainerRequest request) {
|
||||||
|
return Optional.ofNullable(request.getSecurityContext())
|
||||||
|
.map(SecurityContext::getUserPrincipal)
|
||||||
|
.map(principal -> principal instanceof AccountAndAuthenticatedDeviceHolder
|
||||||
|
? ((AccountAndAuthenticatedDeviceHolder) principal).getAccount() : null);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import io.dropwizard.auth.Authenticator;
|
||||||
|
import io.dropwizard.auth.basic.BasicCredentials;
|
||||||
|
import java.util.Optional;
|
||||||
|
import org.whispersystems.textsecuregcm.experiment.ExperimentEnrollmentManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountsManager;
|
||||||
|
|
||||||
|
public class DisabledPermittedAccountAuthenticator extends BaseAccountAuthenticator implements
|
||||||
|
Authenticator<BasicCredentials, DisabledPermittedAuthenticatedAccount> {
|
||||||
|
|
||||||
|
public DisabledPermittedAccountAuthenticator(AccountsManager accountsManager) {
|
||||||
|
super(accountsManager);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Optional<DisabledPermittedAuthenticatedAccount> authenticate(BasicCredentials credentials) {
|
||||||
|
Optional<AuthenticatedAccount> account = super.authenticate(credentials, false);
|
||||||
|
return account.map(DisabledPermittedAuthenticatedAccount::new);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.security.Principal;
|
||||||
|
import javax.security.auth.Subject;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Device;
|
||||||
|
|
||||||
|
public class DisabledPermittedAuthenticatedAccount implements Principal, AccountAndAuthenticatedDeviceHolder {
|
||||||
|
|
||||||
|
private final AuthenticatedAccount authenticatedAccount;
|
||||||
|
|
||||||
|
public DisabledPermittedAuthenticatedAccount(final AuthenticatedAccount authenticatedAccount) {
|
||||||
|
this.authenticatedAccount = authenticatedAccount;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Account getAccount() {
|
||||||
|
return authenticatedAccount.getAccount();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Device getAuthenticatedDevice() {
|
||||||
|
return authenticatedAccount.getAuthenticatedDevice();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Principal implementation
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean implies(Subject subject) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,86 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import java.security.InvalidKeyException;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
|
import java.time.Clock;
|
||||||
|
import javax.crypto.Mac;
|
||||||
|
import javax.crypto.spec.SecretKeySpec;
|
||||||
|
import org.apache.commons.codec.binary.Hex;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Util;
|
||||||
|
|
||||||
|
public class ExternalServiceCredentialGenerator {
|
||||||
|
|
||||||
|
private final byte[] key;
|
||||||
|
private final byte[] userIdKey;
|
||||||
|
private final boolean usernameDerivation;
|
||||||
|
private final boolean prependUsername;
|
||||||
|
private final Clock clock;
|
||||||
|
|
||||||
|
public ExternalServiceCredentialGenerator(byte[] key, byte[] userIdKey) {
|
||||||
|
this(key, userIdKey, true, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExternalServiceCredentialGenerator(byte[] key, boolean prependUsername) {
|
||||||
|
this(key, new byte[0], false, prependUsername);
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public ExternalServiceCredentialGenerator(byte[] key, byte[] userIdKey, boolean usernameDerivation) {
|
||||||
|
this(key, userIdKey, usernameDerivation, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExternalServiceCredentialGenerator(byte[] key, byte[] userIdKey, boolean usernameDerivation,
|
||||||
|
boolean prependUsername) {
|
||||||
|
this(key, userIdKey, usernameDerivation, prependUsername, Clock.systemUTC());
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public ExternalServiceCredentialGenerator(byte[] key, byte[] userIdKey, boolean usernameDerivation,
|
||||||
|
boolean prependUsername, Clock clock) {
|
||||||
|
this.key = key;
|
||||||
|
this.userIdKey = userIdKey;
|
||||||
|
this.usernameDerivation = usernameDerivation;
|
||||||
|
this.prependUsername = prependUsername;
|
||||||
|
this.clock = clock;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExternalServiceCredentials generateFor(String identity) {
|
||||||
|
Mac mac = getMacInstance();
|
||||||
|
String username = getUserId(identity, mac, usernameDerivation);
|
||||||
|
long currentTimeSeconds = clock.millis() / 1000;
|
||||||
|
String prefix = username + ":" + currentTimeSeconds;
|
||||||
|
String output = Hex.encodeHexString(Util.truncate(getHmac(key, prefix.getBytes(), mac), 10));
|
||||||
|
String token = (prependUsername ? prefix : currentTimeSeconds) + ":" + output;
|
||||||
|
|
||||||
|
return new ExternalServiceCredentials(username, token);
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getUserId(String number, Mac mac, boolean usernameDerivation) {
|
||||||
|
if (usernameDerivation) return Hex.encodeHexString(Util.truncate(getHmac(userIdKey, number.getBytes(), mac), 10));
|
||||||
|
else return number;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Mac getMacInstance() {
|
||||||
|
try {
|
||||||
|
return Mac.getInstance("HmacSHA256");
|
||||||
|
} catch (NoSuchAlgorithmException e) {
|
||||||
|
throw new AssertionError(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private byte[] getHmac(byte[] key, byte[] input, Mac mac) {
|
||||||
|
try {
|
||||||
|
mac.init(new SecretKeySpec(key, "HmacSHA256"));
|
||||||
|
return mac.doFinal(input);
|
||||||
|
} catch (InvalidKeyException e) {
|
||||||
|
throw new AssertionError(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
|
||||||
|
public class ExternalServiceCredentials {
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private String username;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private String password;
|
||||||
|
|
||||||
|
public ExternalServiceCredentials(String username, String password) {
|
||||||
|
this.username = username;
|
||||||
|
this.password = password;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExternalServiceCredentials() {}
|
||||||
|
|
||||||
|
public String getUsername() {
|
||||||
|
return username;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getPassword() {
|
||||||
|
return password;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
|
||||||
|
import javax.ws.rs.WebApplicationException;
|
||||||
|
import javax.ws.rs.core.Response.Status;
|
||||||
|
|
||||||
|
public class InvalidAuthorizationHeaderException extends WebApplicationException {
|
||||||
|
public InvalidAuthorizationHeaderException(String s) {
|
||||||
|
super(s, Status.UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
|
||||||
|
public InvalidAuthorizationHeaderException(Exception e) {
|
||||||
|
super(e, Status.UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,80 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Device;
|
||||||
|
|
||||||
|
import javax.ws.rs.NotAuthorizedException;
|
||||||
|
import javax.ws.rs.NotFoundException;
|
||||||
|
import javax.ws.rs.WebApplicationException;
|
||||||
|
import javax.ws.rs.core.Response;
|
||||||
|
import java.security.MessageDigest;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
|
||||||
|
public class OptionalAccess {
|
||||||
|
|
||||||
|
public static final String UNIDENTIFIED = "Unidentified-Access-Key";
|
||||||
|
|
||||||
|
public static void verify(Optional<Account> requestAccount,
|
||||||
|
Optional<Anonymous> accessKey,
|
||||||
|
Optional<Account> targetAccount,
|
||||||
|
String deviceSelector)
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
verify(requestAccount, accessKey, targetAccount);
|
||||||
|
|
||||||
|
if (!deviceSelector.equals("*")) {
|
||||||
|
long deviceId = Long.parseLong(deviceSelector);
|
||||||
|
|
||||||
|
Optional<Device> targetDevice = targetAccount.get().getDevice(deviceId);
|
||||||
|
|
||||||
|
if (targetDevice.isPresent() && targetDevice.get().isEnabled()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (requestAccount.isPresent()) {
|
||||||
|
throw new NotFoundException();
|
||||||
|
} else {
|
||||||
|
throw new NotAuthorizedException(Response.Status.UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
throw new WebApplicationException(Response.status(422).build());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void verify(Optional<Account> requestAccount,
|
||||||
|
Optional<Anonymous> accessKey,
|
||||||
|
Optional<Account> targetAccount)
|
||||||
|
{
|
||||||
|
if (requestAccount.isPresent() && targetAccount.isPresent() && targetAccount.get().isEnabled()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//noinspection ConstantConditions
|
||||||
|
if (requestAccount.isPresent() && (targetAccount.isEmpty() || (targetAccount.isPresent() && !targetAccount.get().isEnabled()))) {
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (accessKey.isPresent() && targetAccount.isPresent() && targetAccount.get().isEnabled() && targetAccount.get().isUnrestrictedUnidentifiedAccess()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (accessKey.isPresent() &&
|
||||||
|
targetAccount.isPresent() &&
|
||||||
|
targetAccount.get().getUnidentifiedAccessKey().isPresent() &&
|
||||||
|
targetAccount.get().isEnabled() &&
|
||||||
|
MessageDigest.isEqual(accessKey.get().getAccessKey(), targetAccount.get().getUnidentifiedAccessKey().get()))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new NotAuthorizedException(Response.Status.UNAUTHORIZED);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,46 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.UUID;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import org.glassfish.jersey.server.monitoring.RequestEvent;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.Account;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Pair;
|
||||||
|
|
||||||
|
public class PhoneNumberChangeRefreshRequirementProvider implements WebsocketRefreshRequirementProvider {
|
||||||
|
|
||||||
|
private static final String INITIAL_NUMBER_KEY =
|
||||||
|
PhoneNumberChangeRefreshRequirementProvider.class.getName() + ".initialNumber";
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void handleRequestFiltered(final RequestEvent requestEvent) {
|
||||||
|
ContainerRequestUtil.getAuthenticatedAccount(requestEvent.getContainerRequest())
|
||||||
|
.ifPresent(account -> requestEvent.getContainerRequest().setProperty(INITIAL_NUMBER_KEY, account.getNumber()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Pair<UUID, Long>> handleRequestFinished(final RequestEvent requestEvent) {
|
||||||
|
final String initialNumber = (String) requestEvent.getContainerRequest().getProperty(INITIAL_NUMBER_KEY);
|
||||||
|
|
||||||
|
if (initialNumber != null) {
|
||||||
|
final Optional<Account> maybeAuthenticatedAccount =
|
||||||
|
ContainerRequestUtil.getAuthenticatedAccount(requestEvent.getContainerRequest());
|
||||||
|
|
||||||
|
return maybeAuthenticatedAccount
|
||||||
|
.filter(account -> !initialNumber.equals(account.getNumber()))
|
||||||
|
.map(account -> account.getDevices().stream()
|
||||||
|
.map(device -> new Pair<>(account.getUuid(), device.getId()))
|
||||||
|
.collect(Collectors.toList()))
|
||||||
|
.orElse(Collections.emptyList());
|
||||||
|
} else {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,58 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Util;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
|
||||||
|
public class StoredRegistrationLock {
|
||||||
|
|
||||||
|
private final Optional<String> registrationLock;
|
||||||
|
|
||||||
|
private final Optional<String> registrationLockSalt;
|
||||||
|
|
||||||
|
private final long lastSeen;
|
||||||
|
|
||||||
|
public StoredRegistrationLock(Optional<String> registrationLock, Optional<String> registrationLockSalt, long lastSeen) {
|
||||||
|
this.registrationLock = registrationLock;
|
||||||
|
this.registrationLockSalt = registrationLockSalt;
|
||||||
|
this.lastSeen = lastSeen;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean requiresClientRegistrationLock() {
|
||||||
|
return registrationLock.isPresent() && registrationLockSalt.isPresent() && System.currentTimeMillis() - lastSeen < TimeUnit.DAYS.toMillis(7);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean needsFailureCredentials() {
|
||||||
|
return registrationLock.isPresent() && registrationLockSalt.isPresent();
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getTimeRemaining() {
|
||||||
|
return TimeUnit.DAYS.toMillis(7) - (System.currentTimeMillis() - lastSeen);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean verify(@Nullable String clientRegistrationLock) {
|
||||||
|
if (Util.isEmpty(clientRegistrationLock)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (registrationLock.isPresent() && registrationLockSalt.isPresent() && !Util.isEmpty(clientRegistrationLock)) {
|
||||||
|
return new AuthenticationCredentials(registrationLock.get(), registrationLockSalt.get()).verify(clientRegistrationLock);
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public StoredRegistrationLock forTime(long timestamp) {
|
||||||
|
return new StoredRegistrationLock(registrationLock, registrationLockSalt, timestamp);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.security.MessageDigest;
|
||||||
|
import java.time.Duration;
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Util;
|
||||||
|
|
||||||
|
public record StoredVerificationCode(String code,
|
||||||
|
long timestamp,
|
||||||
|
String pushCode,
|
||||||
|
@Nullable String twilioVerificationSid,
|
||||||
|
@Nullable byte[] sessionId) {
|
||||||
|
|
||||||
|
public static final Duration EXPIRATION = Duration.ofMinutes(10);
|
||||||
|
|
||||||
|
public boolean isValid(String theirCodeString) {
|
||||||
|
if (Util.isEmpty(code) || Util.isEmpty(theirCodeString)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] ourCode = code.getBytes();
|
||||||
|
byte[] theirCode = theirCodeString.getBytes();
|
||||||
|
|
||||||
|
return MessageDigest.isEqual(ourCode, theirCode);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class TurnToken {
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private String username;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private String password;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private List<String> urls;
|
||||||
|
|
||||||
|
public TurnToken(String username, String password, List<String> urls) {
|
||||||
|
this.username = username;
|
||||||
|
this.password = password;
|
||||||
|
this.urls = urls;
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
List<String> getUrls() {
|
||||||
|
return urls;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,68 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.TurnUriConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.dynamic.DynamicConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.dynamic.DynamicTurnConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.DynamicConfigurationManager;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Pair;
|
||||||
|
import org.whispersystems.textsecuregcm.util.WeightedRandomSelect;
|
||||||
|
|
||||||
|
import javax.crypto.Mac;
|
||||||
|
import javax.crypto.spec.SecretKeySpec;
|
||||||
|
import java.security.InvalidKeyException;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
|
import java.security.SecureRandom;
|
||||||
|
import java.util.Base64;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
public class TurnTokenGenerator {
|
||||||
|
|
||||||
|
private final DynamicConfigurationManager<DynamicConfiguration> dynamicConfiguration;
|
||||||
|
|
||||||
|
public TurnTokenGenerator(final DynamicConfigurationManager<DynamicConfiguration> config) {
|
||||||
|
this.dynamicConfiguration = config;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TurnToken generate(final String e164) {
|
||||||
|
try {
|
||||||
|
byte[] key = dynamicConfiguration.getConfiguration().getTurnConfiguration().getSecret().getBytes();
|
||||||
|
List<String> urls = urls(e164);
|
||||||
|
Mac mac = Mac.getInstance("HmacSHA1");
|
||||||
|
long validUntilSeconds = (System.currentTimeMillis() + TimeUnit.DAYS.toMillis(1)) / 1000;
|
||||||
|
long user = Math.abs(new SecureRandom().nextInt());
|
||||||
|
String userTime = validUntilSeconds + ":" + user;
|
||||||
|
|
||||||
|
mac.init(new SecretKeySpec(key, "HmacSHA1"));
|
||||||
|
String password = Base64.getEncoder().encodeToString(mac.doFinal(userTime.getBytes()));
|
||||||
|
|
||||||
|
return new TurnToken(userTime, password, urls);
|
||||||
|
} catch (NoSuchAlgorithmException | InvalidKeyException e) {
|
||||||
|
throw new AssertionError(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<String> urls(final String e164) {
|
||||||
|
final DynamicTurnConfiguration turnConfig = dynamicConfiguration.getConfiguration().getTurnConfiguration();
|
||||||
|
|
||||||
|
// Check if number is enrolled to test out specific turn servers
|
||||||
|
final Optional<TurnUriConfiguration> enrolled = turnConfig.getUriConfigs().stream()
|
||||||
|
.filter(config -> config.getEnrolledNumbers().contains(e164))
|
||||||
|
.findFirst();
|
||||||
|
if (enrolled.isPresent()) {
|
||||||
|
return enrolled.get().getUris();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise, select from turn server sets by weighted choice
|
||||||
|
return WeightedRandomSelect.select(turnConfig
|
||||||
|
.getUriConfigs()
|
||||||
|
.stream()
|
||||||
|
.map(c -> new Pair<List<String>, Long>(c.getUris(), c.getWeight())).toList());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import javax.crypto.Mac;
|
||||||
|
import javax.crypto.spec.SecretKeySpec;
|
||||||
|
import java.security.InvalidKeyException;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
|
import java.util.Base64;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
|
||||||
|
public class UnidentifiedAccessChecksum {
|
||||||
|
|
||||||
|
public static String generateFor(Optional<byte[]> unidentifiedAccessKey) {
|
||||||
|
try {
|
||||||
|
if (!unidentifiedAccessKey.isPresent()|| unidentifiedAccessKey.get().length != 16) return null;
|
||||||
|
|
||||||
|
Mac mac = Mac.getInstance("HmacSHA256");
|
||||||
|
mac.init(new SecretKeySpec(unidentifiedAccessKey.get(), "HmacSHA256"));
|
||||||
|
|
||||||
|
return Base64.getEncoder().encodeToString(mac.doFinal(new byte[32]));
|
||||||
|
} catch (NoSuchAlgorithmException | InvalidKeyException e) {
|
||||||
|
throw new AssertionError(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,38 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import org.glassfish.jersey.server.monitoring.ApplicationEvent;
|
||||||
|
import org.glassfish.jersey.server.monitoring.ApplicationEventListener;
|
||||||
|
import org.glassfish.jersey.server.monitoring.RequestEvent;
|
||||||
|
import org.glassfish.jersey.server.monitoring.RequestEventListener;
|
||||||
|
import org.whispersystems.textsecuregcm.push.ClientPresenceManager;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountsManager;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delegates request events to a listener that watches for intra-request changes that require websocket refreshes
|
||||||
|
*/
|
||||||
|
public class WebsocketRefreshApplicationEventListener implements ApplicationEventListener {
|
||||||
|
|
||||||
|
private final WebsocketRefreshRequestEventListener websocketRefreshRequestEventListener;
|
||||||
|
|
||||||
|
public WebsocketRefreshApplicationEventListener(final AccountsManager accountsManager,
|
||||||
|
final ClientPresenceManager clientPresenceManager) {
|
||||||
|
|
||||||
|
this.websocketRefreshRequestEventListener = new WebsocketRefreshRequestEventListener(clientPresenceManager,
|
||||||
|
new AuthEnablementRefreshRequirementProvider(accountsManager),
|
||||||
|
new PhoneNumberChangeRefreshRequirementProvider());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onEvent(final ApplicationEvent event) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RequestEventListener onRequest(final RequestEvent requestEvent) {
|
||||||
|
return websocketRefreshRequestEventListener;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,74 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import static org.whispersystems.textsecuregcm.metrics.MetricsUtil.name;
|
||||||
|
|
||||||
|
import io.micrometer.core.instrument.Counter;
|
||||||
|
import io.micrometer.core.instrument.Metrics;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
import javax.ws.rs.container.ResourceInfo;
|
||||||
|
import javax.ws.rs.core.Context;
|
||||||
|
import org.glassfish.jersey.server.monitoring.RequestEvent;
|
||||||
|
import org.glassfish.jersey.server.monitoring.RequestEvent.Type;
|
||||||
|
import org.glassfish.jersey.server.monitoring.RequestEventListener;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.whispersystems.textsecuregcm.push.ClientPresenceManager;
|
||||||
|
|
||||||
|
public class WebsocketRefreshRequestEventListener implements RequestEventListener {
|
||||||
|
|
||||||
|
private final ClientPresenceManager clientPresenceManager;
|
||||||
|
private final WebsocketRefreshRequirementProvider[] providers;
|
||||||
|
|
||||||
|
private static final Counter DISPLACED_ACCOUNTS = Metrics.counter(
|
||||||
|
name(WebsocketRefreshRequestEventListener.class, "displacedAccounts"));
|
||||||
|
|
||||||
|
private static final Counter DISPLACED_DEVICES = Metrics.counter(
|
||||||
|
name(WebsocketRefreshRequestEventListener.class, "displacedDevices"));
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(WebsocketRefreshRequestEventListener.class);
|
||||||
|
|
||||||
|
public WebsocketRefreshRequestEventListener(
|
||||||
|
final ClientPresenceManager clientPresenceManager,
|
||||||
|
final WebsocketRefreshRequirementProvider... providers) {
|
||||||
|
|
||||||
|
this.clientPresenceManager = clientPresenceManager;
|
||||||
|
this.providers = providers;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Context
|
||||||
|
private ResourceInfo resourceInfo;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onEvent(final RequestEvent event) {
|
||||||
|
if (event.getType() == Type.REQUEST_FILTERED) {
|
||||||
|
for (final WebsocketRefreshRequirementProvider provider : providers) {
|
||||||
|
provider.handleRequestFiltered(event);
|
||||||
|
}
|
||||||
|
} else if (event.getType() == Type.FINISHED) {
|
||||||
|
final AtomicInteger displacedDevices = new AtomicInteger(0);
|
||||||
|
|
||||||
|
Arrays.stream(providers)
|
||||||
|
.flatMap(provider -> provider.handleRequestFinished(event).stream())
|
||||||
|
.distinct()
|
||||||
|
.forEach(pair -> {
|
||||||
|
try {
|
||||||
|
displacedDevices.incrementAndGet();
|
||||||
|
clientPresenceManager.disconnectPresence(pair.first(), pair.second());
|
||||||
|
} catch (final Exception e) {
|
||||||
|
logger.error("Could not displace device presence", e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (displacedDevices.get() > 0) {
|
||||||
|
DISPLACED_ACCOUNTS.increment();
|
||||||
|
DISPLACED_DEVICES.increment(displacedDevices.get());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.auth;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
import org.glassfish.jersey.server.monitoring.RequestEvent;
|
||||||
|
import org.whispersystems.textsecuregcm.util.Pair;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A websocket refresh requirement provider watches for intra-request changes (e.g. to authentication status) that
|
||||||
|
* require a websocket refresh.
|
||||||
|
*/
|
||||||
|
public interface WebsocketRefreshRequirementProvider {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Processes a request after filters have run and the request has been mapped to a destination controller.
|
||||||
|
*
|
||||||
|
* @param requestEvent the request event to observe
|
||||||
|
*/
|
||||||
|
void handleRequestFiltered(RequestEvent requestEvent);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Processes a request after all normal request handling has been completed.
|
||||||
|
*
|
||||||
|
* @param requestEvent the request event to observe
|
||||||
|
* @return a list of pairs of account UUID/device ID pairs identifying websockets that need to be refreshed as a
|
||||||
|
* result of the observed request
|
||||||
|
*/
|
||||||
|
List<Pair<UUID, Long>> handleRequestFinished(RequestEvent requestEvent);
|
||||||
|
}
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.badges;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
import org.whispersystems.textsecuregcm.entities.Badge;
|
||||||
|
|
||||||
|
public interface BadgeTranslator {
|
||||||
|
Badge translate(List<Locale> acceptableLanguages, String badgeId);
|
||||||
|
}
|
||||||
@@ -0,0 +1,131 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.badges;
|
||||||
|
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import java.time.Clock;
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.time.Instant;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.ResourceBundle;
|
||||||
|
import java.util.function.Function;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import org.signal.i18n.HeaderControlledResourceBundleLookup;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.BadgeConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.BadgesConfiguration;
|
||||||
|
import org.whispersystems.textsecuregcm.entities.Badge;
|
||||||
|
import org.whispersystems.textsecuregcm.entities.BadgeSvg;
|
||||||
|
import org.whispersystems.textsecuregcm.entities.SelfBadge;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountBadge;
|
||||||
|
|
||||||
|
public class ConfiguredProfileBadgeConverter implements ProfileBadgeConverter, BadgeTranslator {
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
static final String BASE_NAME = "org.signal.badges.Badges";
|
||||||
|
|
||||||
|
private final Clock clock;
|
||||||
|
private final Map<String, BadgeConfiguration> knownBadges;
|
||||||
|
private final List<String> badgeIdsEnabledForAll;
|
||||||
|
private final HeaderControlledResourceBundleLookup headerControlledResourceBundleLookup;
|
||||||
|
|
||||||
|
public ConfiguredProfileBadgeConverter(
|
||||||
|
final Clock clock,
|
||||||
|
final BadgesConfiguration badgesConfiguration,
|
||||||
|
final HeaderControlledResourceBundleLookup headerControlledResourceBundleLookup) {
|
||||||
|
this.clock = clock;
|
||||||
|
this.knownBadges = badgesConfiguration.getBadges().stream()
|
||||||
|
.collect(Collectors.toMap(BadgeConfiguration::getId, Function.identity()));
|
||||||
|
this.badgeIdsEnabledForAll = badgesConfiguration.getBadgeIdsEnabledForAll();
|
||||||
|
this.headerControlledResourceBundleLookup = headerControlledResourceBundleLookup;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Badge translate(final List<Locale> acceptableLanguages, final String badgeId) {
|
||||||
|
final ResourceBundle resourceBundle = headerControlledResourceBundleLookup.getResourceBundle(BASE_NAME,
|
||||||
|
acceptableLanguages);
|
||||||
|
final BadgeConfiguration configuration = knownBadges.get(badgeId);
|
||||||
|
return newBadge(
|
||||||
|
false,
|
||||||
|
configuration.getId(),
|
||||||
|
configuration.getCategory(),
|
||||||
|
resourceBundle.getString(configuration.getId() + "_name"),
|
||||||
|
resourceBundle.getString(configuration.getId() + "_description"),
|
||||||
|
configuration.getSprites(),
|
||||||
|
configuration.getSvg(),
|
||||||
|
configuration.getSvgs(),
|
||||||
|
null,
|
||||||
|
false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Badge> convert(
|
||||||
|
final List<Locale> acceptableLanguages,
|
||||||
|
final List<AccountBadge> accountBadges,
|
||||||
|
final boolean isSelf) {
|
||||||
|
if (accountBadges.isEmpty() && badgeIdsEnabledForAll.isEmpty()) {
|
||||||
|
return List.of();
|
||||||
|
}
|
||||||
|
|
||||||
|
final Instant now = clock.instant();
|
||||||
|
final ResourceBundle resourceBundle = headerControlledResourceBundleLookup.getResourceBundle(BASE_NAME,
|
||||||
|
acceptableLanguages);
|
||||||
|
List<Badge> badges = accountBadges.stream()
|
||||||
|
.filter(accountBadge -> (isSelf || accountBadge.isVisible())
|
||||||
|
&& now.isBefore(accountBadge.getExpiration())
|
||||||
|
&& knownBadges.containsKey(accountBadge.getId()))
|
||||||
|
.map(accountBadge -> {
|
||||||
|
BadgeConfiguration configuration = knownBadges.get(accountBadge.getId());
|
||||||
|
return newBadge(
|
||||||
|
isSelf,
|
||||||
|
accountBadge.getId(),
|
||||||
|
configuration.getCategory(),
|
||||||
|
resourceBundle.getString(accountBadge.getId() + "_name"),
|
||||||
|
resourceBundle.getString(accountBadge.getId() + "_description"),
|
||||||
|
configuration.getSprites(),
|
||||||
|
configuration.getSvg(),
|
||||||
|
configuration.getSvgs(),
|
||||||
|
accountBadge.getExpiration(),
|
||||||
|
accountBadge.isVisible());
|
||||||
|
})
|
||||||
|
.collect(Collectors.toCollection(ArrayList::new));
|
||||||
|
badges.addAll(badgeIdsEnabledForAll.stream().filter(knownBadges::containsKey).map(id -> {
|
||||||
|
BadgeConfiguration configuration = knownBadges.get(id);
|
||||||
|
return newBadge(
|
||||||
|
isSelf,
|
||||||
|
id,
|
||||||
|
configuration.getCategory(),
|
||||||
|
resourceBundle.getString(id + "_name"),
|
||||||
|
resourceBundle.getString(id + "_description"),
|
||||||
|
configuration.getSprites(),
|
||||||
|
configuration.getSvg(),
|
||||||
|
configuration.getSvgs(),
|
||||||
|
now.plus(Duration.ofDays(1)),
|
||||||
|
true);
|
||||||
|
}).collect(Collectors.toList()));
|
||||||
|
return badges;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Badge newBadge(
|
||||||
|
final boolean isSelf,
|
||||||
|
final String id,
|
||||||
|
final String category,
|
||||||
|
final String name,
|
||||||
|
final String description,
|
||||||
|
final List<String> sprites,
|
||||||
|
final String svg,
|
||||||
|
final List<BadgeSvg> svgs,
|
||||||
|
final Instant expiration,
|
||||||
|
final boolean visible) {
|
||||||
|
if (isSelf) {
|
||||||
|
return new SelfBadge(id, category, name, description, sprites, svg, svgs, expiration, visible);
|
||||||
|
} else {
|
||||||
|
return new Badge(id, category, name, description, sprites, svg, svgs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.badges;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
|
||||||
|
public interface LevelTranslator {
|
||||||
|
String translate(List<Locale> acceptableLanguages, String badgeId);
|
||||||
|
}
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.badges;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
import org.whispersystems.textsecuregcm.entities.Badge;
|
||||||
|
import org.whispersystems.textsecuregcm.storage.AccountBadge;
|
||||||
|
|
||||||
|
public interface ProfileBadgeConverter {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts the {@link AccountBadge}s for an account into the objects
|
||||||
|
* that can be returned on a profile fetch.
|
||||||
|
*/
|
||||||
|
List<Badge> convert(List<Locale> acceptableLanguages, List<AccountBadge> accountBadges, boolean isSelf);
|
||||||
|
}
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.badges;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.ResourceBundle;
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
import org.signal.i18n.HeaderControlledResourceBundleLookup;
|
||||||
|
|
||||||
|
public class ResourceBundleLevelTranslator implements LevelTranslator {
|
||||||
|
|
||||||
|
private static final String BASE_NAME = "org.signal.subscriptions.Subscriptions";
|
||||||
|
|
||||||
|
private final HeaderControlledResourceBundleLookup headerControlledResourceBundleLookup;
|
||||||
|
|
||||||
|
public ResourceBundleLevelTranslator(
|
||||||
|
@Nonnull final HeaderControlledResourceBundleLookup headerControlledResourceBundleLookup) {
|
||||||
|
this.headerControlledResourceBundleLookup = Objects.requireNonNull(headerControlledResourceBundleLookup);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String translate(final List<Locale> acceptableLanguages, final String badgeId) {
|
||||||
|
final ResourceBundle resourceBundle = headerControlledResourceBundleLookup.getResourceBundle(BASE_NAME,
|
||||||
|
acceptableLanguages);
|
||||||
|
return resourceBundle.getString(badgeId);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import javax.validation.constraints.NotBlank;
|
||||||
|
|
||||||
|
public class AbusiveMessageFilterConfiguration {
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotBlank
|
||||||
|
private final String environment;
|
||||||
|
|
||||||
|
@JsonCreator
|
||||||
|
public AbusiveMessageFilterConfiguration(@JsonProperty("environment") final String environment) {
|
||||||
|
this.environment = environment;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getEnvironment() {
|
||||||
|
return environment;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
|
||||||
|
public class AccountDatabaseCrawlerConfiguration {
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private int chunkSize = 1000;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private long chunkIntervalMs = 8000L;
|
||||||
|
|
||||||
|
public int getChunkSize() {
|
||||||
|
return chunkSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getChunkIntervalMs() {
|
||||||
|
return chunkIntervalMs;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import org.whispersystems.textsecuregcm.configuration.DynamoDbTables.Table;
|
||||||
|
import javax.validation.constraints.NotBlank;
|
||||||
|
|
||||||
|
public class AccountsTableConfiguration extends Table {
|
||||||
|
|
||||||
|
private final String phoneNumberTableName;
|
||||||
|
private final String phoneNumberIdentifierTableName;
|
||||||
|
private final String usernamesTableName;
|
||||||
|
private final int scanPageSize;
|
||||||
|
|
||||||
|
@JsonCreator
|
||||||
|
public AccountsTableConfiguration(
|
||||||
|
@JsonProperty("tableName") final String tableName,
|
||||||
|
@JsonProperty("phoneNumberTableName") final String phoneNumberTableName,
|
||||||
|
@JsonProperty("phoneNumberIdentifierTableName") final String phoneNumberIdentifierTableName,
|
||||||
|
@JsonProperty("usernamesTableName") final String usernamesTableName,
|
||||||
|
@JsonProperty("scanPageSize") final int scanPageSize) {
|
||||||
|
|
||||||
|
super(tableName);
|
||||||
|
|
||||||
|
this.phoneNumberTableName = phoneNumberTableName;
|
||||||
|
this.phoneNumberIdentifierTableName = phoneNumberIdentifierTableName;
|
||||||
|
this.usernamesTableName = usernamesTableName;
|
||||||
|
this.scanPageSize = scanPageSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotBlank
|
||||||
|
public String getPhoneNumberTableName() {
|
||||||
|
return phoneNumberTableName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotBlank
|
||||||
|
public String getPhoneNumberIdentifierTableName() {
|
||||||
|
return phoneNumberIdentifierTableName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotBlank
|
||||||
|
public String getUsernamesTableName() {
|
||||||
|
return usernamesTableName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getScanPageSize() {
|
||||||
|
return scanPageSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2022 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import javax.validation.constraints.NotEmpty;
|
||||||
|
|
||||||
|
public record AdminEventLoggingConfiguration(
|
||||||
|
@NotEmpty String credentials,
|
||||||
|
@NotEmpty String projectId,
|
||||||
|
@NotEmpty String logName) {
|
||||||
|
}
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import javax.validation.constraints.NotEmpty;
|
||||||
|
|
||||||
|
|
||||||
|
public class ApnConfiguration {
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String teamId;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String keyId;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String signingKey;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String bundleId;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private boolean sandbox = false;
|
||||||
|
|
||||||
|
public String getTeamId() {
|
||||||
|
return teamId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getKeyId() {
|
||||||
|
return keyId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSigningKey() {
|
||||||
|
return signingKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getBundleId() {
|
||||||
|
return bundleId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isSandboxEnabled() {
|
||||||
|
return sandbox;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
|
||||||
|
import javax.validation.constraints.NotEmpty;
|
||||||
|
|
||||||
|
public class AppConfigConfiguration {
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotEmpty
|
||||||
|
private String application;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotEmpty
|
||||||
|
private String environment;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotEmpty
|
||||||
|
private String configuration;
|
||||||
|
|
||||||
|
public String getApplication() {
|
||||||
|
return application;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getEnvironment() {
|
||||||
|
return environment;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getConfigurationName() {
|
||||||
|
return configuration;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import javax.validation.constraints.NotEmpty;
|
||||||
|
|
||||||
|
public class AwsAttachmentsConfiguration {
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String accessKey;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String accessSecret;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String bucket;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String region;
|
||||||
|
|
||||||
|
public String getAccessKey() {
|
||||||
|
return accessKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getAccessSecret() {
|
||||||
|
return accessSecret;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getBucket() {
|
||||||
|
return bucket;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getRegion() {
|
||||||
|
return region;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,68 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import java.util.List;
|
||||||
|
import javax.validation.constraints.NotEmpty;
|
||||||
|
import javax.validation.constraints.NotNull;
|
||||||
|
import org.whispersystems.textsecuregcm.entities.BadgeSvg;
|
||||||
|
import org.whispersystems.textsecuregcm.util.ExactlySize;
|
||||||
|
|
||||||
|
public class BadgeConfiguration {
|
||||||
|
public static final String CATEGORY_TESTING = "testing";
|
||||||
|
|
||||||
|
private final String id;
|
||||||
|
private final String category;
|
||||||
|
private final List<String> sprites;
|
||||||
|
private final String svg;
|
||||||
|
private final List<BadgeSvg> svgs;
|
||||||
|
|
||||||
|
@JsonCreator
|
||||||
|
public BadgeConfiguration(
|
||||||
|
@JsonProperty("id") final String id,
|
||||||
|
@JsonProperty("category") final String category,
|
||||||
|
@JsonProperty("sprites") final List<String> sprites,
|
||||||
|
@JsonProperty("svg") final String svg,
|
||||||
|
@JsonProperty("svgs") final List<BadgeSvg> svgs) {
|
||||||
|
this.id = id;
|
||||||
|
this.category = category;
|
||||||
|
this.sprites = sprites;
|
||||||
|
this.svg = svg;
|
||||||
|
this.svgs = svgs;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
public String getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
public String getCategory() {
|
||||||
|
return category;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@ExactlySize(6)
|
||||||
|
public List<String> getSprites() {
|
||||||
|
return sprites;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
public String getSvg() {
|
||||||
|
return svg;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public List<BadgeSvg> getSvgs() {
|
||||||
|
return svgs;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isTestBadge() {
|
||||||
|
return CATEGORY_TESTING.equals(category);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonIgnore;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonSetter;
|
||||||
|
import com.fasterxml.jackson.annotation.Nulls;
|
||||||
|
import io.dropwizard.validation.ValidationMethod;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import javax.validation.Valid;
|
||||||
|
import javax.validation.constraints.NotNull;
|
||||||
|
|
||||||
|
public class BadgesConfiguration {
|
||||||
|
private final List<BadgeConfiguration> badges;
|
||||||
|
private final List<String> badgeIdsEnabledForAll;
|
||||||
|
private final Map<Long, String> receiptLevels;
|
||||||
|
|
||||||
|
@JsonCreator
|
||||||
|
public BadgesConfiguration(
|
||||||
|
@JsonProperty("badges") @JsonSetter(nulls = Nulls.AS_EMPTY) final List<BadgeConfiguration> badges,
|
||||||
|
@JsonProperty("badgeIdsEnabledForAll") @JsonSetter(nulls = Nulls.AS_EMPTY) final List<String> badgeIdsEnabledForAll,
|
||||||
|
@JsonProperty("receiptLevels") @JsonSetter(nulls = Nulls.AS_EMPTY) final Map<Long, String> receiptLevels) {
|
||||||
|
this.badges = Objects.requireNonNull(badges);
|
||||||
|
this.badgeIdsEnabledForAll = Objects.requireNonNull(badgeIdsEnabledForAll);
|
||||||
|
this.receiptLevels = Objects.requireNonNull(receiptLevels);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
public List<BadgeConfiguration> getBadges() {
|
||||||
|
return badges;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
public List<String> getBadgeIdsEnabledForAll() {
|
||||||
|
return badgeIdsEnabledForAll;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
public Map<Long, String> getReceiptLevels() {
|
||||||
|
return receiptLevels;
|
||||||
|
}
|
||||||
|
|
||||||
|
@JsonIgnore
|
||||||
|
@ValidationMethod(message = "contains receipt level mappings that are not configured badges")
|
||||||
|
public boolean isAllReceiptLevelsConfigured() {
|
||||||
|
final Set<String> badgeNames = badges.stream().map(BadgeConfiguration::getId).collect(Collectors.toSet());
|
||||||
|
return badgeNames.containsAll(receiptLevels.values());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,58 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonCreator;
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import javax.validation.Valid;
|
||||||
|
import javax.validation.constraints.DecimalMin;
|
||||||
|
import javax.validation.constraints.NotEmpty;
|
||||||
|
import javax.validation.constraints.NotNull;
|
||||||
|
import org.whispersystems.textsecuregcm.util.ExactlySize;
|
||||||
|
|
||||||
|
public class BoostConfiguration {
|
||||||
|
|
||||||
|
private final long level;
|
||||||
|
private final Duration expiration;
|
||||||
|
private final Map<String, List<BigDecimal>> currencies;
|
||||||
|
private final String badge;
|
||||||
|
|
||||||
|
@JsonCreator
|
||||||
|
public BoostConfiguration(
|
||||||
|
@JsonProperty("level") long level,
|
||||||
|
@JsonProperty("expiration") Duration expiration,
|
||||||
|
@JsonProperty("currencies") Map<String, List<BigDecimal>> currencies,
|
||||||
|
@JsonProperty("badge") String badge) {
|
||||||
|
this.level = level;
|
||||||
|
this.expiration = expiration;
|
||||||
|
this.currencies = currencies;
|
||||||
|
this.badge = badge;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getLevel() {
|
||||||
|
return level;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
public Duration getExpiration() {
|
||||||
|
return expiration;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Valid
|
||||||
|
@NotNull
|
||||||
|
public Map<@NotEmpty String, @Valid @ExactlySize(6) List<@DecimalMin("0.01") @NotNull BigDecimal>> getCurrencies() {
|
||||||
|
return currencies;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
public String getBadge() {
|
||||||
|
return badge;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import javax.validation.constraints.NotEmpty;
|
||||||
|
|
||||||
|
public class CdnConfiguration {
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String accessKey;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String accessSecret;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String bucket;
|
||||||
|
|
||||||
|
@NotEmpty
|
||||||
|
@JsonProperty
|
||||||
|
private String region;
|
||||||
|
|
||||||
|
public String getAccessKey() {
|
||||||
|
return accessKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getAccessSecret() {
|
||||||
|
return accessSecret;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getBucket() {
|
||||||
|
return bucket;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getRegion() {
|
||||||
|
return region;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,123 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import io.github.resilience4j.circuitbreaker.CircuitBreakerConfig;
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import javax.validation.constraints.Max;
|
||||||
|
import javax.validation.constraints.Min;
|
||||||
|
import javax.validation.constraints.NotNull;
|
||||||
|
|
||||||
|
public class CircuitBreakerConfiguration {
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
@Min(1)
|
||||||
|
@Max(100)
|
||||||
|
private int failureRateThreshold = 50;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
@Min(1)
|
||||||
|
private int permittedNumberOfCallsInHalfOpenState = 10;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
@Min(1)
|
||||||
|
private int slidingWindowSize = 100;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
@Min(1)
|
||||||
|
private int slidingWindowMinimumNumberOfCalls = 100;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
@Min(1)
|
||||||
|
private long waitDurationInOpenStateInSeconds = 10;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
private List<String> ignoredExceptions = Collections.emptyList();
|
||||||
|
|
||||||
|
|
||||||
|
public int getFailureRateThreshold() {
|
||||||
|
return failureRateThreshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getPermittedNumberOfCallsInHalfOpenState() {
|
||||||
|
return permittedNumberOfCallsInHalfOpenState;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getSlidingWindowSize() {
|
||||||
|
return slidingWindowSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getSlidingWindowMinimumNumberOfCalls() {
|
||||||
|
return slidingWindowMinimumNumberOfCalls;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getWaitDurationInOpenStateInSeconds() {
|
||||||
|
return waitDurationInOpenStateInSeconds;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Class<?>> getIgnoredExceptions() {
|
||||||
|
return ignoredExceptions.stream()
|
||||||
|
.map(name -> {
|
||||||
|
try {
|
||||||
|
return Class.forName(name);
|
||||||
|
} catch (final ClassNotFoundException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public void setFailureRateThreshold(int failureRateThreshold) {
|
||||||
|
this.failureRateThreshold = failureRateThreshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public void setSlidingWindowSize(int size) {
|
||||||
|
this.slidingWindowSize = size;
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public void setSlidingWindowMinimumNumberOfCalls(int size) {
|
||||||
|
this.slidingWindowMinimumNumberOfCalls = size;
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public void setPermittedNumberOfCallsInHalfOpenState(int size) {
|
||||||
|
this.permittedNumberOfCallsInHalfOpenState = size;
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public void setWaitDurationInOpenStateInSeconds(int seconds) {
|
||||||
|
this.waitDurationInOpenStateInSeconds = seconds;
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public void setIgnoredExceptions(final List<String> ignoredExceptions) {
|
||||||
|
this.ignoredExceptions = ignoredExceptions;
|
||||||
|
}
|
||||||
|
|
||||||
|
public CircuitBreakerConfig toCircuitBreakerConfig() {
|
||||||
|
return CircuitBreakerConfig.custom()
|
||||||
|
.failureRateThreshold(getFailureRateThreshold())
|
||||||
|
.ignoreExceptions(getIgnoredExceptions().toArray(new Class[0]))
|
||||||
|
.permittedNumberOfCallsInHalfOpenState(getPermittedNumberOfCallsInHalfOpenState())
|
||||||
|
.waitDurationInOpenState(Duration.ofSeconds(getWaitDurationInOpenStateInSeconds()))
|
||||||
|
.slidingWindow(getSlidingWindowSize(), getSlidingWindowMinimumNumberOfCalls(),
|
||||||
|
CircuitBreakerConfig.SlidingWindowType.COUNT_BASED)
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2020 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
|
||||||
|
import javax.validation.constraints.NotNull;
|
||||||
|
|
||||||
|
import io.dropwizard.db.DataSourceFactory;
|
||||||
|
|
||||||
|
public class DatabaseConfiguration extends DataSourceFactory {
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@JsonProperty
|
||||||
|
private CircuitBreakerConfiguration circuitBreaker = new CircuitBreakerConfiguration();
|
||||||
|
|
||||||
|
public CircuitBreakerConfiguration getCircuitBreakerConfiguration() {
|
||||||
|
return circuitBreaker;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2021 Signal Messenger, LLC
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.whispersystems.textsecuregcm.configuration;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import io.micrometer.datadog.DatadogConfig;
|
||||||
|
import javax.validation.constraints.Min;
|
||||||
|
import javax.validation.constraints.NotBlank;
|
||||||
|
import javax.validation.constraints.NotNull;
|
||||||
|
import java.time.Duration;
|
||||||
|
|
||||||
|
public class DatadogConfiguration implements DatadogConfig {
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotBlank
|
||||||
|
private String apiKey;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotNull
|
||||||
|
private Duration step = Duration.ofSeconds(10);
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@NotBlank
|
||||||
|
private String environment;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
@Min(1)
|
||||||
|
private int batchSize = 5_000;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String apiKey() {
|
||||||
|
return apiKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Duration step() {
|
||||||
|
return step;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getEnvironment() {
|
||||||
|
return environment;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int batchSize() {
|
||||||
|
return batchSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String hostTag() {
|
||||||
|
return "host";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String get(final String key) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user