mirror of
https://code.briarproject.org/briar/briar.git
synced 2026-02-12 18:59:06 +01:00
Compare commits
638 Commits
ci-debug
...
control-po
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5751958eaf | ||
|
|
ea006d21f9 | ||
|
|
0237df937f | ||
|
|
4c1fd94c67 | ||
|
|
295e97c2c7 | ||
|
|
39f65fcdaf | ||
|
|
ce04f89f8b | ||
|
|
51f1cb5e9e | ||
|
|
419f37a4a9 | ||
|
|
3d94ffb714 | ||
|
|
bc8bb08853 | ||
|
|
cc67a8fcdd | ||
|
|
f8cf88e6cd | ||
|
|
bc58c47a22 | ||
|
|
aa6879c48e | ||
|
|
4d26628f2a | ||
|
|
abaa70da99 | ||
|
|
6435c3520c | ||
|
|
b5c4c7ae61 | ||
|
|
5d96da3547 | ||
|
|
ed842f781a | ||
|
|
5e30e5e1de | ||
|
|
ce52a36db1 | ||
|
|
f5ef87b34b | ||
|
|
4c6f68c255 | ||
|
|
ae09b4c607 | ||
|
|
880d77922e | ||
|
|
1c227e81e4 | ||
|
|
541acad29a | ||
|
|
60f71648f3 | ||
|
|
270b8af39f | ||
|
|
31d3324701 | ||
|
|
dbe46d60fd | ||
|
|
d10ab96955 | ||
|
|
b2841e245a | ||
|
|
9ccd8d1602 | ||
|
|
ac3942975e | ||
|
|
b6455d40a7 | ||
|
|
2815ad042d | ||
|
|
2055961534 | ||
|
|
741eae34e9 | ||
|
|
50bd4cce6b | ||
|
|
0a5a8310fc | ||
|
|
cc43d5982a | ||
|
|
50675473ce | ||
|
|
de852b2a9f | ||
|
|
b7c712116b | ||
|
|
7dd4897c8c | ||
|
|
7469c0f5e3 | ||
|
|
144ea0c2fc | ||
|
|
a917ebdc76 | ||
|
|
2a389c74dc | ||
|
|
ef16d096f1 | ||
|
|
679455888b | ||
|
|
d4372ddae7 | ||
|
|
c3ef990a94 | ||
|
|
8ae9b7f5a2 | ||
|
|
106d80ef76 | ||
|
|
9422ba2718 | ||
|
|
8343f5c2db | ||
|
|
371c7efb04 | ||
|
|
92d67645ab | ||
|
|
a20e868970 | ||
|
|
dd853f6718 | ||
|
|
16a8ad996a | ||
|
|
e27885f0c8 | ||
|
|
f6ef48bf90 | ||
|
|
e282ca763d | ||
|
|
71016382dc | ||
|
|
d004933fae | ||
|
|
37512c50d8 | ||
|
|
0b61a5d40a | ||
|
|
5dd320f282 | ||
|
|
2a21db5fb6 | ||
|
|
b023593a2c | ||
|
|
5ccf2cae1f | ||
|
|
c2cb89ab73 | ||
|
|
b342759e06 | ||
|
|
93d99b0111 | ||
|
|
61e8d576d2 | ||
|
|
75c37a258e | ||
|
|
e964dae64b | ||
|
|
986d884b40 | ||
|
|
9557afabc6 | ||
|
|
ebe6b0d4c0 | ||
|
|
6e83fb7aef | ||
|
|
7a5ec2af12 | ||
|
|
ce1fde496c | ||
|
|
4b62c51fbf | ||
|
|
226ed3dd73 | ||
|
|
ab07dfb32c | ||
|
|
20c51c1aa4 | ||
|
|
232c2129a7 | ||
|
|
3620edbfc9 | ||
|
|
ad71d69149 | ||
|
|
f73f8ca7e7 | ||
|
|
16c701a71a | ||
|
|
8183b7b26a | ||
|
|
bd48c97eab | ||
|
|
925dc29a1f | ||
|
|
91777fd942 | ||
|
|
fbce8f81c7 | ||
|
|
d7c72c4d68 | ||
|
|
4faf535801 | ||
|
|
526ef7c6d8 | ||
|
|
798dff1a03 | ||
|
|
a4336776c9 | ||
|
|
418451cbd9 | ||
|
|
045fcfc5fa | ||
|
|
ef998577db | ||
|
|
a53345a3c9 | ||
|
|
ed8c09282d | ||
|
|
42197b5b5c | ||
|
|
374fc7035b | ||
|
|
9b796c7cc3 | ||
|
|
532edff642 | ||
|
|
6857252471 | ||
|
|
c229e19452 | ||
|
|
42bca09d16 | ||
|
|
9eacbfa659 | ||
|
|
f14e546dc6 | ||
|
|
684c64a1d9 | ||
|
|
6fdab959b1 | ||
|
|
c8487483ff | ||
|
|
a159b23dc0 | ||
|
|
5070a27a83 | ||
|
|
9ce73a6840 | ||
|
|
6e9928f20f | ||
|
|
b31d61afc5 | ||
|
|
5a99cb93cc | ||
|
|
d0bbebd25e | ||
|
|
4307d26606 | ||
|
|
0089c1ac6d | ||
|
|
2a7aac4930 | ||
|
|
a37b6d81ed | ||
|
|
1d09a6708a | ||
|
|
d3b6f484c8 | ||
|
|
039c6edb66 | ||
|
|
8b9f89eab2 | ||
|
|
1e2c17b170 | ||
|
|
a994966095 | ||
|
|
2bea581654 | ||
|
|
87377666aa | ||
|
|
9d07b2e141 | ||
|
|
5c312b49e2 | ||
|
|
f56efe45cd | ||
|
|
2332a58681 | ||
|
|
8c6dfaa196 | ||
|
|
3cfb04b60d | ||
|
|
e85fbfb952 | ||
|
|
80ee35d926 | ||
|
|
4796902b9c | ||
|
|
149e67c0f7 | ||
|
|
1d5214117f | ||
|
|
b8f248ca9c | ||
|
|
dfb71a03a5 | ||
|
|
961fdc8e72 | ||
|
|
c3d44663cd | ||
|
|
0081472489 | ||
|
|
cdf4f3a24b | ||
|
|
fb1d8e860f | ||
|
|
a3c526ec9a | ||
|
|
dee488d06d | ||
|
|
b29c7d8022 | ||
|
|
0725d207ec | ||
|
|
5a7599a88d | ||
|
|
59cd98db81 | ||
|
|
768488eb04 | ||
|
|
a6b1ad48c3 | ||
|
|
77299a68ed | ||
|
|
5e5705c73b | ||
|
|
e6229a3a13 | ||
|
|
5fbacb4ee4 | ||
|
|
c7f4e976ed | ||
|
|
419f2d966a | ||
|
|
d6c18db9e9 | ||
|
|
8fe49d9961 | ||
|
|
f536cfdab8 | ||
|
|
4d594acad5 | ||
|
|
800dfed5c1 | ||
|
|
54b823e401 | ||
|
|
52ec56d690 | ||
|
|
7b3afcca99 | ||
|
|
a22d03d028 | ||
|
|
d857338ad0 | ||
|
|
a5c9e7c74d | ||
|
|
8a4a343147 | ||
|
|
dcd5e34c6b | ||
|
|
7b22d3b84d | ||
|
|
c8fa23273f | ||
|
|
fbe5df8938 | ||
|
|
008cf95741 | ||
|
|
3eb066a836 | ||
|
|
674b29af25 | ||
|
|
b8ca5ab557 | ||
|
|
6e17709f46 | ||
|
|
726d90145c | ||
|
|
165211eb9b | ||
|
|
868c61e5d6 | ||
|
|
798bb6d4f7 | ||
|
|
bc352a2dc6 | ||
|
|
ce7d6d3db5 | ||
|
|
61276c81d2 | ||
|
|
c09abdb088 | ||
|
|
45a11badd5 | ||
|
|
152ac3df43 | ||
|
|
dd5ad86db8 | ||
|
|
10e9fb308d | ||
|
|
de8e95692a | ||
|
|
d6b52cf4ec | ||
|
|
8a839fb5e4 | ||
|
|
fbf8642edb | ||
|
|
ade6a14342 | ||
|
|
d500ff81c3 | ||
|
|
3053e3cfa7 | ||
|
|
6964a67ca3 | ||
|
|
f4b06e1fb3 | ||
|
|
4db075d643 | ||
|
|
78a8ae6b8e | ||
|
|
7866037d02 | ||
|
|
35716051fb | ||
|
|
6cafea836f | ||
|
|
bd0fd229c6 | ||
|
|
ea05a5c703 | ||
|
|
4103eaf639 | ||
|
|
753a25bc2a | ||
|
|
8f4c3c4528 | ||
|
|
636a7dfe72 | ||
|
|
08e99edd42 | ||
|
|
e28bc475df | ||
|
|
88276a4d44 | ||
|
|
f9987c89df | ||
|
|
2c8cb8301f | ||
|
|
c00ee80f0f | ||
|
|
3bfedfdc3d | ||
|
|
ecb63d1acb | ||
|
|
b24914408d | ||
|
|
5ede63edd5 | ||
|
|
4e523c5fbc | ||
|
|
cf79ed5633 | ||
|
|
0a0a6a6369 | ||
|
|
4784980e7b | ||
|
|
3bfa5e2081 | ||
|
|
f2b09deac4 | ||
|
|
ad4729b2f9 | ||
|
|
50cc0a6815 | ||
|
|
32c8ac6576 | ||
|
|
c12422d949 | ||
|
|
3841713c18 | ||
|
|
79232eb558 | ||
|
|
d02b30e751 | ||
|
|
043662a092 | ||
|
|
efc85fb88f | ||
|
|
8b3983ef9e | ||
|
|
6766fb76b2 | ||
|
|
7f74bd1c38 | ||
|
|
951ee30b95 | ||
|
|
c386a0f5eb | ||
|
|
52c778dce3 | ||
|
|
e846a13f50 | ||
|
|
23e9b119d1 | ||
|
|
e6f380296f | ||
|
|
794fb9686b | ||
|
|
bb22b9db10 | ||
|
|
d4f015d054 | ||
|
|
41e5d8900c | ||
|
|
c3cb966009 | ||
|
|
f964d1ef07 | ||
|
|
cccaeeda6c | ||
|
|
483106e00c | ||
|
|
934f14ef31 | ||
|
|
e3abff5ad8 | ||
|
|
391732b239 | ||
|
|
4738bfdd85 | ||
|
|
be1ca89309 | ||
|
|
866be99179 | ||
|
|
48822e2133 | ||
|
|
6883c5caa9 | ||
|
|
8b709969ab | ||
|
|
fe94b65b3b | ||
|
|
f54df1d787 | ||
|
|
a7e826ccf5 | ||
|
|
845eb3262b | ||
|
|
0a46ad439f | ||
|
|
d14d93ea35 | ||
|
|
12a1cf8f8b | ||
|
|
fb2ab861db | ||
|
|
aa15b68d24 | ||
|
|
7059f376f1 | ||
|
|
9313c191c1 | ||
|
|
7746364ae9 | ||
|
|
7429857b28 | ||
|
|
4db64f51a8 | ||
|
|
78172038ef | ||
|
|
7d0c418877 | ||
|
|
5ae4f8f6cb | ||
|
|
45dd10db9d | ||
|
|
5cc8c268ca | ||
|
|
93a6bf2f52 | ||
|
|
852dd46a1b | ||
|
|
4a42e767d3 | ||
|
|
8547b4dc91 | ||
|
|
f6c8a8cec7 | ||
|
|
db8796049e | ||
|
|
e1f31ad381 | ||
|
|
3a15e47ddd | ||
|
|
cf616905d6 | ||
|
|
d3f774f339 | ||
|
|
6c7c488892 | ||
|
|
3fe7aae97e | ||
|
|
ecf417c93b | ||
|
|
c1785c5b13 | ||
|
|
fb2c321a3d | ||
|
|
6e3adc0874 | ||
|
|
cdbe2a00f5 | ||
|
|
a51dc7e0d5 | ||
|
|
0ee7465429 | ||
|
|
7c202189a2 | ||
|
|
da3b2c1591 | ||
|
|
6c5e8ce4cf | ||
|
|
ca700d8d23 | ||
|
|
db11dad61e | ||
|
|
69e7366226 | ||
|
|
90b7b4e67f | ||
|
|
d29812a42b | ||
|
|
baf64e1129 | ||
|
|
88adfabe09 | ||
|
|
6565172e10 | ||
|
|
7447468ce5 | ||
|
|
2db2a1a208 | ||
|
|
0d7e4feaf2 | ||
|
|
eb3983f6b2 | ||
|
|
e2ce49c30e | ||
|
|
adc6fb2fd5 | ||
|
|
969150bff0 | ||
|
|
8fc622f85d | ||
|
|
22eed91019 | ||
|
|
fcb88ed58c | ||
|
|
0d940fc7d7 | ||
|
|
ca6dc33cdd | ||
|
|
53da13794f | ||
|
|
2ab03f48cc | ||
|
|
436f45554d | ||
|
|
51209b5eec | ||
|
|
822597b4c6 | ||
|
|
7c01bc59c0 | ||
|
|
825d342f9b | ||
|
|
34955fecbb | ||
|
|
5c28b60a6b | ||
|
|
389b2b5b8e | ||
|
|
78abfa3698 | ||
|
|
9c4fb4fd34 | ||
|
|
3d6a336f6d | ||
|
|
e47d41596a | ||
|
|
8cf54bcedb | ||
|
|
89d5145665 | ||
|
|
0706498b03 | ||
|
|
b296500e7a | ||
|
|
60a8b03344 | ||
|
|
ae16a93522 | ||
|
|
c9a2ff71ae | ||
|
|
16f4c60a56 | ||
|
|
76121eb871 | ||
|
|
47c91a96ae | ||
|
|
14befb957b | ||
|
|
4b7a81177c | ||
|
|
b464fe1653 | ||
|
|
09c6f09805 | ||
|
|
a93093182d | ||
|
|
9515e93857 | ||
|
|
efe15df940 | ||
|
|
de611857cf | ||
|
|
8935ec2c2e | ||
|
|
bd00fb1c04 | ||
|
|
3192015cfd | ||
|
|
e776ee02b0 | ||
|
|
c0553ec11f | ||
|
|
75a871a2f8 | ||
|
|
d6d3d5acef | ||
|
|
a361a2613c | ||
|
|
b68dbd6a75 | ||
|
|
f1e89a3ff4 | ||
|
|
056c23167d | ||
|
|
79d5612645 | ||
|
|
a030f92275 | ||
|
|
b3615b4a77 | ||
|
|
8a15fb242a | ||
|
|
e3686186ee | ||
|
|
18ae388137 | ||
|
|
775031e893 | ||
|
|
9f91b91a4f | ||
|
|
280f3ba1fc | ||
|
|
66619fd3a4 | ||
|
|
c7eb0cbb6d | ||
|
|
1617a95bb9 | ||
|
|
6f54718756 | ||
|
|
ea749f2128 | ||
|
|
b4b0d3daa6 | ||
|
|
609c90f57e | ||
|
|
5cf68fa134 | ||
|
|
61c9c6b8eb | ||
|
|
e97608da40 | ||
|
|
0bb80b1a15 | ||
|
|
bda52ea548 | ||
|
|
cf033dc29d | ||
|
|
c12cedc371 | ||
|
|
4b5e9bd64f | ||
|
|
8d55911dab | ||
|
|
e381f83512 | ||
|
|
e4c7f13832 | ||
|
|
b089a204d3 | ||
|
|
85fcb34997 | ||
|
|
d6d132a9cf | ||
|
|
98d1ea7730 | ||
|
|
159fd34c0c | ||
|
|
9e7a387ea4 | ||
|
|
138e520e6c | ||
|
|
5783c1dfd8 | ||
|
|
348968018a | ||
|
|
33c509cd1f | ||
|
|
bea77151bd | ||
|
|
787e62345f | ||
|
|
48f6a3b91f | ||
|
|
a798e25bf2 | ||
|
|
31e4045cf7 | ||
|
|
5334a8c9ca | ||
|
|
d11f1d2805 | ||
|
|
0d1ebddcd2 | ||
|
|
6c296c1348 | ||
|
|
87701e5f07 | ||
|
|
3aae01d152 | ||
|
|
bc298ba68a | ||
|
|
2623eaa149 | ||
|
|
7359b6942a | ||
|
|
3bcc532b4b | ||
|
|
4d08c69779 | ||
|
|
a6cd8937f7 | ||
|
|
e8566906ef | ||
|
|
929102ed60 | ||
|
|
3b871f5932 | ||
|
|
b972d1fc13 | ||
|
|
ccbeee60a7 | ||
|
|
074b10e177 | ||
|
|
031516ccce | ||
|
|
7d2f1abb94 | ||
|
|
00b9c76bb8 | ||
|
|
4d9fab85cb | ||
|
|
bd2514a299 | ||
|
|
e795efc7fc | ||
|
|
6691d2164f | ||
|
|
a384450c36 | ||
|
|
b375e9873c | ||
|
|
cb30c3885a | ||
|
|
6ee81eb24c | ||
|
|
c14ebe82ce | ||
|
|
00e9f894b1 | ||
|
|
499c586a59 | ||
|
|
64f9ce7306 | ||
|
|
39478a7914 | ||
|
|
112e71a9cb | ||
|
|
5650bef310 | ||
|
|
2a87171c49 | ||
|
|
071d961ed1 | ||
|
|
cb9efc5fec | ||
|
|
f9e292f734 | ||
|
|
15cb5409e7 | ||
|
|
fd07dc006d | ||
|
|
cc87c4e37d | ||
|
|
4a10e876f6 | ||
|
|
fad0057c4a | ||
|
|
5aabfcea9a | ||
|
|
f7d928c774 | ||
|
|
bd983d9796 | ||
|
|
de8d1b7d96 | ||
|
|
9155f62d0b | ||
|
|
86684e228a | ||
|
|
9615eff649 | ||
|
|
9381d46f51 | ||
|
|
e4a3a1ad40 | ||
|
|
905dc2a662 | ||
|
|
c2b7f85b8e | ||
|
|
ae81eb3737 | ||
|
|
60d949c342 | ||
|
|
1c90e64894 | ||
|
|
f0e2d5281f | ||
|
|
c7522dae1f | ||
|
|
097d14b9a1 | ||
|
|
0491c3cace | ||
|
|
cbae13feca | ||
|
|
b7c8859c82 | ||
|
|
2e120f752c | ||
|
|
031eac54c5 | ||
|
|
2c2596afdd | ||
|
|
d1be14effe | ||
|
|
b56e7ab07d | ||
|
|
089e9589ed | ||
|
|
660ba16a14 | ||
|
|
b101c4b636 | ||
|
|
fdfddd2667 | ||
|
|
296546544f | ||
|
|
ad579a6ba3 | ||
|
|
90e82357ba | ||
|
|
b3b40753d8 | ||
|
|
e60df3cece | ||
|
|
da3cb95151 | ||
|
|
c27885072f | ||
|
|
6557d564c9 | ||
|
|
53edcaf3e9 | ||
|
|
5122c961b4 | ||
|
|
f83b9244d4 | ||
|
|
81292967e0 | ||
|
|
b72f6b4fc3 | ||
|
|
488be49c93 | ||
|
|
90db45817a | ||
|
|
81863b9db6 | ||
|
|
da069adb57 | ||
|
|
46425b09fa | ||
|
|
41e1a436c9 | ||
|
|
989394d18b | ||
|
|
b6b3f9c292 | ||
|
|
a52547f73b | ||
|
|
24f823a3ce | ||
|
|
a045d7d306 | ||
|
|
a29d5efd93 | ||
|
|
37cd1cdddf | ||
|
|
4f495bb4d3 | ||
|
|
1a70200b65 | ||
|
|
6925dfcbdd | ||
|
|
7d479063a9 | ||
|
|
2309e73216 | ||
|
|
4b325f797b | ||
|
|
9be83c3cc7 | ||
|
|
86f650503b | ||
|
|
d430b4fd2d | ||
|
|
fcf7cf72ea | ||
|
|
b78dfea95f | ||
|
|
183fe08565 | ||
|
|
7e32697696 | ||
|
|
29758b174a | ||
|
|
61e18f104e | ||
|
|
ffeca8817f | ||
|
|
59fae2fa3c | ||
|
|
2d9345c018 | ||
|
|
817df9c75a | ||
|
|
745515457e | ||
|
|
ba5928218a | ||
|
|
9476782ced | ||
|
|
74445acb55 | ||
|
|
e32771f964 | ||
|
|
d7bf1ee374 | ||
|
|
10bee05856 | ||
|
|
fc626d0921 | ||
|
|
30f87e626a | ||
|
|
a0d91da569 | ||
|
|
c90a72617e | ||
|
|
8813bc36af | ||
|
|
049cf3ad27 | ||
|
|
de8a6b23e5 | ||
|
|
30193a240b | ||
|
|
a52ad8b4cc | ||
|
|
6a1a8b6872 | ||
|
|
50ad42a0a2 | ||
|
|
08005bdf56 | ||
|
|
e32cc3af6d | ||
|
|
28a68ff625 | ||
|
|
2bef2ac828 | ||
|
|
b2febbc6e9 | ||
|
|
e12601dd08 | ||
|
|
3388682dda | ||
|
|
74e4a9cbdf | ||
|
|
8ad3047f87 | ||
|
|
0cffaf8646 | ||
|
|
7b116f15df | ||
|
|
ced0f72fba | ||
|
|
24c030f06f | ||
|
|
a3fa15e90e | ||
|
|
57841be447 | ||
|
|
c5d374af04 | ||
|
|
8d592ad2ee | ||
|
|
055c381cc9 | ||
|
|
1d259bd51c | ||
|
|
de63141997 | ||
|
|
dee8f68477 | ||
|
|
59048f106a | ||
|
|
da7cf4af28 | ||
|
|
0d4cf4db68 | ||
|
|
9efd2d113a | ||
|
|
8e6cd12f07 | ||
|
|
3a49ca0d97 | ||
|
|
c03868e800 | ||
|
|
d6c129e919 | ||
|
|
271efdd2bc | ||
|
|
ad4e8d51e9 | ||
|
|
eb19c6e08d | ||
|
|
83bfeb9075 | ||
|
|
428501cf5f | ||
|
|
d8b04edcd0 | ||
|
|
0bc07cd0c1 | ||
|
|
cb3026959a | ||
|
|
48933637d8 | ||
|
|
5626f3d761 | ||
|
|
0fce224d88 | ||
|
|
3db35f7061 | ||
|
|
751375035d | ||
|
|
27a169c6e2 | ||
|
|
d4a4351786 | ||
|
|
fbd38dbb94 | ||
|
|
cd4897e6c9 | ||
|
|
d84e176bb4 | ||
|
|
da8b49bec2 | ||
|
|
6c8cc79d87 | ||
|
|
a5271eee29 | ||
|
|
4dfc96996d | ||
|
|
3139f308a2 | ||
|
|
cc6daffa61 | ||
|
|
f08f441f5f | ||
|
|
83886c78f1 | ||
|
|
5ed0e9efec | ||
|
|
169c59349e | ||
|
|
764f60b3fe | ||
|
|
e51c437a06 | ||
|
|
9fbf740ba7 | ||
|
|
db7686ea52 | ||
|
|
7fe21e079f | ||
|
|
be72e624a3 | ||
|
|
d9e9741112 | ||
|
|
656ca8d67a | ||
|
|
d3e44358a4 | ||
|
|
920a1d0431 | ||
|
|
4b9a9771f8 | ||
|
|
d64252aaf3 | ||
|
|
825ed451a3 | ||
|
|
bffd78d404 | ||
|
|
04ffff0953 | ||
|
|
21f95ed9af | ||
|
|
c8b516196c | ||
|
|
941a0cccc3 | ||
|
|
9b17836595 |
@@ -11,8 +11,8 @@ test:
|
|||||||
- .gradle/caches
|
- .gradle/caches
|
||||||
|
|
||||||
script:
|
script:
|
||||||
- ./gradlew --no-daemon animalSnifferMain animalSnifferTest
|
- ./gradlew --no-daemon -Djava.security.egd=file:/dev/urandom animalSnifferMain animalSnifferTest
|
||||||
- ./gradlew --no-daemon test
|
- ./gradlew --no-daemon -Djava.security.egd=file:/dev/urandom test
|
||||||
|
|
||||||
after_script:
|
after_script:
|
||||||
# these file change every time but should not be cached
|
# these file change every time but should not be cached
|
||||||
|
|||||||
9
.idea/codeStyles/Project.xml
generated
9
.idea/codeStyles/Project.xml
generated
@@ -36,6 +36,9 @@
|
|||||||
<option name="JD_ALIGN_PARAM_COMMENTS" value="false" />
|
<option name="JD_ALIGN_PARAM_COMMENTS" value="false" />
|
||||||
<option name="JD_ALIGN_EXCEPTION_COMMENTS" value="false" />
|
<option name="JD_ALIGN_EXCEPTION_COMMENTS" value="false" />
|
||||||
</JavaCodeStyleSettings>
|
</JavaCodeStyleSettings>
|
||||||
|
<JetCodeStyleSettings>
|
||||||
|
<option name="CODE_STYLE_DEFAULTS" value="KOTLIN_OFFICIAL" />
|
||||||
|
</JetCodeStyleSettings>
|
||||||
<Objective-C-extensions>
|
<Objective-C-extensions>
|
||||||
<file>
|
<file>
|
||||||
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Import" />
|
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Import" />
|
||||||
@@ -257,5 +260,11 @@
|
|||||||
</rules>
|
</rules>
|
||||||
</arrangement>
|
</arrangement>
|
||||||
</codeStyleSettings>
|
</codeStyleSettings>
|
||||||
|
<codeStyleSettings language="kotlin">
|
||||||
|
<option name="CODE_STYLE_DEFAULTS" value="KOTLIN_OFFICIAL" />
|
||||||
|
<option name="PARAMETER_ANNOTATION_WRAP" value="1" />
|
||||||
|
<option name="VARIABLE_ANNOTATION_WRAP" value="1" />
|
||||||
|
<option name="ENUM_CONSTANTS_WRAP" value="1" />
|
||||||
|
</codeStyleSettings>
|
||||||
</code_scheme>
|
</code_scheme>
|
||||||
</component>
|
</component>
|
||||||
20
.idea/runConfigurations/All_in_briar_headless.xml
generated
Normal file
20
.idea/runConfigurations/All_in_briar_headless.xml
generated
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<component name="ProjectRunConfigurationManager">
|
||||||
|
<configuration default="false" name="All in briar-headless" type="AndroidJUnit" factoryName="Android JUnit" nameIsGenerated="true">
|
||||||
|
<module name="briar-headless" />
|
||||||
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
|
<option name="PACKAGE_NAME" value="org.briarproject.briar.headless" />
|
||||||
|
<option name="MAIN_CLASS_NAME" value="" />
|
||||||
|
<option name="METHOD_NAME" value="" />
|
||||||
|
<option name="TEST_OBJECT" value="package" />
|
||||||
|
<option name="VM_PARAMETERS" value="" />
|
||||||
|
<option name="PARAMETERS" value="" />
|
||||||
|
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/briar-headless" />
|
||||||
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
|
<value defaultName="singleModule" />
|
||||||
|
</option>
|
||||||
|
<patterns />
|
||||||
|
<method />
|
||||||
|
</configuration>
|
||||||
|
</component>
|
||||||
3
.idea/runConfigurations/All_tests.xml
generated
3
.idea/runConfigurations/All_tests.xml
generated
@@ -22,8 +22,9 @@
|
|||||||
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in bramble-api" run_configuration_type="AndroidJUnit" />
|
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in bramble-api" run_configuration_type="AndroidJUnit" />
|
||||||
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in bramble-core" run_configuration_type="AndroidJUnit" />
|
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in bramble-core" run_configuration_type="AndroidJUnit" />
|
||||||
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in bramble-android" run_configuration_type="AndroidJUnit" />
|
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in bramble-android" run_configuration_type="AndroidJUnit" />
|
||||||
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in bramble-j2se" run_configuration_type="AndroidJUnit" />
|
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in bramble-java" run_configuration_type="AndroidJUnit" />
|
||||||
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in briar-core" run_configuration_type="AndroidJUnit" />
|
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All tests in briar-core" run_configuration_type="AndroidJUnit" />
|
||||||
|
<option name="RunConfigurationTask" enabled="true" run_configuration_name="All in briar-headless" run_configuration_type="AndroidJUnit" />
|
||||||
</method>
|
</method>
|
||||||
</configuration>
|
</configuration>
|
||||||
</component>
|
</component>
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
<component name="ProjectRunConfigurationManager">
|
<component name="ProjectRunConfigurationManager">
|
||||||
<configuration default="false" name="All tests in bramble-android" type="AndroidJUnit" factoryName="Android JUnit">
|
<configuration default="false" name="All tests in bramble-android" type="AndroidJUnit" factoryName="Android JUnit">
|
||||||
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
|
||||||
<module name="bramble-android" />
|
<module name="bramble-android" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH" />
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
@@ -11,12 +10,10 @@
|
|||||||
<option name="VM_PARAMETERS" value="-ea" />
|
<option name="VM_PARAMETERS" value="-ea" />
|
||||||
<option name="PARAMETERS" value="" />
|
<option name="PARAMETERS" value="" />
|
||||||
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/bramble-android" />
|
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/bramble-android" />
|
||||||
<option name="ENV_VARIABLES" />
|
|
||||||
<option name="PASS_PARENT_ENVS" value="true" />
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
<option name="TEST_SEARCH_SCOPE">
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
<value defaultName="singleModule" />
|
<value defaultName="singleModule" />
|
||||||
</option>
|
</option>
|
||||||
<envs />
|
|
||||||
<patterns />
|
<patterns />
|
||||||
<method />
|
<method />
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
<component name="ProjectRunConfigurationManager">
|
<component name="ProjectRunConfigurationManager">
|
||||||
<configuration default="false" name="All tests in bramble-api" type="AndroidJUnit" factoryName="Android JUnit">
|
<configuration default="false" name="All tests in bramble-api" type="AndroidJUnit" factoryName="Android JUnit">
|
||||||
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
|
||||||
<module name="bramble-api" />
|
<module name="bramble-api" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH" />
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
@@ -11,12 +10,10 @@
|
|||||||
<option name="VM_PARAMETERS" value="-ea" />
|
<option name="VM_PARAMETERS" value="-ea" />
|
||||||
<option name="PARAMETERS" value="" />
|
<option name="PARAMETERS" value="" />
|
||||||
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/bramble-api" />
|
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/bramble-api" />
|
||||||
<option name="ENV_VARIABLES" />
|
|
||||||
<option name="PASS_PARENT_ENVS" value="true" />
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
<option name="TEST_SEARCH_SCOPE">
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
<value defaultName="singleModule" />
|
<value defaultName="singleModule" />
|
||||||
</option>
|
</option>
|
||||||
<envs />
|
|
||||||
<patterns />
|
<patterns />
|
||||||
<method />
|
<method />
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
<component name="ProjectRunConfigurationManager">
|
<component name="ProjectRunConfigurationManager">
|
||||||
<configuration default="false" name="All tests in bramble-core" type="AndroidJUnit" factoryName="Android JUnit">
|
<configuration default="false" name="All tests in bramble-core" type="AndroidJUnit" factoryName="Android JUnit">
|
||||||
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
|
||||||
<module name="bramble-core" />
|
<module name="bramble-core" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH" />
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
@@ -11,12 +10,10 @@
|
|||||||
<option name="VM_PARAMETERS" value="-ea" />
|
<option name="VM_PARAMETERS" value="-ea" />
|
||||||
<option name="PARAMETERS" value="" />
|
<option name="PARAMETERS" value="" />
|
||||||
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/bramble-core" />
|
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/bramble-core" />
|
||||||
<option name="ENV_VARIABLES" />
|
|
||||||
<option name="PASS_PARENT_ENVS" value="true" />
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
<option name="TEST_SEARCH_SCOPE">
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
<value defaultName="singleModule" />
|
<value defaultName="singleModule" />
|
||||||
</option>
|
</option>
|
||||||
<envs />
|
|
||||||
<patterns />
|
<patterns />
|
||||||
<method />
|
<method />
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
<component name="ProjectRunConfigurationManager">
|
<component name="ProjectRunConfigurationManager">
|
||||||
<configuration default="false" name="All tests in bramble-j2se" type="AndroidJUnit" factoryName="Android JUnit">
|
<configuration default="false" name="All tests in bramble-java" type="AndroidJUnit" factoryName="Android JUnit">
|
||||||
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
<module name="bramble-java" />
|
||||||
<module name="bramble-j2se" />
|
|
||||||
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH" />
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
<option name="PACKAGE_NAME" value="" />
|
<option name="PACKAGE_NAME" value="" />
|
||||||
@@ -10,13 +9,11 @@
|
|||||||
<option name="TEST_OBJECT" value="package" />
|
<option name="TEST_OBJECT" value="package" />
|
||||||
<option name="VM_PARAMETERS" value="-ea -Djava.library.path=libs" />
|
<option name="VM_PARAMETERS" value="-ea -Djava.library.path=libs" />
|
||||||
<option name="PARAMETERS" value="" />
|
<option name="PARAMETERS" value="" />
|
||||||
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/bramble-j2se" />
|
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/bramble-java" />
|
||||||
<option name="ENV_VARIABLES" />
|
|
||||||
<option name="PASS_PARENT_ENVS" value="true" />
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
<option name="TEST_SEARCH_SCOPE">
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
<value defaultName="singleModule" />
|
<value defaultName="singleModule" />
|
||||||
</option>
|
</option>
|
||||||
<envs />
|
|
||||||
<patterns />
|
<patterns />
|
||||||
<method />
|
<method />
|
||||||
</configuration>
|
</configuration>
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
<component name="ProjectRunConfigurationManager">
|
<component name="ProjectRunConfigurationManager">
|
||||||
<configuration default="false" name="All tests in briar-android" type="AndroidJUnit" factoryName="Android JUnit">
|
<configuration default="false" name="All tests in briar-android" type="AndroidJUnit" factoryName="Android JUnit">
|
||||||
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
|
||||||
<module name="briar-android" />
|
<module name="briar-android" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH" />
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
@@ -11,12 +10,10 @@
|
|||||||
<option name="VM_PARAMETERS" value="-ea" />
|
<option name="VM_PARAMETERS" value="-ea" />
|
||||||
<option name="PARAMETERS" value="" />
|
<option name="PARAMETERS" value="" />
|
||||||
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/briar-android" />
|
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/briar-android" />
|
||||||
<option name="ENV_VARIABLES" />
|
|
||||||
<option name="PASS_PARENT_ENVS" value="true" />
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
<option name="TEST_SEARCH_SCOPE">
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
<value defaultName="singleModule" />
|
<value defaultName="singleModule" />
|
||||||
</option>
|
</option>
|
||||||
<envs />
|
|
||||||
<patterns />
|
<patterns />
|
||||||
<method />
|
<method />
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
<component name="ProjectRunConfigurationManager">
|
<component name="ProjectRunConfigurationManager">
|
||||||
<configuration default="false" name="All tests in briar-core" type="AndroidJUnit" factoryName="Android JUnit">
|
<configuration default="false" name="All tests in briar-core" type="AndroidJUnit" factoryName="Android JUnit">
|
||||||
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
|
||||||
<module name="briar-core" />
|
<module name="briar-core" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH" />
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
@@ -11,12 +10,10 @@
|
|||||||
<option name="VM_PARAMETERS" value="-ea" />
|
<option name="VM_PARAMETERS" value="-ea" />
|
||||||
<option name="PARAMETERS" value="" />
|
<option name="PARAMETERS" value="" />
|
||||||
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/briar-core" />
|
<option name="WORKING_DIRECTORY" value="file://$PROJECT_DIR$/briar-core" />
|
||||||
<option name="ENV_VARIABLES" />
|
|
||||||
<option name="PASS_PARENT_ENVS" value="true" />
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
<option name="TEST_SEARCH_SCOPE">
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
<value defaultName="singleModule" />
|
<value defaultName="singleModule" />
|
||||||
</option>
|
</option>
|
||||||
<envs />
|
|
||||||
<patterns />
|
<patterns />
|
||||||
<method />
|
<method />
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|||||||
3
.idea/runConfigurations/H2_Performance_Test.xml
generated
3
.idea/runConfigurations/H2_Performance_Test.xml
generated
@@ -1,6 +1,5 @@
|
|||||||
<component name="ProjectRunConfigurationManager">
|
<component name="ProjectRunConfigurationManager">
|
||||||
<configuration default="false" name="H2 Performance Test" type="AndroidJUnit" factoryName="Android JUnit">
|
<configuration default="false" name="H2 Performance Test" type="AndroidJUnit" factoryName="Android JUnit">
|
||||||
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
|
||||||
<module name="bramble-core" />
|
<module name="bramble-core" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH" />
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
@@ -11,12 +10,10 @@
|
|||||||
<option name="VM_PARAMETERS" value="-ea" />
|
<option name="VM_PARAMETERS" value="-ea" />
|
||||||
<option name="PARAMETERS" value="" />
|
<option name="PARAMETERS" value="" />
|
||||||
<option name="WORKING_DIRECTORY" value="" />
|
<option name="WORKING_DIRECTORY" value="" />
|
||||||
<option name="ENV_VARIABLES" />
|
|
||||||
<option name="PASS_PARENT_ENVS" value="true" />
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
<option name="TEST_SEARCH_SCOPE">
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
<value defaultName="singleModule" />
|
<value defaultName="singleModule" />
|
||||||
</option>
|
</option>
|
||||||
<envs />
|
|
||||||
<patterns />
|
<patterns />
|
||||||
<method />
|
<method />
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
<component name="ProjectRunConfigurationManager">
|
<component name="ProjectRunConfigurationManager">
|
||||||
<configuration default="false" name="HyperSQL Performance Test" type="AndroidJUnit" factoryName="Android JUnit">
|
<configuration default="false" name="HyperSQL Performance Test" type="AndroidJUnit" factoryName="Android JUnit">
|
||||||
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
|
||||||
<module name="bramble-core" />
|
<module name="bramble-core" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
<option name="ALTERNATIVE_JRE_PATH" />
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
@@ -11,12 +10,10 @@
|
|||||||
<option name="VM_PARAMETERS" value="-ea" />
|
<option name="VM_PARAMETERS" value="-ea" />
|
||||||
<option name="PARAMETERS" value="" />
|
<option name="PARAMETERS" value="" />
|
||||||
<option name="WORKING_DIRECTORY" value="" />
|
<option name="WORKING_DIRECTORY" value="" />
|
||||||
<option name="ENV_VARIABLES" />
|
|
||||||
<option name="PASS_PARENT_ENVS" value="true" />
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
<option name="TEST_SEARCH_SCOPE">
|
<option name="TEST_SEARCH_SCOPE">
|
||||||
<value defaultName="singleModule" />
|
<value defaultName="singleModule" />
|
||||||
</option>
|
</option>
|
||||||
<envs />
|
|
||||||
<patterns />
|
<patterns />
|
||||||
<method />
|
<method />
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|||||||
17
.idea/runConfigurations/briar_headless.xml
generated
Normal file
17
.idea/runConfigurations/briar_headless.xml
generated
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<component name="ProjectRunConfigurationManager">
|
||||||
|
<configuration default="false" name="briar-headless" type="JetRunConfigurationType" factoryName="Kotlin" singleton="true">
|
||||||
|
<extension name="coverage" enabled="false" merge="false" sample_coverage="true" runner="idea" />
|
||||||
|
<option name="VM_PARAMETERS" value="" />
|
||||||
|
<option name="PROGRAM_PARAMETERS" value="-v" />
|
||||||
|
<option name="ALTERNATIVE_JRE_PATH_ENABLED" value="false" />
|
||||||
|
<option name="ALTERNATIVE_JRE_PATH" />
|
||||||
|
<option name="PASS_PARENT_ENVS" value="true" />
|
||||||
|
<option name="MAIN_CLASS_NAME" value="org.briarproject.briar.headless.MainKt" />
|
||||||
|
<option name="WORKING_DIRECTORY" value="" />
|
||||||
|
<module name="briar-headless" />
|
||||||
|
<envs />
|
||||||
|
<method>
|
||||||
|
<option name="Gradle.BeforeRunTask" enabled="true" tasks="jar" externalProjectPath="$PROJECT_DIR$/briar-headless" vmOptions="" scriptParameters="" />
|
||||||
|
</method>
|
||||||
|
</configuration>
|
||||||
|
</component>
|
||||||
@@ -1,16 +1,18 @@
|
|||||||
|
import com.android.build.gradle.tasks.MergeResources
|
||||||
|
|
||||||
apply plugin: 'com.android.library'
|
apply plugin: 'com.android.library'
|
||||||
apply plugin: 'witness'
|
apply plugin: 'witness'
|
||||||
apply from: 'witness.gradle'
|
apply from: 'witness.gradle'
|
||||||
|
|
||||||
android {
|
android {
|
||||||
compileSdkVersion 27
|
compileSdkVersion 28
|
||||||
buildToolsVersion '27.0.3'
|
buildToolsVersion '28.0.3'
|
||||||
|
|
||||||
defaultConfig {
|
defaultConfig {
|
||||||
minSdkVersion 14
|
minSdkVersion 14
|
||||||
targetSdkVersion 26
|
targetSdkVersion 26
|
||||||
versionCode 10013
|
versionCode 10106
|
||||||
versionName "1.0.13"
|
versionName "1.1.6"
|
||||||
consumerProguardFiles 'proguard-rules.txt'
|
consumerProguardFiles 'proguard-rules.txt'
|
||||||
|
|
||||||
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
|
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
|
||||||
@@ -28,9 +30,10 @@ configurations {
|
|||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation project(path: ':bramble-core', configuration: 'default')
|
implementation project(path: ':bramble-core', configuration: 'default')
|
||||||
tor 'org.briarproject:tor-android:0.2.9.16@zip'
|
tor 'org.briarproject:tor-android:0.3.5.8@zip'
|
||||||
|
tor 'org.briarproject:obfs4proxy-android:0.0.9@zip'
|
||||||
|
|
||||||
annotationProcessor 'com.google.dagger:dagger-compiler:2.0.2'
|
annotationProcessor 'com.google.dagger:dagger-compiler:2.19'
|
||||||
|
|
||||||
compileOnly 'javax.annotation:jsr250-api:1.0'
|
compileOnly 'javax.annotation:jsr250-api:1.0'
|
||||||
|
|
||||||
@@ -39,19 +42,29 @@ dependencies {
|
|||||||
testImplementation "org.jmock:jmock:2.8.2"
|
testImplementation "org.jmock:jmock:2.8.2"
|
||||||
testImplementation "org.jmock:jmock-junit4:2.8.2"
|
testImplementation "org.jmock:jmock-junit4:2.8.2"
|
||||||
testImplementation "org.jmock:jmock-legacy:2.8.2"
|
testImplementation "org.jmock:jmock-legacy:2.8.2"
|
||||||
testImplementation "org.hamcrest:hamcrest-library:1.3"
|
|
||||||
testImplementation "org.hamcrest:hamcrest-core:1.3"
|
|
||||||
|
|
||||||
androidTestImplementation project(path: ':bramble-api', configuration: 'testOutput')
|
|
||||||
androidTestImplementation project(path: ':bramble-core', configuration: 'testOutput')
|
|
||||||
androidTestImplementation 'com.android.support.test:runner:1.0.2'
|
|
||||||
androidTestAnnotationProcessor 'com.google.dagger:dagger-compiler:2.0.2'
|
|
||||||
androidTestCompileOnly 'javax.annotation:jsr250-api:1.0'
|
|
||||||
}
|
}
|
||||||
|
|
||||||
project.afterEvaluate {
|
def torBinariesDir = 'src/main/res/raw'
|
||||||
copy {
|
|
||||||
from configurations.tor.collect { zipTree(it) }
|
task cleanTorBinaries {
|
||||||
into 'src/main/res/raw'
|
doLast {
|
||||||
|
delete fileTree(torBinariesDir) { include '*.zip' }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
clean.dependsOn cleanTorBinaries
|
||||||
|
|
||||||
|
task unpackTorBinaries {
|
||||||
|
doLast {
|
||||||
|
copy {
|
||||||
|
from configurations.tor.collect { zipTree(it) }
|
||||||
|
into torBinariesDir
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dependsOn cleanTorBinaries
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.withType(MergeResources) {
|
||||||
|
inputs.dir torBinariesDir
|
||||||
|
dependsOn unpackTorBinaries
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,20 +0,0 @@
|
|||||||
package org.briarproject.bramble.test;
|
|
||||||
|
|
||||||
import android.app.Application;
|
|
||||||
|
|
||||||
import javax.inject.Singleton;
|
|
||||||
|
|
||||||
import dagger.Module;
|
|
||||||
import dagger.Provides;
|
|
||||||
|
|
||||||
import static android.support.test.InstrumentationRegistry.getTargetContext;
|
|
||||||
|
|
||||||
@Module
|
|
||||||
class ApplicationModule {
|
|
||||||
|
|
||||||
@Provides
|
|
||||||
@Singleton
|
|
||||||
Application provideApplication() {
|
|
||||||
return (Application) getTargetContext().getApplicationContext();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
package org.briarproject.bramble;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.battery.AndroidBatteryModule;
|
||||||
|
import org.briarproject.bramble.network.AndroidNetworkModule;
|
||||||
|
|
||||||
|
public interface BrambleAndroidEagerSingletons {
|
||||||
|
|
||||||
|
void inject(AndroidBatteryModule.EagerSingletons init);
|
||||||
|
|
||||||
|
void inject(AndroidNetworkModule.EagerSingletons init);
|
||||||
|
}
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
package org.briarproject.bramble;
|
package org.briarproject.bramble;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.battery.AndroidBatteryModule;
|
||||||
import org.briarproject.bramble.network.AndroidNetworkModule;
|
import org.briarproject.bramble.network.AndroidNetworkModule;
|
||||||
import org.briarproject.bramble.plugin.tor.CircumventionModule;
|
import org.briarproject.bramble.plugin.tor.CircumventionModule;
|
||||||
import org.briarproject.bramble.system.AndroidSystemModule;
|
import org.briarproject.bramble.system.AndroidSystemModule;
|
||||||
@@ -7,10 +8,15 @@ import org.briarproject.bramble.system.AndroidSystemModule;
|
|||||||
import dagger.Module;
|
import dagger.Module;
|
||||||
|
|
||||||
@Module(includes = {
|
@Module(includes = {
|
||||||
|
AndroidBatteryModule.class,
|
||||||
AndroidNetworkModule.class,
|
AndroidNetworkModule.class,
|
||||||
AndroidSystemModule.class,
|
AndroidSystemModule.class,
|
||||||
CircumventionModule.class
|
CircumventionModule.class
|
||||||
})
|
})
|
||||||
public class BrambleAndroidModule {
|
public class BrambleAndroidModule {
|
||||||
|
|
||||||
|
public static void initEagerSingletons(BrambleAndroidEagerSingletons c) {
|
||||||
|
c.inject(new AndroidBatteryModule.EagerSingletons());
|
||||||
|
c.inject(new AndroidNetworkModule.EagerSingletons());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,11 +12,15 @@ import org.briarproject.bramble.api.identity.IdentityManager;
|
|||||||
import org.briarproject.bramble.util.IoUtils;
|
import org.briarproject.bramble.util.IoUtils;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
|
|
||||||
|
import static android.os.Build.VERSION.SDK_INT;
|
||||||
|
|
||||||
class AndroidAccountManager extends AccountManagerImpl
|
class AndroidAccountManager extends AccountManagerImpl
|
||||||
implements AccountManager {
|
implements AccountManager {
|
||||||
|
|
||||||
@@ -89,20 +93,42 @@ class AndroidAccountManager extends AccountManagerImpl
|
|||||||
LOG.warning("Could not clear shared preferences");
|
LOG.warning("Could not clear shared preferences");
|
||||||
}
|
}
|
||||||
// Delete files, except lib and shared_prefs directories
|
// Delete files, except lib and shared_prefs directories
|
||||||
|
Set<File> files = new HashSet<>();
|
||||||
File dataDir = new File(appContext.getApplicationInfo().dataDir);
|
File dataDir = new File(appContext.getApplicationInfo().dataDir);
|
||||||
File[] children = dataDir.listFiles();
|
@Nullable
|
||||||
if (children == null) {
|
File[] fileArray = dataDir.listFiles();
|
||||||
|
if (fileArray == null) {
|
||||||
LOG.warning("Could not list files in app data dir");
|
LOG.warning("Could not list files in app data dir");
|
||||||
} else {
|
} else {
|
||||||
for (File child : children) {
|
for (File file : fileArray) {
|
||||||
String name = child.getName();
|
String name = file.getName();
|
||||||
if (!name.equals("lib") && !name.equals("shared_prefs")) {
|
if (!name.equals("lib") && !name.equals("shared_prefs")) {
|
||||||
IoUtils.deleteFileOrDir(child);
|
files.add(file);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
files.add(appContext.getFilesDir());
|
||||||
|
files.add(appContext.getCacheDir());
|
||||||
|
addIfNotNull(files, appContext.getExternalCacheDir());
|
||||||
|
if (SDK_INT >= 19) {
|
||||||
|
for (File file : appContext.getExternalCacheDirs()) {
|
||||||
|
addIfNotNull(files, file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (SDK_INT >= 21) {
|
||||||
|
for (File file : appContext.getExternalMediaDirs()) {
|
||||||
|
addIfNotNull(files, file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (File file : files) {
|
||||||
|
IoUtils.deleteFileOrDir(file);
|
||||||
|
}
|
||||||
// Recreate the cache dir as some OpenGL drivers expect it to exist
|
// Recreate the cache dir as some OpenGL drivers expect it to exist
|
||||||
if (!new File(dataDir, "cache").mkdir())
|
if (!new File(dataDir, "cache").mkdirs())
|
||||||
LOG.warning("Could not recreate cache dir");
|
LOG.warning("Could not recreate cache dir");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void addIfNotNull(Set<File> files, @Nullable File file) {
|
||||||
|
if (file != null) files.add(file);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,81 @@
|
|||||||
|
package org.briarproject.bramble.battery;
|
||||||
|
|
||||||
|
import android.app.Application;
|
||||||
|
import android.content.BroadcastReceiver;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.IntentFilter;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.battery.BatteryManager;
|
||||||
|
import org.briarproject.bramble.api.battery.event.BatteryEvent;
|
||||||
|
import org.briarproject.bramble.api.event.EventBus;
|
||||||
|
import org.briarproject.bramble.api.lifecycle.Service;
|
||||||
|
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
|
||||||
|
import static android.content.Intent.ACTION_BATTERY_CHANGED;
|
||||||
|
import static android.content.Intent.ACTION_POWER_CONNECTED;
|
||||||
|
import static android.content.Intent.ACTION_POWER_DISCONNECTED;
|
||||||
|
import static android.os.BatteryManager.EXTRA_PLUGGED;
|
||||||
|
import static java.util.logging.Level.INFO;
|
||||||
|
import static java.util.logging.Logger.getLogger;
|
||||||
|
|
||||||
|
class AndroidBatteryManager implements BatteryManager, Service {
|
||||||
|
|
||||||
|
private static final Logger LOG =
|
||||||
|
getLogger(AndroidBatteryManager.class.getName());
|
||||||
|
|
||||||
|
private final Context appContext;
|
||||||
|
private final EventBus eventBus;
|
||||||
|
private final AtomicBoolean used = new AtomicBoolean(false);
|
||||||
|
|
||||||
|
private volatile BroadcastReceiver batteryReceiver = null;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
AndroidBatteryManager(Application app, EventBus eventBus) {
|
||||||
|
this.appContext = app.getApplicationContext();
|
||||||
|
this.eventBus = eventBus;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isCharging() {
|
||||||
|
// Get the sticky intent for ACTION_BATTERY_CHANGED
|
||||||
|
IntentFilter filter = new IntentFilter(ACTION_BATTERY_CHANGED);
|
||||||
|
Intent i = appContext.registerReceiver(null, filter);
|
||||||
|
if (i == null) return false;
|
||||||
|
int status = i.getIntExtra(EXTRA_PLUGGED, 0);
|
||||||
|
return status != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void startService() {
|
||||||
|
if (used.getAndSet(true)) throw new IllegalStateException();
|
||||||
|
batteryReceiver = new BatteryReceiver();
|
||||||
|
IntentFilter filter = new IntentFilter();
|
||||||
|
filter.addAction(ACTION_POWER_CONNECTED);
|
||||||
|
filter.addAction(ACTION_POWER_DISCONNECTED);
|
||||||
|
appContext.registerReceiver(batteryReceiver, filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void stopService() {
|
||||||
|
if (batteryReceiver != null)
|
||||||
|
appContext.unregisterReceiver(batteryReceiver);
|
||||||
|
}
|
||||||
|
|
||||||
|
private class BatteryReceiver extends BroadcastReceiver {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onReceive(Context ctx, Intent i) {
|
||||||
|
String action = i.getAction();
|
||||||
|
if (LOG.isLoggable(INFO)) LOG.info("Received broadcast " + action);
|
||||||
|
if (ACTION_POWER_CONNECTED.equals(action))
|
||||||
|
eventBus.broadcast(new BatteryEvent(true));
|
||||||
|
else if (ACTION_POWER_DISCONNECTED.equals(action))
|
||||||
|
eventBus.broadcast(new BatteryEvent(false));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
package org.briarproject.bramble.battery;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.battery.BatteryManager;
|
||||||
|
import org.briarproject.bramble.api.lifecycle.LifecycleManager;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
|
import javax.inject.Singleton;
|
||||||
|
|
||||||
|
import dagger.Module;
|
||||||
|
import dagger.Provides;
|
||||||
|
|
||||||
|
@Module
|
||||||
|
public class AndroidBatteryModule {
|
||||||
|
|
||||||
|
public static class EagerSingletons {
|
||||||
|
@Inject
|
||||||
|
BatteryManager batteryManager;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Provides
|
||||||
|
@Singleton
|
||||||
|
BatteryManager provideBatteryManager(LifecycleManager lifecycleManager,
|
||||||
|
AndroidBatteryManager batteryManager) {
|
||||||
|
lifecycleManager.registerService(batteryManager);
|
||||||
|
return batteryManager;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -3,6 +3,7 @@ package org.briarproject.bramble.network;
|
|||||||
import org.briarproject.bramble.api.lifecycle.LifecycleManager;
|
import org.briarproject.bramble.api.lifecycle.LifecycleManager;
|
||||||
import org.briarproject.bramble.api.network.NetworkManager;
|
import org.briarproject.bramble.api.network.NetworkManager;
|
||||||
|
|
||||||
|
import javax.inject.Inject;
|
||||||
import javax.inject.Singleton;
|
import javax.inject.Singleton;
|
||||||
|
|
||||||
import dagger.Module;
|
import dagger.Module;
|
||||||
@@ -11,6 +12,11 @@ import dagger.Provides;
|
|||||||
@Module
|
@Module
|
||||||
public class AndroidNetworkModule {
|
public class AndroidNetworkModule {
|
||||||
|
|
||||||
|
public static class EagerSingletons {
|
||||||
|
@Inject
|
||||||
|
NetworkManager networkManager;
|
||||||
|
}
|
||||||
|
|
||||||
@Provides
|
@Provides
|
||||||
@Singleton
|
@Singleton
|
||||||
NetworkManager provideNetworkManager(LifecycleManager lifecycleManager,
|
NetworkManager provideNetworkManager(LifecycleManager lifecycleManager,
|
||||||
|
|||||||
@@ -16,18 +16,27 @@ import org.briarproject.bramble.api.plugin.PluginException;
|
|||||||
import org.briarproject.bramble.api.plugin.duplex.DuplexPluginCallback;
|
import org.briarproject.bramble.api.plugin.duplex.DuplexPluginCallback;
|
||||||
import org.briarproject.bramble.api.plugin.duplex.DuplexTransportConnection;
|
import org.briarproject.bramble.api.plugin.duplex.DuplexTransportConnection;
|
||||||
import org.briarproject.bramble.api.system.AndroidExecutor;
|
import org.briarproject.bramble.api.system.AndroidExecutor;
|
||||||
|
import org.briarproject.bramble.api.system.Clock;
|
||||||
import org.briarproject.bramble.util.AndroidUtils;
|
import org.briarproject.bramble.util.AndroidUtils;
|
||||||
|
import org.briarproject.bramble.util.IoUtils;
|
||||||
|
|
||||||
import java.io.Closeable;
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.security.SecureRandom;
|
import java.security.SecureRandom;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
import java.util.concurrent.BlockingQueue;
|
||||||
import java.util.concurrent.ExecutionException;
|
import java.util.concurrent.ExecutionException;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
|
import java.util.concurrent.LinkedBlockingQueue;
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
import static android.bluetooth.BluetoothAdapter.ACTION_DISCOVERY_FINISHED;
|
||||||
|
import static android.bluetooth.BluetoothAdapter.ACTION_DISCOVERY_STARTED;
|
||||||
import static android.bluetooth.BluetoothAdapter.ACTION_SCAN_MODE_CHANGED;
|
import static android.bluetooth.BluetoothAdapter.ACTION_SCAN_MODE_CHANGED;
|
||||||
import static android.bluetooth.BluetoothAdapter.ACTION_STATE_CHANGED;
|
import static android.bluetooth.BluetoothAdapter.ACTION_STATE_CHANGED;
|
||||||
import static android.bluetooth.BluetoothAdapter.EXTRA_SCAN_MODE;
|
import static android.bluetooth.BluetoothAdapter.EXTRA_SCAN_MODE;
|
||||||
@@ -37,8 +46,12 @@ import static android.bluetooth.BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERA
|
|||||||
import static android.bluetooth.BluetoothAdapter.SCAN_MODE_NONE;
|
import static android.bluetooth.BluetoothAdapter.SCAN_MODE_NONE;
|
||||||
import static android.bluetooth.BluetoothAdapter.STATE_OFF;
|
import static android.bluetooth.BluetoothAdapter.STATE_OFF;
|
||||||
import static android.bluetooth.BluetoothAdapter.STATE_ON;
|
import static android.bluetooth.BluetoothAdapter.STATE_ON;
|
||||||
|
import static android.bluetooth.BluetoothDevice.ACTION_FOUND;
|
||||||
|
import static android.bluetooth.BluetoothDevice.EXTRA_DEVICE;
|
||||||
|
import static java.util.concurrent.TimeUnit.MILLISECONDS;
|
||||||
|
import static java.util.logging.Level.INFO;
|
||||||
import static java.util.logging.Level.WARNING;
|
import static java.util.logging.Level.WARNING;
|
||||||
import static org.briarproject.bramble.util.LogUtils.logException;
|
import static org.briarproject.bramble.util.PrivacyUtils.scrubMacAddress;
|
||||||
|
|
||||||
@MethodsNotNullByDefault
|
@MethodsNotNullByDefault
|
||||||
@ParametersNotNullByDefault
|
@ParametersNotNullByDefault
|
||||||
@@ -47,8 +60,11 @@ class AndroidBluetoothPlugin extends BluetoothPlugin<BluetoothServerSocket> {
|
|||||||
private static final Logger LOG =
|
private static final Logger LOG =
|
||||||
Logger.getLogger(AndroidBluetoothPlugin.class.getName());
|
Logger.getLogger(AndroidBluetoothPlugin.class.getName());
|
||||||
|
|
||||||
|
private static final int MAX_DISCOVERY_MS = 10_000;
|
||||||
|
|
||||||
private final AndroidExecutor androidExecutor;
|
private final AndroidExecutor androidExecutor;
|
||||||
private final Context appContext;
|
private final Context appContext;
|
||||||
|
private final Clock clock;
|
||||||
|
|
||||||
private volatile boolean wasEnabledByUs = false;
|
private volatile boolean wasEnabledByUs = false;
|
||||||
private volatile BluetoothStateReceiver receiver = null;
|
private volatile BluetoothStateReceiver receiver = null;
|
||||||
@@ -58,12 +74,13 @@ class AndroidBluetoothPlugin extends BluetoothPlugin<BluetoothServerSocket> {
|
|||||||
|
|
||||||
AndroidBluetoothPlugin(BluetoothConnectionLimiter connectionLimiter,
|
AndroidBluetoothPlugin(BluetoothConnectionLimiter connectionLimiter,
|
||||||
Executor ioExecutor, AndroidExecutor androidExecutor,
|
Executor ioExecutor, AndroidExecutor androidExecutor,
|
||||||
Context appContext, SecureRandom secureRandom, Backoff backoff,
|
Context appContext, SecureRandom secureRandom, Clock clock,
|
||||||
DuplexPluginCallback callback, int maxLatency) {
|
Backoff backoff, DuplexPluginCallback callback, int maxLatency) {
|
||||||
super(connectionLimiter, ioExecutor, secureRandom, backoff, callback,
|
super(connectionLimiter, ioExecutor, secureRandom, backoff, callback,
|
||||||
maxLatency);
|
maxLatency);
|
||||||
this.androidExecutor = androidExecutor;
|
this.androidExecutor = androidExecutor;
|
||||||
this.appContext = appContext;
|
this.appContext = appContext;
|
||||||
|
this.clock = clock;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -143,11 +160,7 @@ class AndroidBluetoothPlugin extends BluetoothPlugin<BluetoothServerSocket> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
void tryToClose(@Nullable BluetoothServerSocket ss) {
|
void tryToClose(@Nullable BluetoothServerSocket ss) {
|
||||||
try {
|
IoUtils.tryToClose(ss, LOG, WARNING);
|
||||||
if (ss != null) ss.close();
|
|
||||||
} catch (IOException e) {
|
|
||||||
logException(LOG, WARNING, e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -177,17 +190,77 @@ class AndroidBluetoothPlugin extends BluetoothPlugin<BluetoothServerSocket> {
|
|||||||
s.connect();
|
s.connect();
|
||||||
return wrapSocket(s);
|
return wrapSocket(s);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
tryToClose(s);
|
IoUtils.tryToClose(s, LOG, WARNING);
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void tryToClose(@Nullable Closeable c) {
|
@Override
|
||||||
try {
|
@Nullable
|
||||||
if (c != null) c.close();
|
DuplexTransportConnection discoverAndConnect(String uuid) {
|
||||||
} catch (IOException e) {
|
if (adapter == null) return null;
|
||||||
logException(LOG, WARNING, e);
|
for (String address : discoverDevices()) {
|
||||||
|
try {
|
||||||
|
if (LOG.isLoggable(INFO))
|
||||||
|
LOG.info("Connecting to " + scrubMacAddress(address));
|
||||||
|
return connectTo(address, uuid);
|
||||||
|
} catch (IOException e) {
|
||||||
|
if (LOG.isLoggable(INFO)) {
|
||||||
|
LOG.info("Could not connect to "
|
||||||
|
+ scrubMacAddress(address));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
LOG.info("Could not connect to any devices");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Collection<String> discoverDevices() {
|
||||||
|
List<String> addresses = new ArrayList<>();
|
||||||
|
BlockingQueue<Intent> intents = new LinkedBlockingQueue<>();
|
||||||
|
DiscoveryReceiver receiver = new DiscoveryReceiver(intents);
|
||||||
|
IntentFilter filter = new IntentFilter();
|
||||||
|
filter.addAction(ACTION_DISCOVERY_STARTED);
|
||||||
|
filter.addAction(ACTION_DISCOVERY_FINISHED);
|
||||||
|
filter.addAction(ACTION_FOUND);
|
||||||
|
appContext.registerReceiver(receiver, filter);
|
||||||
|
try {
|
||||||
|
if (adapter.startDiscovery()) {
|
||||||
|
long now = clock.currentTimeMillis();
|
||||||
|
long end = now + MAX_DISCOVERY_MS;
|
||||||
|
while (now < end) {
|
||||||
|
Intent i = intents.poll(end - now, MILLISECONDS);
|
||||||
|
if (i == null) break;
|
||||||
|
String action = i.getAction();
|
||||||
|
if (ACTION_DISCOVERY_STARTED.equals(action)) {
|
||||||
|
LOG.info("Discovery started");
|
||||||
|
} else if (ACTION_DISCOVERY_FINISHED.equals(action)) {
|
||||||
|
LOG.info("Discovery finished");
|
||||||
|
break;
|
||||||
|
} else if (ACTION_FOUND.equals(action)) {
|
||||||
|
BluetoothDevice d = i.getParcelableExtra(EXTRA_DEVICE);
|
||||||
|
String address = d.getAddress();
|
||||||
|
if (LOG.isLoggable(INFO))
|
||||||
|
LOG.info("Discovered " + scrubMacAddress(address));
|
||||||
|
if (!addresses.contains(address))
|
||||||
|
addresses.add(address);
|
||||||
|
}
|
||||||
|
now = clock.currentTimeMillis();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
LOG.info("Could not start discovery");
|
||||||
|
}
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
LOG.info("Interrupted while discovering devices");
|
||||||
|
Thread.currentThread().interrupt();
|
||||||
|
} finally {
|
||||||
|
LOG.info("Cancelling discovery");
|
||||||
|
adapter.cancelDiscovery();
|
||||||
|
appContext.unregisterReceiver(receiver);
|
||||||
|
}
|
||||||
|
// Shuffle the addresses so we don't always try the same one first
|
||||||
|
Collections.shuffle(addresses);
|
||||||
|
return addresses;
|
||||||
}
|
}
|
||||||
|
|
||||||
private class BluetoothStateReceiver extends BroadcastReceiver {
|
private class BluetoothStateReceiver extends BroadcastReceiver {
|
||||||
@@ -207,4 +280,18 @@ class AndroidBluetoothPlugin extends BluetoothPlugin<BluetoothServerSocket> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static class DiscoveryReceiver extends BroadcastReceiver {
|
||||||
|
|
||||||
|
private final BlockingQueue<Intent> intents;
|
||||||
|
|
||||||
|
private DiscoveryReceiver(BlockingQueue<Intent> intents) {
|
||||||
|
this.intents = intents;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onReceive(Context ctx, Intent intent) {
|
||||||
|
intents.add(intent);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import org.briarproject.bramble.api.plugin.duplex.DuplexPlugin;
|
|||||||
import org.briarproject.bramble.api.plugin.duplex.DuplexPluginCallback;
|
import org.briarproject.bramble.api.plugin.duplex.DuplexPluginCallback;
|
||||||
import org.briarproject.bramble.api.plugin.duplex.DuplexPluginFactory;
|
import org.briarproject.bramble.api.plugin.duplex.DuplexPluginFactory;
|
||||||
import org.briarproject.bramble.api.system.AndroidExecutor;
|
import org.briarproject.bramble.api.system.AndroidExecutor;
|
||||||
|
import org.briarproject.bramble.api.system.Clock;
|
||||||
|
|
||||||
import java.security.SecureRandom;
|
import java.security.SecureRandom;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
@@ -33,17 +34,19 @@ public class AndroidBluetoothPluginFactory implements DuplexPluginFactory {
|
|||||||
private final Context appContext;
|
private final Context appContext;
|
||||||
private final SecureRandom secureRandom;
|
private final SecureRandom secureRandom;
|
||||||
private final EventBus eventBus;
|
private final EventBus eventBus;
|
||||||
|
private final Clock clock;
|
||||||
private final BackoffFactory backoffFactory;
|
private final BackoffFactory backoffFactory;
|
||||||
|
|
||||||
public AndroidBluetoothPluginFactory(Executor ioExecutor,
|
public AndroidBluetoothPluginFactory(Executor ioExecutor,
|
||||||
AndroidExecutor androidExecutor, Context appContext,
|
AndroidExecutor androidExecutor, Context appContext,
|
||||||
SecureRandom secureRandom, EventBus eventBus,
|
SecureRandom secureRandom, EventBus eventBus, Clock clock,
|
||||||
BackoffFactory backoffFactory) {
|
BackoffFactory backoffFactory) {
|
||||||
this.ioExecutor = ioExecutor;
|
this.ioExecutor = ioExecutor;
|
||||||
this.androidExecutor = androidExecutor;
|
this.androidExecutor = androidExecutor;
|
||||||
this.appContext = appContext;
|
this.appContext = appContext;
|
||||||
this.secureRandom = secureRandom;
|
this.secureRandom = secureRandom;
|
||||||
this.eventBus = eventBus;
|
this.eventBus = eventBus;
|
||||||
|
this.clock = clock;
|
||||||
this.backoffFactory = backoffFactory;
|
this.backoffFactory = backoffFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -65,7 +68,7 @@ public class AndroidBluetoothPluginFactory implements DuplexPluginFactory {
|
|||||||
MAX_POLLING_INTERVAL, BACKOFF_BASE);
|
MAX_POLLING_INTERVAL, BACKOFF_BASE);
|
||||||
AndroidBluetoothPlugin plugin = new AndroidBluetoothPlugin(
|
AndroidBluetoothPlugin plugin = new AndroidBluetoothPlugin(
|
||||||
connectionLimiter, ioExecutor, androidExecutor, appContext,
|
connectionLimiter, ioExecutor, androidExecutor, appContext,
|
||||||
secureRandom, backoff, callback, MAX_LATENCY);
|
secureRandom, clock, backoff, callback, MAX_LATENCY);
|
||||||
eventBus.addListener(plugin);
|
eventBus.addListener(plugin);
|
||||||
return plugin;
|
return plugin;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import android.content.pm.PackageManager;
|
|||||||
import android.content.pm.PackageManager.NameNotFoundException;
|
import android.content.pm.PackageManager.NameNotFoundException;
|
||||||
import android.os.PowerManager;
|
import android.os.PowerManager;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.battery.BatteryManager;
|
||||||
import org.briarproject.bramble.api.network.NetworkManager;
|
import org.briarproject.bramble.api.network.NetworkManager;
|
||||||
import org.briarproject.bramble.api.nullsafety.MethodsNotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.MethodsNotNullByDefault;
|
||||||
import org.briarproject.bramble.api.nullsafety.ParametersNotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.ParametersNotNullByDefault;
|
||||||
@@ -41,17 +42,18 @@ class AndroidTorPlugin extends TorPlugin {
|
|||||||
Context appContext, NetworkManager networkManager,
|
Context appContext, NetworkManager networkManager,
|
||||||
LocationUtils locationUtils, SocketFactory torSocketFactory,
|
LocationUtils locationUtils, SocketFactory torSocketFactory,
|
||||||
Clock clock, ResourceProvider resourceProvider,
|
Clock clock, ResourceProvider resourceProvider,
|
||||||
CircumventionProvider circumventionProvider, Backoff backoff,
|
CircumventionProvider circumventionProvider,
|
||||||
|
BatteryManager batteryManager, Backoff backoff,
|
||||||
DuplexPluginCallback callback, String architecture, int maxLatency,
|
DuplexPluginCallback callback, String architecture, int maxLatency,
|
||||||
int maxIdleTime) {
|
int maxIdleTime) {
|
||||||
super(ioExecutor, networkManager, locationUtils, torSocketFactory,
|
super(ioExecutor, networkManager, locationUtils, torSocketFactory,
|
||||||
clock, resourceProvider, circumventionProvider, backoff,
|
clock, resourceProvider, circumventionProvider, batteryManager,
|
||||||
callback, architecture, maxLatency, maxIdleTime,
|
backoff, callback, architecture, maxLatency, maxIdleTime,
|
||||||
appContext.getDir("tor", MODE_PRIVATE));
|
appContext.getDir("tor", MODE_PRIVATE));
|
||||||
this.appContext = appContext;
|
this.appContext = appContext;
|
||||||
PowerManager pm = (PowerManager)
|
PowerManager pm = (PowerManager)
|
||||||
appContext.getSystemService(POWER_SERVICE);
|
appContext.getSystemService(POWER_SERVICE);
|
||||||
assert pm != null;
|
if (pm == null) throw new AssertionError();
|
||||||
wakeLock = new RenewableWakeLock(pm, scheduler, PARTIAL_WAKE_LOCK,
|
wakeLock = new RenewableWakeLock(pm, scheduler, PARTIAL_WAKE_LOCK,
|
||||||
WAKE_LOCK_TAG, 1, MINUTES);
|
WAKE_LOCK_TAG, 1, MINUTES);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ package org.briarproject.bramble.plugin.tor;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.battery.BatteryManager;
|
||||||
import org.briarproject.bramble.api.event.EventBus;
|
import org.briarproject.bramble.api.event.EventBus;
|
||||||
import org.briarproject.bramble.api.network.NetworkManager;
|
import org.briarproject.bramble.api.network.NetworkManager;
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
@@ -48,6 +49,7 @@ public class AndroidTorPluginFactory implements DuplexPluginFactory {
|
|||||||
private final BackoffFactory backoffFactory;
|
private final BackoffFactory backoffFactory;
|
||||||
private final ResourceProvider resourceProvider;
|
private final ResourceProvider resourceProvider;
|
||||||
private final CircumventionProvider circumventionProvider;
|
private final CircumventionProvider circumventionProvider;
|
||||||
|
private final BatteryManager batteryManager;
|
||||||
private final Clock clock;
|
private final Clock clock;
|
||||||
|
|
||||||
public AndroidTorPluginFactory(Executor ioExecutor,
|
public AndroidTorPluginFactory(Executor ioExecutor,
|
||||||
@@ -55,7 +57,8 @@ public class AndroidTorPluginFactory implements DuplexPluginFactory {
|
|||||||
NetworkManager networkManager, LocationUtils locationUtils,
|
NetworkManager networkManager, LocationUtils locationUtils,
|
||||||
EventBus eventBus, SocketFactory torSocketFactory,
|
EventBus eventBus, SocketFactory torSocketFactory,
|
||||||
BackoffFactory backoffFactory, ResourceProvider resourceProvider,
|
BackoffFactory backoffFactory, ResourceProvider resourceProvider,
|
||||||
CircumventionProvider circumventionProvider, Clock clock) {
|
CircumventionProvider circumventionProvider,
|
||||||
|
BatteryManager batteryManager, Clock clock) {
|
||||||
this.ioExecutor = ioExecutor;
|
this.ioExecutor = ioExecutor;
|
||||||
this.scheduler = scheduler;
|
this.scheduler = scheduler;
|
||||||
this.appContext = appContext;
|
this.appContext = appContext;
|
||||||
@@ -66,6 +69,7 @@ public class AndroidTorPluginFactory implements DuplexPluginFactory {
|
|||||||
this.backoffFactory = backoffFactory;
|
this.backoffFactory = backoffFactory;
|
||||||
this.resourceProvider = resourceProvider;
|
this.resourceProvider = resourceProvider;
|
||||||
this.circumventionProvider = circumventionProvider;
|
this.circumventionProvider = circumventionProvider;
|
||||||
|
this.batteryManager = batteryManager;
|
||||||
this.clock = clock;
|
this.clock = clock;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -104,8 +108,8 @@ public class AndroidTorPluginFactory implements DuplexPluginFactory {
|
|||||||
MAX_POLLING_INTERVAL, BACKOFF_BASE);
|
MAX_POLLING_INTERVAL, BACKOFF_BASE);
|
||||||
AndroidTorPlugin plugin = new AndroidTorPlugin(ioExecutor, scheduler,
|
AndroidTorPlugin plugin = new AndroidTorPlugin(ioExecutor, scheduler,
|
||||||
appContext, networkManager, locationUtils, torSocketFactory,
|
appContext, networkManager, locationUtils, torSocketFactory,
|
||||||
clock, resourceProvider, circumventionProvider, backoff,
|
clock, resourceProvider, circumventionProvider, batteryManager,
|
||||||
callback, architecture, MAX_LATENCY, MAX_IDLE_TIME);
|
backoff, callback, architecture, MAX_LATENCY, MAX_IDLE_TIME);
|
||||||
eventBus.addListener(plugin);
|
eventBus.addListener(plugin);
|
||||||
return plugin;
|
return plugin;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,9 +22,11 @@ class AndroidResourceProvider implements ResourceProvider {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public InputStream getResourceInputStream(String name) {
|
public InputStream getResourceInputStream(String name, String extension) {
|
||||||
Resources res = appContext.getResources();
|
Resources res = appContext.getResources();
|
||||||
int resId = res.getIdentifier(name, "raw", appContext.getPackageName());
|
// extension is ignored on Android, resources are retrieved without it
|
||||||
|
int resId =
|
||||||
|
res.getIdentifier(name, "raw", appContext.getPackageName());
|
||||||
return res.openRawResource(resId);
|
return res.openRawResource(resId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
package org.briarproject.bramble.system;
|
package org.briarproject.bramble.system;
|
||||||
|
|
||||||
|
import android.annotation.SuppressLint;
|
||||||
import android.app.Application;
|
import android.app.Application;
|
||||||
import android.bluetooth.BluetoothAdapter;
|
import android.bluetooth.BluetoothAdapter;
|
||||||
import android.bluetooth.BluetoothDevice;
|
import android.bluetooth.BluetoothDevice;
|
||||||
@@ -26,7 +27,7 @@ import static android.provider.Settings.Secure.ANDROID_ID;
|
|||||||
|
|
||||||
@Immutable
|
@Immutable
|
||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
class AndroidSecureRandomProvider extends LinuxSecureRandomProvider {
|
class AndroidSecureRandomProvider extends UnixSecureRandomProvider {
|
||||||
|
|
||||||
private static final int SEED_LENGTH = 32;
|
private static final int SEED_LENGTH = 32;
|
||||||
|
|
||||||
@@ -37,6 +38,7 @@ class AndroidSecureRandomProvider extends LinuxSecureRandomProvider {
|
|||||||
appContext = app.getApplicationContext();
|
appContext = app.getApplicationContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressLint("HardwareIds")
|
||||||
@Override
|
@Override
|
||||||
protected void writeToEntropyPool(DataOutputStream out) throws IOException {
|
protected void writeToEntropyPool(DataOutputStream out) throws IOException {
|
||||||
super.writeToEntropyPool(out);
|
super.writeToEntropyPool(out);
|
||||||
@@ -49,12 +51,14 @@ class AndroidSecureRandomProvider extends LinuxSecureRandomProvider {
|
|||||||
String id = Settings.Secure.getString(contentResolver, ANDROID_ID);
|
String id = Settings.Secure.getString(contentResolver, ANDROID_ID);
|
||||||
if (id != null) out.writeUTF(id);
|
if (id != null) out.writeUTF(id);
|
||||||
Parcel parcel = Parcel.obtain();
|
Parcel parcel = Parcel.obtain();
|
||||||
WifiManager wm =
|
WifiManager wm = (WifiManager) appContext.getApplicationContext()
|
||||||
(WifiManager) appContext.getSystemService(WIFI_SERVICE);
|
.getSystemService(WIFI_SERVICE);
|
||||||
List<WifiConfiguration> configs = wm.getConfiguredNetworks();
|
if (wm != null) {
|
||||||
if (configs != null) {
|
List<WifiConfiguration> configs = wm.getConfiguredNetworks();
|
||||||
for (WifiConfiguration config : configs)
|
if (configs != null) {
|
||||||
parcel.writeParcelable(config, 0);
|
for (WifiConfiguration config : configs)
|
||||||
|
parcel.writeParcelable(config, 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
BluetoothAdapter bt = BluetoothAdapter.getDefaultAdapter();
|
BluetoothAdapter bt = BluetoothAdapter.getDefaultAdapter();
|
||||||
if (bt != null) {
|
if (bt != null) {
|
||||||
@@ -77,13 +81,13 @@ class AndroidSecureRandomProvider extends LinuxSecureRandomProvider {
|
|||||||
|
|
||||||
// Based on https://android-developers.googleblog.com/2013/08/some-securerandom-thoughts.html
|
// Based on https://android-developers.googleblog.com/2013/08/some-securerandom-thoughts.html
|
||||||
private void applyOpenSslFix() {
|
private void applyOpenSslFix() {
|
||||||
byte[] seed = new LinuxSecureRandomSpi().engineGenerateSeed(
|
byte[] seed = new UnixSecureRandomSpi().engineGenerateSeed(
|
||||||
SEED_LENGTH);
|
SEED_LENGTH);
|
||||||
try {
|
try {
|
||||||
// Seed the OpenSSL PRNG
|
// Seed the OpenSSL PRNG
|
||||||
Class.forName("org.apache.harmony.xnet.provider.jsse.NativeCrypto")
|
Class.forName("org.apache.harmony.xnet.provider.jsse.NativeCrypto")
|
||||||
.getMethod("RAND_seed", byte[].class)
|
.getMethod("RAND_seed", byte[].class)
|
||||||
.invoke(null, seed);
|
.invoke(null, (Object) seed);
|
||||||
// Mix the output of the Linux PRNG into the OpenSSL PRNG
|
// Mix the output of the Linux PRNG into the OpenSSL PRNG
|
||||||
int bytesRead = (Integer) Class.forName(
|
int bytesRead = (Integer) Class.forName(
|
||||||
"org.apache.harmony.xnet.provider.jsse.NativeCrypto")
|
"org.apache.harmony.xnet.provider.jsse.NativeCrypto")
|
||||||
|
|||||||
@@ -11,15 +11,12 @@ import java.util.ArrayList;
|
|||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.logging.Logger;
|
|
||||||
|
|
||||||
import static android.content.Context.MODE_PRIVATE;
|
import static android.content.Context.MODE_PRIVATE;
|
||||||
|
import static android.os.Build.VERSION.SDK_INT;
|
||||||
|
|
||||||
public class AndroidUtils {
|
public class AndroidUtils {
|
||||||
|
|
||||||
private static final Logger LOG =
|
|
||||||
Logger.getLogger(AndroidUtils.class.getName());
|
|
||||||
|
|
||||||
// Fake Bluetooth address returned by BluetoothAdapter on API 23 and later
|
// Fake Bluetooth address returned by BluetoothAdapter on API 23 and later
|
||||||
private static final String FAKE_BLUETOOTH_ADDRESS = "02:00:00:00:00:00";
|
private static final String FAKE_BLUETOOTH_ADDRESS = "02:00:00:00:00:00";
|
||||||
|
|
||||||
@@ -28,7 +25,7 @@ public class AndroidUtils {
|
|||||||
@SuppressWarnings("deprecation")
|
@SuppressWarnings("deprecation")
|
||||||
public static Collection<String> getSupportedArchitectures() {
|
public static Collection<String> getSupportedArchitectures() {
|
||||||
List<String> abis = new ArrayList<>();
|
List<String> abis = new ArrayList<>();
|
||||||
if (Build.VERSION.SDK_INT >= 21) {
|
if (SDK_INT >= 21) {
|
||||||
abis.addAll(Arrays.asList(Build.SUPPORTED_ABIS));
|
abis.addAll(Arrays.asList(Build.SUPPORTED_ABIS));
|
||||||
} else {
|
} else {
|
||||||
abis.add(Build.CPU_ABI);
|
abis.add(Build.CPU_ABI);
|
||||||
|
|||||||
@@ -1,8 +0,0 @@
|
|||||||
Bridge 131.252.210.150:8081 0E858AC201BF0F3FA3C462F64844CBFFC7297A42
|
|
||||||
Bridge 67.205.189.122:8443 12D64D5D44E20169585E7378580C0D33A872AD98
|
|
||||||
Bridge 45.32.148.146:8443 0CE016FB2462D8BF179AE71F7D702D09DEAC3F1D
|
|
||||||
Bridge 148.251.90.59:7510 019F727CA6DCA6CA5C90B55E477B7D87981E75BC
|
|
||||||
Bridge 195.91.239.8:9001 BA83F62551545655BBEBBFF353A45438D73FD45A
|
|
||||||
Bridge 185.165.184.217:6429 64CC94BEC51254E4409AD059192833854CCB95F0
|
|
||||||
Bridge 45.55.1.74:8443 6F18FEFBB0CAECD5ABA755312FCCB34FC11A7AB8
|
|
||||||
Bridge 95.85.40.163:9001 40057BE9CF76B6C5BDBE713753468BE0A990DE9C
|
|
||||||
@@ -112,6 +112,8 @@ public class AndroidAccountManagerTest extends BrambleMockTestCase {
|
|||||||
// Other directories should be deleted
|
// Other directories should be deleted
|
||||||
File potatoDir = new File(testDir, ".potato");
|
File potatoDir = new File(testDir, ".potato");
|
||||||
File potatoFile = new File(potatoDir, "file");
|
File potatoFile = new File(potatoDir, "file");
|
||||||
|
File filesDir = new File(testDir, "filesDir");
|
||||||
|
File externalCacheDir = new File(testDir, "externalCacheDir");
|
||||||
|
|
||||||
context.checking(new Expectations() {{
|
context.checking(new Expectations() {{
|
||||||
oneOf(prefs).edit();
|
oneOf(prefs).edit();
|
||||||
@@ -128,6 +130,12 @@ public class AndroidAccountManagerTest extends BrambleMockTestCase {
|
|||||||
will(returnValue(true));
|
will(returnValue(true));
|
||||||
oneOf(app).getApplicationInfo();
|
oneOf(app).getApplicationInfo();
|
||||||
will(returnValue(applicationInfo));
|
will(returnValue(applicationInfo));
|
||||||
|
oneOf(app).getFilesDir();
|
||||||
|
will(returnValue(filesDir));
|
||||||
|
oneOf(app).getCacheDir();
|
||||||
|
will(returnValue(cacheDir));
|
||||||
|
oneOf(app).getExternalCacheDir();
|
||||||
|
will(returnValue(externalCacheDir));
|
||||||
}});
|
}});
|
||||||
|
|
||||||
assertTrue(dbDir.mkdirs());
|
assertTrue(dbDir.mkdirs());
|
||||||
@@ -140,6 +148,8 @@ public class AndroidAccountManagerTest extends BrambleMockTestCase {
|
|||||||
assertTrue(cacheFile.createNewFile());
|
assertTrue(cacheFile.createNewFile());
|
||||||
assertTrue(potatoDir.mkdirs());
|
assertTrue(potatoDir.mkdirs());
|
||||||
assertTrue(potatoFile.createNewFile());
|
assertTrue(potatoFile.createNewFile());
|
||||||
|
assertTrue(filesDir.mkdirs());
|
||||||
|
assertTrue(externalCacheDir.mkdirs());
|
||||||
|
|
||||||
accountManager.deleteAccount();
|
accountManager.deleteAccount();
|
||||||
|
|
||||||
@@ -153,6 +163,8 @@ public class AndroidAccountManagerTest extends BrambleMockTestCase {
|
|||||||
assertFalse(cacheFile.exists());
|
assertFalse(cacheFile.exists());
|
||||||
assertFalse(potatoDir.exists());
|
assertFalse(potatoDir.exists());
|
||||||
assertFalse(potatoFile.exists());
|
assertFalse(potatoFile.exists());
|
||||||
|
assertFalse(filesDir.exists());
|
||||||
|
assertFalse(externalCacheDir.exists());
|
||||||
}
|
}
|
||||||
|
|
||||||
@After
|
@After
|
||||||
|
|||||||
@@ -1,53 +1,55 @@
|
|||||||
dependencyVerification {
|
dependencyVerification {
|
||||||
verify = [
|
verify = [
|
||||||
'cglib:cglib:3.2.0:cglib-3.2.0.jar:adb13bab79712ad6bdf1bd59f2a3918018a8016e722e8a357065afb9e6690861',
|
'cglib:cglib:3.2.0:cglib-3.2.0.jar:adb13bab79712ad6bdf1bd59f2a3918018a8016e722e8a357065afb9e6690861',
|
||||||
'com.android.support.test:monitor:1.0.2:monitor-1.0.2.aar:38ef4fa98a32dc55550ff49bb36a583e178b3a9b830fcb8dcc27bfc4254bc2bc',
|
'com.android.tools.analytics-library:protos:26.3.2:protos-26.3.2.jar:50238fb4298b297217b184b9cd93c14f83536fcee829eb0ca850bdb5b53251f0',
|
||||||
'com.android.support.test:runner:1.0.2:runner-1.0.2.aar:f04b9ae342975ba1cb3e4a06e13426e3e6b8a73faa45acba604493d83c9a4f00',
|
'com.android.tools.analytics-library:shared:26.3.2:shared-26.3.2.jar:ddd80dcf21905018b7c0583ba72b7282f446084d4952422609a09fbf8237ef71',
|
||||||
'com.android.support:support-annotations:27.1.1:support-annotations-27.1.1.jar:3365960206c3d2b09e845f555e7f88f8effc8d2f00b369e66c4be384029299cf',
|
'com.android.tools.analytics-library:tracker:26.3.2:tracker-26.3.2.jar:28c575d2d1af003e96d7b375a668ee10b2673a2dd0f6438750aa8c3b42e7d0ad',
|
||||||
'com.android.tools.analytics-library:protos:26.1.3:protos-26.1.3.jar:818c9f256f141d9dafec03a1aa2b94d240b2c140acfd7ee31a8b3e6c2b9479e3',
|
'com.android.tools.build:apksig:3.3.2:apksig-3.3.2.jar:84c4aaa20127c6c1fe6bdd334b3f5df71f54ad080be9029c8a10f43b6a908acd',
|
||||||
'com.android.tools.analytics-library:shared:26.1.3:shared-26.1.3.jar:7110706c7ada96c8b6f5ca80c478291bc7899d46277de2c48527e045442401a3',
|
'com.android.tools.build:apkzlib:3.3.2:apkzlib-3.3.2.jar:d34e523278e5dff565eba3ef3c089d515b2b5cc7b47dc77e2f3465e5e47176ac',
|
||||||
'com.android.tools.analytics-library:tracker:26.1.3:tracker-26.1.3.jar:4155424bf2ce4872da83332579a1707252bc66cbd77c5144fdc4483d0f2e1418',
|
'com.android.tools.build:builder-model:3.3.2:builder-model-3.3.2.jar:055e3db0ecee9e06b9f024034999a29cd92cb1885207b37542126bd8bcc57f46',
|
||||||
'com.android.tools.build:apksig:3.1.3:apksig-3.1.3.jar:7e1f8e675a6e768e5b56405e41d6c3cc05befe62e601b04177de1029902c9c89',
|
'com.android.tools.build:builder-test-api:3.3.2:builder-test-api-3.3.2.jar:0b2e4cd7615bbcad14a3c91fe45ae26693508d06e40ba06c5968b8bc24416618',
|
||||||
'com.android.tools.build:builder-model:3.1.3:builder-model-3.1.3.jar:06ad1c422d679fc698451479cb40ba863849d67bfd1de23f6d2c16d78b024b0b',
|
'com.android.tools.build:builder:3.3.2:builder-3.3.2.jar:65649704da7aef0487235fd326f0f2e99ed5cf958e80f204496e6e08a42bd9f5',
|
||||||
'com.android.tools.build:builder-test-api:3.1.3:builder-test-api-3.1.3.jar:4d989f780436794f0f8b2f50e9e079b786571eac90f26c208ab2ae6d4012f389',
|
'com.android.tools.build:gradle-api:3.3.2:gradle-api-3.3.2.jar:3cbd47e41bb70330dd72ec2c9fe51e6173554b484a03829b5a2de9e00841e040',
|
||||||
'com.android.tools.build:builder:3.1.3:builder-3.1.3.jar:8a1092012c89d0ec1ee2eff09c5708c71ef4482a6862df8d3a44a67fccace01c',
|
'com.android.tools.build:manifest-merger:26.3.2:manifest-merger-26.3.2.jar:05c4a6d8b02fb9f08744876477d0a68547c03a8a9069b1f086684fa04af97c33',
|
||||||
'com.android.tools.build:gradle-api:3.1.3:gradle-api-3.1.3.jar:01e4df521456aef66514336f1d492346730dd1fb8f6433a89f62da834941ed72',
|
'com.android.tools.ddms:ddmlib:26.3.2:ddmlib-26.3.2.jar:d248da8a563d6e46d2c7ebbf371a4877e00510f4ca763c0bb272d5a281bf8b85',
|
||||||
'com.android.tools.build:manifest-merger:26.1.3:manifest-merger-26.1.3.jar:1e4fc7e932adb4607082409800e5e6fccb42e6c5360ae5990094bf522f3ada55',
|
'com.android.tools.external.com-intellij:intellij-core:26.3.2:intellij-core-26.3.2.jar:6c5ecc968230e9f4dcd0fef28885379feace1f0cd8130de6f61d649c86139bf3',
|
||||||
'com.android.tools.ddms:ddmlib:26.1.3:ddmlib-26.1.3.jar:c54931cd68df5d1ea2923b3b320eae47cd2307a5a916bb8674c0acf93cd1d3cd',
|
'com.android.tools.external.com-intellij:kotlin-compiler:26.3.2:kotlin-compiler-26.3.2.jar:1007d9b07ccb49cd8eaf30fda10ed4681d4714f2f9ab2ecda39b4e539cc51bbe',
|
||||||
'com.android.tools.external.com-intellij:intellij-core:26.1.3:intellij-core-26.1.3.jar:af67f5535fef2e1a28b1007a4acb8c5deb6a1e33b8afe7b11d012c9e778ebcec',
|
'com.android.tools.external.org-jetbrains:uast:26.3.2:uast-26.3.2.jar:5d1833e562ea4f38a89708dfde695f0a162cbd39d003d3dde818c3fdc2b05317',
|
||||||
'com.android.tools.external.com-intellij:kotlin-compiler:26.1.3:kotlin-compiler-26.1.3.jar:c746d2859dc11cc05c84b692b3498d3a621e0929511f8440ee009c6557838fd4',
|
'com.android.tools.layoutlib:layoutlib-api:26.3.2:layoutlib-api-26.3.2.jar:d7e61e874ab95f5c350dd38b6a95b5c9dbe0083a02001884264cdb390cb255b8',
|
||||||
'com.android.tools.external.org-jetbrains:uast:26.1.3:uast-26.1.3.jar:3f3f6651d0c7685a77ecb22e9c82d6b49fdf24322c17360768dc530678f43265',
|
'com.android.tools.lint:lint-api:26.3.2:lint-api-26.3.2.jar:5867dfd7fb4a4e161a816a5d29d045f9b542d34594c00a1efec46fb4cd0e1033',
|
||||||
'com.android.tools.layoutlib:layoutlib-api:26.1.3:layoutlib-api-26.1.3.jar:10bc73ce706c45629872d6a999dbe12116df64e24f47ff93b7b13121ff57b4b0',
|
'com.android.tools.lint:lint-checks:26.3.2:lint-checks-26.3.2.jar:4b163b9c93790d2771e92ba8de58a0d9e0671ffcf2ccef3cf496efd442e27517',
|
||||||
'com.android.tools.lint:lint-api:26.1.3:lint-api-26.1.3.jar:6f97323f9af8deda86278717885b5c927f3766757db89709f52d11d42b6fb751',
|
'com.android.tools.lint:lint-gradle-api:26.3.2:lint-gradle-api-26.3.2.jar:54cb282e0c054f9bed3f51302ce08b003c8ab7961dfd5a4f6de26c23cc23062f',
|
||||||
'com.android.tools.lint:lint-checks:26.1.3:lint-checks-26.1.3.jar:73c3d53784c9ce3e6d5968506581918e0179645d20809927ca4a001dd766b001',
|
'com.android.tools.lint:lint-gradle:26.3.2:lint-gradle-26.3.2.jar:bb139615f4ce97d42cc394b9389b49b76a6eb85be6785a5d272991543b519013',
|
||||||
'com.android.tools.lint:lint-gradle-api:26.1.3:lint-gradle-api-26.1.3.jar:7ca3c4866ec21dc21d53a9d86f752b77ace6f6c610a0c9dc877313856c733d9d',
|
'com.android.tools.lint:lint:26.3.2:lint-26.3.2.jar:ef7b369f8a56a92ccb0f4c1c357666b9339e4a711a9d84747d446441746cfe4e',
|
||||||
'com.android.tools.lint:lint-gradle:26.1.3:lint-gradle-26.1.3.jar:db0c354b8f4b6f6637e31f91c564785a59ff896325331fcbc3de7458e0b6c067',
|
'com.android.tools:annotations:26.3.2:annotations-26.3.2.jar:5bcce8e98b6a2f5ccf13ebcefd8f734e0b35f8b19e456575665631442ce1f7a1',
|
||||||
'com.android.tools.lint:lint-kotlin:26.1.3:lint-kotlin-26.1.3.jar:94e2b0f4565a241561cfb8fc1222bb3f132a3b98d2a90421dbb72ee8358e7d68',
|
'com.android.tools:common:26.3.2:common-26.3.2.jar:d9f8e7f0669e9a701568e3db6a87c89cf12d8fa6811c9991e969f950215ecfac',
|
||||||
'com.android.tools.lint:lint:26.1.3:lint-26.1.3.jar:8d5f32c989c6d191d712e90ad3ca2d1c409313599551d04d834caa44d26c78df',
|
'com.android.tools:dvlib:26.3.2:dvlib-26.3.2.jar:d84aad56161c7773579303d69714ded6897c64c6ddfd7d456e453231e4dfe811',
|
||||||
'com.android.tools:annotations:26.1.3:annotations-26.1.3.jar:c950430b24ac5d58fc97e7283b8f0115f99587e76e08b4e1e2aaa780f2d77323',
|
'com.android.tools:repository:26.3.2:repository-26.3.2.jar:da611eeb06e9ab8750d25b9e2901e10db8e5ec6304eb4c8b7103d39e0921ea40',
|
||||||
'com.android.tools:common:26.1.3:common-26.1.3.jar:7c31a90581a148ab219f615a59667f0dded7fa39b248529784474da3c2274ef2',
|
'com.android.tools:sdk-common:26.3.2:sdk-common-26.3.2.jar:82823a3bf25e64fac33a286490f0cf5ac50c2cdb3c540149b030896bb44bf96c',
|
||||||
'com.android.tools:dvlib:26.1.3:dvlib-26.1.3.jar:0cae87906f53d3f1088366a916ed180a7312b6d9919b90797f238875c8492855',
|
'com.android.tools:sdklib:26.3.2:sdklib-26.3.2.jar:424d15492af67321900963238646d27495ab60de2a5b19e6a416963bc5d6932b',
|
||||||
'com.android.tools:repository:26.1.3:repository-26.1.3.jar:52d4539cc68db91b261e2a33b2c8206b26e05539078758dc28cfb3854adb4f59',
|
|
||||||
'com.android.tools:sdk-common:26.1.3:sdk-common-26.1.3.jar:1948603ca9ff22c7ebb3178000bffa3a9dd2ca1cc5cb0c793cae08468b8fcfc1',
|
|
||||||
'com.android.tools:sdklib:26.1.3:sdklib-26.1.3.jar:4adcfaad9514607098d2c51503c39811112d3050f4d1e744c01c7f08f591032b',
|
|
||||||
'com.google.code.findbugs:jsr305:1.3.9:jsr305-1.3.9.jar:905721a0eea90a81534abb7ee6ef4ea2e5e645fa1def0a5cd88402df1b46c9ed',
|
'com.google.code.findbugs:jsr305:1.3.9:jsr305-1.3.9.jar:905721a0eea90a81534abb7ee6ef4ea2e5e645fa1def0a5cd88402df1b46c9ed',
|
||||||
'com.google.code.gson:gson:2.7:gson-2.7.jar:2d43eb5ea9e133d2ee2405cc14f5ee08951b8361302fdd93494a3a997b508d32',
|
'com.google.code.findbugs:jsr305:3.0.2:jsr305-3.0.2.jar:766ad2a0783f2687962c8ad74ceecc38a28b9f72a2d085ee438b7813e928d0c7',
|
||||||
'com.google.dagger:dagger-compiler:2.0.2:dagger-compiler-2.0.2.jar:b74bc9de063dd4c6400b232231f2ef5056145b8fbecbf5382012007dd1c071b3',
|
'com.google.code.gson:gson:2.8.0:gson-2.8.0.jar:c6221763bd79c4f1c3dc7f750b5f29a0bb38b367b81314c4f71896e340c40825',
|
||||||
'com.google.dagger:dagger-producers:2.0-beta:dagger-producers-2.0-beta.jar:99ec15e8a0507ba569e7655bc1165ee5e5ca5aa914b3c8f7e2c2458f724edd6b',
|
'com.google.dagger:dagger-compiler:2.19:dagger-compiler-2.19.jar:27a4b202a2de908182edb261f8c0a264e08e5e4733d7514bc7fbf0d31da5c0fc',
|
||||||
'com.google.dagger:dagger:2.0.2:dagger-2.0.2.jar:84c0282ed8be73a29e0475d639da030b55dee72369e58dd35ae7d4fe6243dcf9',
|
'com.google.dagger:dagger-producers:2.19:dagger-producers-2.19.jar:a17663abe0fc38b676026950907d4c5f5e2bf338375415861eaff6e3bdb0b768',
|
||||||
'com.google.errorprone:error_prone_annotations:2.0.18:error_prone_annotations-2.0.18.jar:cb4cfad870bf563a07199f3ebea5763f0dec440fcda0b318640b1feaa788656b',
|
'com.google.dagger:dagger-spi:2.19:dagger-spi-2.19.jar:e7a6379d82c841f6aac2866948ad1eed716528707814602842a8d844ce04e2e1',
|
||||||
'com.google.guava:guava:18.0:guava-18.0.jar:d664fbfc03d2e5ce9cab2a44fb01f1d0bf9dfebeccc1a473b1f9ea31f79f6f99',
|
'com.google.dagger:dagger:2.19:dagger-2.19.jar:514b6f1e0727c6572e1d65cb27e4ae668b7aeaeb93a29515182965265b609939',
|
||||||
'com.google.guava:guava:22.0:guava-22.0.jar:1158e94c7de4da480873f0b4ab4a1da14c0d23d4b1902cc94a58a6f0f9ab579e',
|
'com.google.errorprone:error_prone_annotations:2.1.3:error_prone_annotations-2.1.3.jar:03d0329547c13da9e17c634d1049ea2ead093925e290567e1a364fd6b1fc7ff8',
|
||||||
|
'com.google.errorprone:javac-shaded:9-dev-r4023-3:javac-shaded-9-dev-r4023-3.jar:65bfccf60986c47fbc17c9ebab0be626afc41741e0a6ec7109e0768817a36f30',
|
||||||
|
'com.google.googlejavaformat:google-java-format:1.5:google-java-format-1.5.jar:aa19ad7850fb85178aa22f2fddb163b84d6ce4d0035872f30d4408195ca1144e',
|
||||||
|
'com.google.guava:guava:25.0-jre:guava-25.0-jre.jar:3fd4341776428c7e0e5c18a7c10de129475b69ab9d30aeafbb5c277bb6074fa9',
|
||||||
|
'com.google.guava:guava:26.0-jre:guava-26.0-jre.jar:a0e9cabad665bc20bcd2b01f108e5fc03f756e13aea80abaadb9f407033bea2c',
|
||||||
'com.google.j2objc:j2objc-annotations:1.1:j2objc-annotations-1.1.jar:40ceb7157feb263949e0f503fe5f71689333a621021aa20ce0d0acee3badaa0f',
|
'com.google.j2objc:j2objc-annotations:1.1:j2objc-annotations-1.1.jar:40ceb7157feb263949e0f503fe5f71689333a621021aa20ce0d0acee3badaa0f',
|
||||||
'com.google.jimfs:jimfs:1.1:jimfs-1.1.jar:c4828e28d7c0a930af9387510b3bada7daa5c04d7c25a75c7b8b081f1c257ddd',
|
'com.google.jimfs:jimfs:1.1:jimfs-1.1.jar:c4828e28d7c0a930af9387510b3bada7daa5c04d7c25a75c7b8b081f1c257ddd',
|
||||||
'com.google.protobuf:protobuf-java:3.4.0:protobuf-java-3.4.0.jar:dce7e66b32456a1b1198da0caff3a8acb71548658391e798c79369241e6490a4',
|
'com.google.protobuf:protobuf-java:3.4.0:protobuf-java-3.4.0.jar:dce7e66b32456a1b1198da0caff3a8acb71548658391e798c79369241e6490a4',
|
||||||
'com.googlecode.json-simple:json-simple:1.1:json-simple-1.1.jar:2d9484f4c649f708f47f9a479465fc729770ee65617dca3011836602264f6439',
|
'com.googlecode.json-simple:json-simple:1.1:json-simple-1.1.jar:2d9484f4c649f708f47f9a479465fc729770ee65617dca3011836602264f6439',
|
||||||
|
'com.squareup:javapoet:1.11.1:javapoet-1.11.1.jar:9cbf2107be499ec6e95afd36b58e3ca122a24166cdd375732e51267d64058e90',
|
||||||
'com.squareup:javawriter:2.5.0:javawriter-2.5.0.jar:fcfb09fb0ea0aa97d3cfe7ea792398081348e468f126b3603cb3803f240197f0',
|
'com.squareup:javawriter:2.5.0:javawriter-2.5.0.jar:fcfb09fb0ea0aa97d3cfe7ea792398081348e468f126b3603cb3803f240197f0',
|
||||||
'com.sun.activation:javax.activation:1.2.0:javax.activation-1.2.0.jar:993302b16cd7056f21e779cc577d175a810bb4900ef73cd8fbf2b50f928ba9ce',
|
'com.sun.activation:javax.activation:1.2.0:javax.activation-1.2.0.jar:993302b16cd7056f21e779cc577d175a810bb4900ef73cd8fbf2b50f928ba9ce',
|
||||||
'com.sun.istack:istack-commons-runtime:2.21:istack-commons-runtime-2.21.jar:c33e67a0807095f02a0e2da139412dd7c4f9cc1a4c054b3e434f96831ba950f4',
|
'com.sun.istack:istack-commons-runtime:2.21:istack-commons-runtime-2.21.jar:c33e67a0807095f02a0e2da139412dd7c4f9cc1a4c054b3e434f96831ba950f4',
|
||||||
'com.sun.xml.fastinfoset:FastInfoset:1.2.13:FastInfoset-1.2.13.jar:27a77db909f3c2833c0b1a37c55af1db06045118ad2eed96ce567b6632bce038',
|
'com.sun.xml.fastinfoset:FastInfoset:1.2.13:FastInfoset-1.2.13.jar:27a77db909f3c2833c0b1a37c55af1db06045118ad2eed96ce567b6632bce038',
|
||||||
'commons-codec:commons-codec:1.6:commons-codec-1.6.jar:54b34e941b8e1414bd3e40d736efd3481772dc26db3296f6aa45cec9f6203d86',
|
'commons-codec:commons-codec:1.9:commons-codec-1.9.jar:ad19d2601c3abf0b946b5c3a4113e226a8c1e3305e395b90013b78dd94a723ce',
|
||||||
'commons-logging:commons-logging:1.1.1:commons-logging-1.1.1.jar:ce6f913cad1f0db3aad70186d65c5bc7ffcc9a99e3fe8e0b137312819f7c362f',
|
'commons-logging:commons-logging:1.2:commons-logging-1.2.jar:daddea1ea0be0f56978ab3006b8ac92834afeefbd9b7e4e6316fca57df0fa636',
|
||||||
'it.unimi.dsi:fastutil:7.2.0:fastutil-7.2.0.jar:74fa208043740642f7e6eb09faba15965218ad2f50ce3020efb100136e4b591c',
|
'it.unimi.dsi:fastutil:7.2.0:fastutil-7.2.0.jar:74fa208043740642f7e6eb09faba15965218ad2f50ce3020efb100136e4b591c',
|
||||||
'javax.annotation:jsr250-api:1.0:jsr250-api-1.0.jar:a1a922d0d9b6d183ed3800dfac01d1e1eb159f0e8c6f94736931c1def54a941f',
|
'javax.annotation:jsr250-api:1.0:jsr250-api-1.0.jar:a1a922d0d9b6d183ed3800dfac01d1e1eb159f0e8c6f94736931c1def54a941f',
|
||||||
'javax.inject:javax.inject:1:javax.inject-1.jar:91c77044a50c481636c32d916fd89c9118a72195390452c81065080f957de7ff',
|
'javax.inject:javax.inject:1:javax.inject-1.jar:91c77044a50c481636c32d916fd89c9118a72195390452c81065080f957de7ff',
|
||||||
@@ -58,24 +60,28 @@ dependencyVerification {
|
|||||||
'org.apache.ant:ant-launcher:1.9.4:ant-launcher-1.9.4.jar:7bccea20b41801ca17bcbc909a78c835d0f443f12d639c77bd6ae3d05861608d',
|
'org.apache.ant:ant-launcher:1.9.4:ant-launcher-1.9.4.jar:7bccea20b41801ca17bcbc909a78c835d0f443f12d639c77bd6ae3d05861608d',
|
||||||
'org.apache.ant:ant:1.9.4:ant-1.9.4.jar:649ae0730251de07b8913f49286d46bba7b92d47c5f332610aa426c4f02161d8',
|
'org.apache.ant:ant:1.9.4:ant-1.9.4.jar:649ae0730251de07b8913f49286d46bba7b92d47c5f332610aa426c4f02161d8',
|
||||||
'org.apache.commons:commons-compress:1.12:commons-compress-1.12.jar:2c1542faf343185b7cab9c3d55c8ae5471d6d095d3887a4adefdbdf2984dc0b6',
|
'org.apache.commons:commons-compress:1.12:commons-compress-1.12.jar:2c1542faf343185b7cab9c3d55c8ae5471d6d095d3887a4adefdbdf2984dc0b6',
|
||||||
'org.apache.httpcomponents:httpclient:4.2.6:httpclient-4.2.6.jar:362e9324ee7c697e21279e20077b52737ddef3f1b2c1a7abe5ad34b465145550',
|
'org.apache.httpcomponents:httpclient:4.5.2:httpclient-4.5.2.jar:0dffc621400d6c632f55787d996b8aeca36b30746a716e079a985f24d8074057',
|
||||||
'org.apache.httpcomponents:httpcore:4.2.5:httpcore-4.2.5.jar:e5e82da4cc66c8d917bbf743e3c0752efe8522735e7fc9dbddb65bccea81cfe9',
|
'org.apache.httpcomponents:httpcore:4.4.5:httpcore-4.4.5.jar:64d5453874cab7e40a7065cb01a9a9ca1053845a9786b478878b679e0580cec3',
|
||||||
'org.apache.httpcomponents:httpmime:4.1:httpmime-4.1.jar:31629566148e8a47688ae43b420abc3ecd783ed15b33bebc00824bf24c9b15aa',
|
'org.apache.httpcomponents:httpmime:4.5.2:httpmime-4.5.2.jar:231a3f7e4962053db2be8461d5422e68fc458a3a7dd7d8ada803a348e21f8f07',
|
||||||
'org.beanshell:bsh:1.3.0:bsh-1.3.0.jar:9b04edc75d19db54f1b4e8b5355e9364384c6cf71eb0a1b9724c159d779879f8',
|
'org.beanshell:bsh:1.3.0:bsh-1.3.0.jar:9b04edc75d19db54f1b4e8b5355e9364384c6cf71eb0a1b9724c159d779879f8',
|
||||||
'org.bouncycastle:bcpkix-jdk15on:1.56:bcpkix-jdk15on-1.56.jar:7043dee4e9e7175e93e0b36f45b1ec1ecb893c5f755667e8b916eb8dd201c6ca',
|
'org.bouncycastle:bcpkix-jdk15on:1.56:bcpkix-jdk15on-1.56.jar:7043dee4e9e7175e93e0b36f45b1ec1ecb893c5f755667e8b916eb8dd201c6ca',
|
||||||
'org.bouncycastle:bcprov-jdk15on:1.56:bcprov-jdk15on-1.56.jar:963e1ee14f808ffb99897d848ddcdb28fa91ddda867eb18d303e82728f878349',
|
'org.bouncycastle:bcprov-jdk15on:1.56:bcprov-jdk15on-1.56.jar:963e1ee14f808ffb99897d848ddcdb28fa91ddda867eb18d303e82728f878349',
|
||||||
'org.briarproject:tor-android:0.2.9.16:tor-android-0.2.9.16.zip:515e33dda6a30853c885a2de2c79ae1ab9ad8b6db44f5db8890333ec2e24f4ae',
|
'org.briarproject:obfs4proxy-android:0.0.9:obfs4proxy-android-0.0.9.zip:9b7e9181535ea8d8bbe8ae6338e08cf4c5fc1e357a779393e0ce49586d459ae0',
|
||||||
'org.codehaus.groovy:groovy-all:2.4.12:groovy-all-2.4.12.jar:6a56af4bd48903d56bec62821876cadefafd007360cc6bd0d8f7aa8d72b38be4',
|
'org.briarproject:tor-android:0.3.5.8:tor-android-0.3.5.8.zip:42a13a6f185be1a62f42e3f30ce66a3c099ac5ec890a65e7593111b65b44a54a',
|
||||||
|
'org.checkerframework:checker-compat-qual:2.5.3:checker-compat-qual-2.5.3.jar:d76b9afea61c7c082908023f0cbc1427fab9abd2df915c8b8a3e7a509bccbc6d',
|
||||||
|
'org.checkerframework:checker-qual:2.5.2:checker-qual-2.5.2.jar:64b02691c8b9d4e7700f8ee2e742dce7ea2c6e81e662b7522c9ee3bf568c040a',
|
||||||
|
'org.codehaus.groovy:groovy-all:2.4.15:groovy-all-2.4.15.jar:51d6c4e71782e85674239189499854359d380fb75e1a703756e3aaa5b98a5af0',
|
||||||
'org.codehaus.mojo:animal-sniffer-annotations:1.14:animal-sniffer-annotations-1.14.jar:2068320bd6bad744c3673ab048f67e30bef8f518996fa380033556600669905d',
|
'org.codehaus.mojo:animal-sniffer-annotations:1.14:animal-sniffer-annotations-1.14.jar:2068320bd6bad744c3673ab048f67e30bef8f518996fa380033556600669905d',
|
||||||
'org.glassfish.jaxb:jaxb-core:2.2.11:jaxb-core-2.2.11.jar:37bcaee8ebb04362c8352a5bf6221b86967ecdab5164c696b10b9a2bb587b2aa',
|
'org.glassfish.jaxb:jaxb-core:2.2.11:jaxb-core-2.2.11.jar:37bcaee8ebb04362c8352a5bf6221b86967ecdab5164c696b10b9a2bb587b2aa',
|
||||||
'org.glassfish.jaxb:jaxb-runtime:2.2.11:jaxb-runtime-2.2.11.jar:a874f2351cfba8e2946be3002d10c18a6da8f21b52ba2acf52f2b85d5520ed70',
|
'org.glassfish.jaxb:jaxb-runtime:2.2.11:jaxb-runtime-2.2.11.jar:a874f2351cfba8e2946be3002d10c18a6da8f21b52ba2acf52f2b85d5520ed70',
|
||||||
'org.glassfish.jaxb:txw2:2.2.11:txw2-2.2.11.jar:272a3ccad45a4511351920cd2a8633c53cab8d5220c7a92954da5526bb5eafea',
|
'org.glassfish.jaxb:txw2:2.2.11:txw2-2.2.11.jar:272a3ccad45a4511351920cd2a8633c53cab8d5220c7a92954da5526bb5eafea',
|
||||||
'org.hamcrest:hamcrest-core:1.3:hamcrest-core-1.3.jar:66fdef91e9739348df7a096aa384a5685f4e875584cce89386a7a47251c4d8e9',
|
'org.hamcrest:hamcrest-core:1.3:hamcrest-core-1.3.jar:66fdef91e9739348df7a096aa384a5685f4e875584cce89386a7a47251c4d8e9',
|
||||||
'org.hamcrest:hamcrest-library:1.3:hamcrest-library-1.3.jar:711d64522f9ec410983bd310934296da134be4254a125080a0416ec178dfad1c',
|
'org.hamcrest:hamcrest-library:1.3:hamcrest-library-1.3.jar:711d64522f9ec410983bd310934296da134be4254a125080a0416ec178dfad1c',
|
||||||
'org.jetbrains.kotlin:kotlin-reflect:1.2.0:kotlin-reflect-1.2.0.jar:4f48a872bad6e4d9c053f4ad610d11e4012ad7e58dc19a03dd5eb811f36069dd',
|
'org.jetbrains.kotlin:kotlin-reflect:1.3.21:kotlin-reflect-1.3.21.jar:a3065c822633191e0a3e3ee12a29bec234fc4b2864a6bb87ef48cce3e9e0c26a',
|
||||||
'org.jetbrains.kotlin:kotlin-stdlib-jre7:1.2.0:kotlin-stdlib-jre7-1.2.0.jar:c7a20fb951d437797afe8980aff6c1e5a03f310c661ba58ba1d4fa90cb0f2926',
|
'org.jetbrains.kotlin:kotlin-stdlib-common:1.3.21:kotlin-stdlib-common-1.3.21.jar:cea61f7b611895e64f58569a9757fc0ab0d582f107211e1930e0ce2a0add52a7',
|
||||||
'org.jetbrains.kotlin:kotlin-stdlib-jre8:1.2.0:kotlin-stdlib-jre8-1.2.0.jar:633524eee6ef1941f7cb1dab7ee3927b0a221ceee9047aeb5515f4cbb990c82a',
|
'org.jetbrains.kotlin:kotlin-stdlib-jdk7:1.3.21:kotlin-stdlib-jdk7-1.3.21.jar:a87875604fd42140da6938ae4d35ee61081f4482536efc6d2615b8b626a198af',
|
||||||
'org.jetbrains.kotlin:kotlin-stdlib:1.2.0:kotlin-stdlib-1.2.0.jar:05cfd9f5ac0b41910703a8925f7211a495909b27a2ffdd1c5106f1689aeafcd4',
|
'org.jetbrains.kotlin:kotlin-stdlib-jdk8:1.3.21:kotlin-stdlib-jdk8-1.3.21.jar:5823ed66ac122a1c55442ebca5a209a843ccd87f562edc31a787f3d2e47f74d4',
|
||||||
|
'org.jetbrains.kotlin:kotlin-stdlib:1.3.21:kotlin-stdlib-1.3.21.jar:38ba2370d9f06f50433e06b2ca775b94473c2e2785f410926079ab793c72b034',
|
||||||
'org.jetbrains.trove4j:trove4j:20160824:trove4j-20160824.jar:1917871c8deb468307a584680c87a44572f5a8b0b98c6d397fc0f5f86596dbe7',
|
'org.jetbrains.trove4j:trove4j:20160824:trove4j-20160824.jar:1917871c8deb468307a584680c87a44572f5a8b0b98c6d397fc0f5f86596dbe7',
|
||||||
'org.jetbrains:annotations:13.0:annotations-13.0.jar:ace2a10dc8e2d5fd34925ecac03e4988b2c0f851650c94b8cef49ba1bd111478',
|
'org.jetbrains:annotations:13.0:annotations-13.0.jar:ace2a10dc8e2d5fd34925ecac03e4988b2c0f851650c94b8cef49ba1bd111478',
|
||||||
'org.jmock:jmock-junit4:2.8.2:jmock-junit4-2.8.2.jar:f7ee4df4f7bd7b7f1cafad3b99eb74d579f109d5992ff625347352edb55e674c',
|
'org.jmock:jmock-junit4:2.8.2:jmock-junit4-2.8.2.jar:f7ee4df4f7bd7b7f1cafad3b99eb74d579f109d5992ff625347352edb55e674c',
|
||||||
@@ -84,11 +90,11 @@ dependencyVerification {
|
|||||||
'org.jmock:jmock:2.8.2:jmock-2.8.2.jar:6c73cb4a2e6dbfb61fd99c9a768539c170ab6568e57846bd60dbf19596b65b16',
|
'org.jmock:jmock:2.8.2:jmock-2.8.2.jar:6c73cb4a2e6dbfb61fd99c9a768539c170ab6568e57846bd60dbf19596b65b16',
|
||||||
'org.jvnet.staxex:stax-ex:1.7.7:stax-ex-1.7.7.jar:a31ff7d77163c0deb09e7fee59ad35ae44c2cee2cc8552a116ccd1583d813fb4',
|
'org.jvnet.staxex:stax-ex:1.7.7:stax-ex-1.7.7.jar:a31ff7d77163c0deb09e7fee59ad35ae44c2cee2cc8552a116ccd1583d813fb4',
|
||||||
'org.objenesis:objenesis:2.1:objenesis-2.1.jar:c74330cc6b806c804fd37e74487b4fe5d7c2750c5e15fbc6efa13bdee1bdef80',
|
'org.objenesis:objenesis:2.1:objenesis-2.1.jar:c74330cc6b806c804fd37e74487b4fe5d7c2750c5e15fbc6efa13bdee1bdef80',
|
||||||
'org.ow2.asm:asm-analysis:5.1:asm-analysis-5.1.jar:a34658f5c5de4b573eef21131cc32cc25f7b66407944f312b28ec2e56abb1fa9',
|
'org.ow2.asm:asm-analysis:6.0:asm-analysis-6.0.jar:2f1a6387219c3a6cc4856481f221b03bd9f2408a326d416af09af5d6f608c1f4',
|
||||||
'org.ow2.asm:asm-commons:5.1:asm-commons-5.1.jar:97b3786e1f55e74bddf8ad102bf50e33bbcbc1f6b7fd7b36f0bbbb25cd4981be',
|
'org.ow2.asm:asm-commons:6.0:asm-commons-6.0.jar:f1bce5c648a96a017bdcd01fe5d59af9845297fd7b79b81c015a6fbbd9719abf',
|
||||||
'org.ow2.asm:asm-tree:5.1:asm-tree-5.1.jar:c0de2bbc4cb8297419659813ecd4ed1d077ed1dd5c1f5544cc5143e493e84c10',
|
'org.ow2.asm:asm-tree:6.0:asm-tree-6.0.jar:887998fb69727c8759e4d253f856822801e33f9fd4caa566b3ac58ee92106215',
|
||||||
'org.ow2.asm:asm-util:5.1:asm-util-5.1.jar:ee032c39ae5e3cd099148fbba9a2124f9ed613e5cb93e03ee0fa8808ce364040',
|
'org.ow2.asm:asm-util:6.0:asm-util-6.0.jar:356afebdb0f870175262e5188f8709a3b17aa2a5a6a4b0340b04d4b449bca5f6',
|
||||||
'org.ow2.asm:asm:5.0.4:asm-5.0.4.jar:896618ed8ae62702521a78bc7be42b7c491a08e6920a15f89a3ecdec31e9a220',
|
'org.ow2.asm:asm:5.0.4:asm-5.0.4.jar:896618ed8ae62702521a78bc7be42b7c491a08e6920a15f89a3ecdec31e9a220',
|
||||||
'org.ow2.asm:asm:5.1:asm-5.1.jar:d2da399a9967c69f0a21739256fa79d284222c223082cacadc17372244764b54',
|
'org.ow2.asm:asm:6.0:asm-6.0.jar:dd8971c74a4e697899a8e95caae4ea8760ea6c486dc6b97b1795e75760420461',
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,15 +7,13 @@ apply plugin: 'witness'
|
|||||||
apply from: 'witness.gradle'
|
apply from: 'witness.gradle'
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation "com.google.dagger:dagger:2.0.2"
|
implementation "com.google.dagger:dagger:2.19"
|
||||||
implementation 'com.google.code.findbugs:jsr305:3.0.2'
|
implementation 'com.google.code.findbugs:jsr305:3.0.2'
|
||||||
|
|
||||||
testImplementation 'junit:junit:4.12'
|
testImplementation 'junit:junit:4.12'
|
||||||
testImplementation "org.jmock:jmock:2.8.2"
|
testImplementation "org.jmock:jmock:2.8.2"
|
||||||
testImplementation "org.jmock:jmock-junit4:2.8.2"
|
testImplementation "org.jmock:jmock-junit4:2.8.2"
|
||||||
testImplementation "org.jmock:jmock-legacy:2.8.2"
|
testImplementation "org.jmock:jmock-legacy:2.8.2"
|
||||||
testImplementation "org.hamcrest:hamcrest-library:1.3"
|
|
||||||
testImplementation "org.hamcrest:hamcrest-core:1.3"
|
|
||||||
|
|
||||||
signature 'org.codehaus.mojo.signature:java16:1.1@signature'
|
signature 'org.codehaus.mojo.signature:java16:1.1@signature'
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,10 @@
|
|||||||
|
package org.briarproject.bramble.api;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
@NotNullByDefault
|
||||||
|
public interface Nameable {
|
||||||
|
|
||||||
|
String getName();
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
package org.briarproject.bramble.api;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
|
@Immutable
|
||||||
|
@NotNullByDefault
|
||||||
|
public class Pair<A, B> {
|
||||||
|
|
||||||
|
private final A first;
|
||||||
|
private final B second;
|
||||||
|
|
||||||
|
public Pair(A first, B second) {
|
||||||
|
this.first = first;
|
||||||
|
this.second = second;
|
||||||
|
}
|
||||||
|
|
||||||
|
public A getFirst() {
|
||||||
|
return first;
|
||||||
|
}
|
||||||
|
|
||||||
|
public B getSecond() {
|
||||||
|
return second;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -38,4 +38,18 @@ public abstract class StringMap extends Hashtable<String, String> {
|
|||||||
public void putInt(String key, int value) {
|
public void putInt(String key, int value) {
|
||||||
put(key, String.valueOf(value));
|
put(key, String.valueOf(value));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public long getLong(String key, long defaultValue) {
|
||||||
|
String s = get(key);
|
||||||
|
if (s == null) return defaultValue;
|
||||||
|
try {
|
||||||
|
return Long.valueOf(s);
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void putLong(String key, long value) {
|
||||||
|
put(key, String.valueOf(value));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +0,0 @@
|
|||||||
package org.briarproject.bramble.api;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An exception that indicates an unrecoverable version mismatch.
|
|
||||||
*/
|
|
||||||
public class UnsupportedVersionException extends IOException {
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
package org.briarproject.bramble.api.battery;
|
||||||
|
|
||||||
|
public interface BatteryManager {
|
||||||
|
|
||||||
|
boolean isCharging();
|
||||||
|
}
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
package org.briarproject.bramble.api.battery.event;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.event.Event;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An event that is broadcast when the device starts or stops charging.
|
||||||
|
*/
|
||||||
|
public class BatteryEvent extends Event {
|
||||||
|
|
||||||
|
private final boolean charging;
|
||||||
|
|
||||||
|
public BatteryEvent(boolean charging) {
|
||||||
|
this.charging = charging;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isCharging() {
|
||||||
|
return charging;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
package org.briarproject.briar.client;
|
package org.briarproject.bramble.api.client;
|
||||||
|
|
||||||
import org.briarproject.bramble.api.FormatException;
|
import org.briarproject.bramble.api.FormatException;
|
||||||
import org.briarproject.bramble.api.client.ClientHelper;
|
|
||||||
import org.briarproject.bramble.api.data.BdfDictionary;
|
import org.briarproject.bramble.api.data.BdfDictionary;
|
||||||
import org.briarproject.bramble.api.data.BdfList;
|
import org.briarproject.bramble.api.data.BdfList;
|
||||||
import org.briarproject.bramble.api.data.MetadataParser;
|
import org.briarproject.bramble.api.data.MetadataParser;
|
||||||
@@ -12,7 +11,7 @@ import org.briarproject.bramble.api.db.Transaction;
|
|||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
import org.briarproject.bramble.api.sync.InvalidMessageException;
|
import org.briarproject.bramble.api.sync.InvalidMessageException;
|
||||||
import org.briarproject.bramble.api.sync.Message;
|
import org.briarproject.bramble.api.sync.Message;
|
||||||
import org.briarproject.bramble.api.sync.ValidationManager.IncomingMessageHook;
|
import org.briarproject.bramble.api.sync.validation.IncomingMessageHook;
|
||||||
|
|
||||||
import javax.annotation.concurrent.Immutable;
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
@@ -62,5 +61,4 @@ public abstract class BdfIncomingMessageHook implements IncomingMessageHook {
|
|||||||
throw new InvalidMessageException(e);
|
throw new InvalidMessageException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -9,14 +9,13 @@ import org.briarproject.bramble.api.sync.Group;
|
|||||||
import org.briarproject.bramble.api.sync.InvalidMessageException;
|
import org.briarproject.bramble.api.sync.InvalidMessageException;
|
||||||
import org.briarproject.bramble.api.sync.Message;
|
import org.briarproject.bramble.api.sync.Message;
|
||||||
import org.briarproject.bramble.api.sync.MessageContext;
|
import org.briarproject.bramble.api.sync.MessageContext;
|
||||||
import org.briarproject.bramble.api.sync.ValidationManager.MessageValidator;
|
import org.briarproject.bramble.api.sync.validation.MessageValidator;
|
||||||
import org.briarproject.bramble.api.system.Clock;
|
import org.briarproject.bramble.api.system.Clock;
|
||||||
|
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
import javax.annotation.concurrent.Immutable;
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
import static org.briarproject.bramble.api.sync.SyncConstants.MESSAGE_HEADER_LENGTH;
|
|
||||||
import static org.briarproject.bramble.api.transport.TransportConstants.MAX_CLOCK_DIFFERENCE;
|
import static org.briarproject.bramble.api.transport.TransportConstants.MAX_CLOCK_DIFFERENCE;
|
||||||
|
|
||||||
@Immutable
|
@Immutable
|
||||||
@@ -49,14 +48,9 @@ public abstract class BdfMessageValidator implements MessageValidator {
|
|||||||
throw new InvalidMessageException(
|
throw new InvalidMessageException(
|
||||||
"Timestamp is too far in the future");
|
"Timestamp is too far in the future");
|
||||||
}
|
}
|
||||||
byte[] raw = m.getRaw();
|
|
||||||
if (raw.length <= MESSAGE_HEADER_LENGTH) {
|
|
||||||
throw new InvalidMessageException("Message is too short");
|
|
||||||
}
|
|
||||||
try {
|
try {
|
||||||
BdfList body = clientHelper.toList(raw, MESSAGE_HEADER_LENGTH,
|
BdfList bodyList = clientHelper.toList(m.getBody());
|
||||||
raw.length - MESSAGE_HEADER_LENGTH);
|
BdfMessageContext result = validateMessage(m, g, bodyList);
|
||||||
BdfMessageContext result = validateMessage(m, g, body);
|
|
||||||
Metadata meta = metadataEncoder.encode(result.getDictionary());
|
Metadata meta = metadataEncoder.encode(result.getDictionary());
|
||||||
return new MessageContext(meta, result.getDependencies());
|
return new MessageContext(meta, result.getDependencies());
|
||||||
} catch (FormatException e) {
|
} catch (FormatException e) {
|
||||||
|
|||||||
@@ -16,8 +16,6 @@ import org.briarproject.bramble.api.sync.MessageId;
|
|||||||
import java.security.GeneralSecurityException;
|
import java.security.GeneralSecurityException;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
|
||||||
|
|
||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
public interface ClientHelper {
|
public interface ClientHelper {
|
||||||
|
|
||||||
@@ -32,16 +30,12 @@ public interface ClientHelper {
|
|||||||
|
|
||||||
Message createMessageForStoringMetadata(GroupId g);
|
Message createMessageForStoringMetadata(GroupId g);
|
||||||
|
|
||||||
@Nullable
|
|
||||||
Message getMessage(MessageId m) throws DbException;
|
Message getMessage(MessageId m) throws DbException;
|
||||||
|
|
||||||
@Nullable
|
|
||||||
Message getMessage(Transaction txn, MessageId m) throws DbException;
|
Message getMessage(Transaction txn, MessageId m) throws DbException;
|
||||||
|
|
||||||
@Nullable
|
|
||||||
BdfList getMessageAsList(MessageId m) throws DbException, FormatException;
|
BdfList getMessageAsList(MessageId m) throws DbException, FormatException;
|
||||||
|
|
||||||
@Nullable
|
|
||||||
BdfList getMessageAsList(Transaction txn, MessageId m) throws DbException,
|
BdfList getMessageAsList(Transaction txn, MessageId m) throws DbException,
|
||||||
FormatException;
|
FormatException;
|
||||||
|
|
||||||
|
|||||||
@@ -4,8 +4,12 @@ import org.briarproject.bramble.api.identity.Author;
|
|||||||
import org.briarproject.bramble.api.identity.AuthorId;
|
import org.briarproject.bramble.api.identity.AuthorId;
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
import javax.annotation.concurrent.Immutable;
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
|
import static org.briarproject.bramble.api.identity.AuthorConstants.MAX_AUTHOR_NAME_LENGTH;
|
||||||
|
import static org.briarproject.bramble.util.StringUtils.toUtf8;
|
||||||
|
|
||||||
@Immutable
|
@Immutable
|
||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
public class Contact {
|
public class Contact {
|
||||||
@@ -13,13 +17,21 @@ public class Contact {
|
|||||||
private final ContactId id;
|
private final ContactId id;
|
||||||
private final Author author;
|
private final Author author;
|
||||||
private final AuthorId localAuthorId;
|
private final AuthorId localAuthorId;
|
||||||
|
@Nullable
|
||||||
|
private final String alias;
|
||||||
private final boolean verified, active;
|
private final boolean verified, active;
|
||||||
|
|
||||||
public Contact(ContactId id, Author author, AuthorId localAuthorId,
|
public Contact(ContactId id, Author author, AuthorId localAuthorId,
|
||||||
boolean verified, boolean active) {
|
@Nullable String alias, boolean verified, boolean active) {
|
||||||
|
if (alias != null) {
|
||||||
|
int aliasLength = toUtf8(alias).length;
|
||||||
|
if (aliasLength == 0 || aliasLength > MAX_AUTHOR_NAME_LENGTH)
|
||||||
|
throw new IllegalArgumentException();
|
||||||
|
}
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.author = author;
|
this.author = author;
|
||||||
this.localAuthorId = localAuthorId;
|
this.localAuthorId = localAuthorId;
|
||||||
|
this.alias = alias;
|
||||||
this.verified = verified;
|
this.verified = verified;
|
||||||
this.active = active;
|
this.active = active;
|
||||||
}
|
}
|
||||||
@@ -36,6 +48,11 @@ public class Contact {
|
|||||||
return localAuthorId;
|
return localAuthorId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public String getAlias() {
|
||||||
|
return alias;
|
||||||
|
}
|
||||||
|
|
||||||
public boolean isVerified() {
|
public boolean isVerified() {
|
||||||
return verified;
|
return verified;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +0,0 @@
|
|||||||
package org.briarproject.bramble.api.contact;
|
|
||||||
|
|
||||||
import org.briarproject.bramble.api.identity.Author;
|
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
|
||||||
|
|
||||||
@NotNullByDefault
|
|
||||||
public interface ContactExchangeListener {
|
|
||||||
|
|
||||||
void contactExchangeSucceeded(Author remoteAuthor);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The exchange failed because the contact already exists.
|
|
||||||
*/
|
|
||||||
void duplicateContact(Author remoteAuthor);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A general failure.
|
|
||||||
*/
|
|
||||||
void contactExchangeFailed();
|
|
||||||
}
|
|
||||||
@@ -41,8 +41,7 @@ public interface ContactExchangeTask {
|
|||||||
/**
|
/**
|
||||||
* Exchanges contact information with a remote peer.
|
* Exchanges contact information with a remote peer.
|
||||||
*/
|
*/
|
||||||
void startExchange(ContactExchangeListener listener,
|
void startExchange(LocalAuthor localAuthor, SecretKey masterSecret,
|
||||||
LocalAuthor localAuthor, SecretKey masterSecret,
|
|
||||||
DuplexTransportConnection conn, TransportId transportId,
|
DuplexTransportConnection conn, TransportId transportId,
|
||||||
boolean alice);
|
boolean alice);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,11 +5,16 @@ import org.briarproject.bramble.api.db.DbException;
|
|||||||
import org.briarproject.bramble.api.db.Transaction;
|
import org.briarproject.bramble.api.db.Transaction;
|
||||||
import org.briarproject.bramble.api.identity.Author;
|
import org.briarproject.bramble.api.identity.Author;
|
||||||
import org.briarproject.bramble.api.identity.AuthorId;
|
import org.briarproject.bramble.api.identity.AuthorId;
|
||||||
|
import org.briarproject.bramble.api.identity.AuthorInfo;
|
||||||
import org.briarproject.bramble.api.lifecycle.LifecycleManager;
|
import org.briarproject.bramble.api.lifecycle.LifecycleManager;
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
import static org.briarproject.bramble.api.contact.PendingContact.PendingContactState.FAILED;
|
||||||
|
|
||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
public interface ContactManager {
|
public interface ContactManager {
|
||||||
|
|
||||||
@@ -49,6 +54,35 @@ public interface ContactManager {
|
|||||||
long timestamp, boolean alice, boolean verified, boolean active)
|
long timestamp, boolean alice, boolean verified, boolean active)
|
||||||
throws DbException;
|
throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the static link that needs to be sent to the contact to be added.
|
||||||
|
*/
|
||||||
|
String getRemoteContactLink();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true if the given link is syntactically valid.
|
||||||
|
*/
|
||||||
|
boolean isValidRemoteContactLink(String link);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Requests a new contact to be added via the given {@code link}.
|
||||||
|
*
|
||||||
|
* @param link The link received from the contact we want to add.
|
||||||
|
* @param alias The alias the user has given this contact.
|
||||||
|
* @return A PendingContact representing the contact to be added.
|
||||||
|
*/
|
||||||
|
PendingContact addRemoteContactRequest(String link, String alias);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a list of {@link PendingContact}s.
|
||||||
|
*/
|
||||||
|
Collection<PendingContact> getPendingContacts();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes a {@link PendingContact} that is in state {@link FAILED}.
|
||||||
|
*/
|
||||||
|
void removePendingContact(PendingContact pendingContact);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the contact with the given ID.
|
* Returns the contact with the given ID.
|
||||||
*/
|
*/
|
||||||
@@ -93,6 +127,18 @@ public interface ContactManager {
|
|||||||
void setContactActive(Transaction txn, ContactId c, boolean active)
|
void setContactActive(Transaction txn, ContactId c, boolean active)
|
||||||
throws DbException;
|
throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets an alias name for the contact or unsets it if alias is null.
|
||||||
|
*/
|
||||||
|
void setContactAlias(Transaction txn, ContactId c, @Nullable String alias)
|
||||||
|
throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets an alias name for the contact or unsets it if alias is null.
|
||||||
|
*/
|
||||||
|
void setContactAlias(ContactId c, @Nullable String alias)
|
||||||
|
throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return true if a contact with this name and public key already exists
|
* Return true if a contact with this name and public key already exists
|
||||||
*/
|
*/
|
||||||
@@ -105,6 +151,16 @@ public interface ContactManager {
|
|||||||
boolean contactExists(AuthorId remoteAuthorId, AuthorId localAuthorId)
|
boolean contactExists(AuthorId remoteAuthorId, AuthorId localAuthorId)
|
||||||
throws DbException;
|
throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the {@link AuthorInfo} for the given author.
|
||||||
|
*/
|
||||||
|
AuthorInfo getAuthorInfo(AuthorId a) throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the {@link AuthorInfo} for the given author.
|
||||||
|
*/
|
||||||
|
AuthorInfo getAuthorInfo(Transaction txn, AuthorId a) throws DbException;
|
||||||
|
|
||||||
interface ContactHook {
|
interface ContactHook {
|
||||||
|
|
||||||
void addingContact(Transaction txn, Contact c) throws DbException;
|
void addingContact(Transaction txn, Contact c) throws DbException;
|
||||||
|
|||||||
@@ -0,0 +1,54 @@
|
|||||||
|
package org.briarproject.bramble.api.contact;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
|
@Immutable
|
||||||
|
@NotNullByDefault
|
||||||
|
public class PendingContact {
|
||||||
|
|
||||||
|
public enum PendingContactState {
|
||||||
|
WAITING_FOR_CONNECTION,
|
||||||
|
CONNECTED,
|
||||||
|
ADDING_CONTACT,
|
||||||
|
FAILED
|
||||||
|
}
|
||||||
|
|
||||||
|
private final PendingContactId id;
|
||||||
|
private final String alias;
|
||||||
|
private final PendingContactState state;
|
||||||
|
private final long timestamp;
|
||||||
|
|
||||||
|
public PendingContact(PendingContactId id, String alias,
|
||||||
|
PendingContactState state, long timestamp) {
|
||||||
|
this.id = id;
|
||||||
|
this.alias = alias;
|
||||||
|
this.state = state;
|
||||||
|
this.timestamp = timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getAlias() {
|
||||||
|
return alias;
|
||||||
|
}
|
||||||
|
|
||||||
|
public PendingContactState getState() {
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getTimestamp() {
|
||||||
|
return timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return id.hashCode();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
return o instanceof PendingContact &&
|
||||||
|
id.equals(((PendingContact) o).id);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
package org.briarproject.bramble.api.contact;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.UniqueId;
|
||||||
|
|
||||||
|
public class PendingContactId extends UniqueId {
|
||||||
|
|
||||||
|
public PendingContactId(byte[] id) {
|
||||||
|
super(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
package org.briarproject.bramble.api.contact.event;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.event.Event;
|
||||||
|
import org.briarproject.bramble.api.identity.Author;
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
@NotNullByDefault
|
||||||
|
public class ContactExchangeFailedEvent extends Event {
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
private final Author duplicateRemoteAuthor;
|
||||||
|
|
||||||
|
public ContactExchangeFailedEvent(@Nullable Author duplicateRemoteAuthor) {
|
||||||
|
this.duplicateRemoteAuthor = duplicateRemoteAuthor;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ContactExchangeFailedEvent() {
|
||||||
|
this(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public Author getDuplicateRemoteAuthor() {
|
||||||
|
return duplicateRemoteAuthor;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean wasDuplicateContact() {
|
||||||
|
return duplicateRemoteAuthor != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
package org.briarproject.bramble.api.contact.event;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.event.Event;
|
||||||
|
import org.briarproject.bramble.api.identity.Author;
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
@NotNullByDefault
|
||||||
|
public class ContactExchangeSucceededEvent extends Event {
|
||||||
|
|
||||||
|
private final Author remoteAuthor;
|
||||||
|
|
||||||
|
public ContactExchangeSucceededEvent(Author remoteAuthor) {
|
||||||
|
this.remoteAuthor = remoteAuthor;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Author getRemoteAuthor() {
|
||||||
|
return remoteAuthor;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
package org.briarproject.bramble.api.contact.event;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.contact.PendingContact.PendingContactState;
|
||||||
|
import org.briarproject.bramble.api.contact.PendingContactId;
|
||||||
|
import org.briarproject.bramble.api.event.Event;
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An event that is broadcast when a pending contact's state is changed.
|
||||||
|
*/
|
||||||
|
@Immutable
|
||||||
|
@NotNullByDefault
|
||||||
|
public class PendingContactStateChangedEvent extends Event {
|
||||||
|
|
||||||
|
private final PendingContactId id;
|
||||||
|
private final PendingContactState state;
|
||||||
|
|
||||||
|
public PendingContactStateChangedEvent(PendingContactId id,
|
||||||
|
PendingContactState state) {
|
||||||
|
this.id = id;
|
||||||
|
this.state = state;
|
||||||
|
}
|
||||||
|
|
||||||
|
public PendingContactId getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public PendingContactState getPendingContactState() {
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,71 @@
|
|||||||
|
package org.briarproject.bramble.api.data;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.Bytes;
|
||||||
|
import org.briarproject.bramble.api.FormatException;
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Map.Entry;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
import static org.briarproject.bramble.api.data.BdfDictionary.NULL_VALUE;
|
||||||
|
import static org.briarproject.bramble.util.StringUtils.toHexString;
|
||||||
|
|
||||||
|
@NotNullByDefault
|
||||||
|
public class BdfStringUtils {
|
||||||
|
|
||||||
|
public static String toString(@Nullable Object o) throws FormatException {
|
||||||
|
return toString(o, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String toString(@Nullable Object o, int indent)
|
||||||
|
throws FormatException {
|
||||||
|
if (o == null || o == NULL_VALUE) return "null";
|
||||||
|
if (o instanceof Boolean) return o.toString();
|
||||||
|
if (o instanceof Number) return o.toString();
|
||||||
|
if (o instanceof String) return "\"" + o + "\"";
|
||||||
|
if (o instanceof Bytes)
|
||||||
|
return "x" + toHexString(((Bytes) o).getBytes());
|
||||||
|
if (o instanceof byte[])
|
||||||
|
return "x" + toHexString((byte[]) o);
|
||||||
|
if (o instanceof List) {
|
||||||
|
List<?> list = (List) o;
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
sb.append("[\n");
|
||||||
|
int i = 0, size = list.size();
|
||||||
|
for (Object e : list) {
|
||||||
|
indent(sb, indent + 1);
|
||||||
|
sb.append(toString(e, indent + 1));
|
||||||
|
if (i++ < size - 1) sb.append(',');
|
||||||
|
sb.append('\n');
|
||||||
|
}
|
||||||
|
indent(sb, indent);
|
||||||
|
sb.append(']');
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
if (o instanceof Map) {
|
||||||
|
Map<?, ?> map = (Map) o;
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
sb.append("{\n");
|
||||||
|
int i = 0, size = map.size();
|
||||||
|
for (Entry e : map.entrySet()) {
|
||||||
|
indent(sb, indent + 1);
|
||||||
|
sb.append(toString(e.getKey(), indent + 1));
|
||||||
|
sb.append(": ");
|
||||||
|
sb.append(toString(e.getValue(), indent + 1));
|
||||||
|
if (i++ < size - 1) sb.append(',');
|
||||||
|
sb.append('\n');
|
||||||
|
}
|
||||||
|
indent(sb, indent);
|
||||||
|
sb.append('}');
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
throw new FormatException();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void indent(StringBuilder sb, int indent) {
|
||||||
|
for (int i = 0; i < indent; i++) sb.append('\t');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -19,6 +19,7 @@ import org.briarproject.bramble.api.sync.MessageId;
|
|||||||
import org.briarproject.bramble.api.sync.MessageStatus;
|
import org.briarproject.bramble.api.sync.MessageStatus;
|
||||||
import org.briarproject.bramble.api.sync.Offer;
|
import org.briarproject.bramble.api.sync.Offer;
|
||||||
import org.briarproject.bramble.api.sync.Request;
|
import org.briarproject.bramble.api.sync.Request;
|
||||||
|
import org.briarproject.bramble.api.sync.validation.MessageState;
|
||||||
import org.briarproject.bramble.api.transport.KeySet;
|
import org.briarproject.bramble.api.transport.KeySet;
|
||||||
import org.briarproject.bramble.api.transport.KeySetId;
|
import org.briarproject.bramble.api.transport.KeySetId;
|
||||||
import org.briarproject.bramble.api.transport.TransportKeys;
|
import org.briarproject.bramble.api.transport.TransportKeys;
|
||||||
@@ -28,8 +29,6 @@ import java.util.Map;
|
|||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
import static org.briarproject.bramble.api.sync.ValidationManager.State;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Encapsulates the database implementation and exposes high-level operations
|
* Encapsulates the database implementation and exposes high-level operations
|
||||||
* to other components.
|
* to other components.
|
||||||
@@ -76,6 +75,27 @@ public interface DatabaseComponent {
|
|||||||
*/
|
*/
|
||||||
void endTransaction(Transaction txn);
|
void endTransaction(Transaction txn);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Runs the given task within a transaction.
|
||||||
|
*/
|
||||||
|
<E extends Exception> void transaction(boolean readOnly,
|
||||||
|
DbRunnable<E> task) throws DbException, E;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Runs the given task within a transaction and returns the result of the
|
||||||
|
* task.
|
||||||
|
*/
|
||||||
|
<R, E extends Exception> R transactionWithResult(boolean readOnly,
|
||||||
|
DbCallable<R, E> task) throws DbException, E;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Runs the given task within a transaction and returns the result of the
|
||||||
|
* task, which may be null.
|
||||||
|
*/
|
||||||
|
@Nullable
|
||||||
|
<R, E extends Exception> R transactionWithNullableResult(boolean readOnly,
|
||||||
|
NullableDbCallable<R, E> task) throws DbException, E;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stores a contact associated with the given local and remote pseudonyms,
|
* Stores a contact associated with the given local and remote pseudonyms,
|
||||||
* and returns an ID for the contact.
|
* and returns an ID for the contact.
|
||||||
@@ -151,13 +171,13 @@ public interface DatabaseComponent {
|
|||||||
throws DbException;
|
throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a batch of raw messages for the given contact, with a total
|
* Returns a batch of messages for the given contact, with a total length
|
||||||
* length less than or equal to the given length, for transmission over a
|
* less than or equal to the given length, for transmission over a
|
||||||
* transport with the given maximum latency. Returns null if there are no
|
* transport with the given maximum latency. Returns null if there are no
|
||||||
* sendable messages that fit in the given length.
|
* sendable messages that fit in the given length.
|
||||||
*/
|
*/
|
||||||
@Nullable
|
@Nullable
|
||||||
Collection<byte[]> generateBatch(Transaction txn, ContactId c,
|
Collection<Message> generateBatch(Transaction txn, ContactId c,
|
||||||
int maxLength, int maxLatency) throws DbException;
|
int maxLength, int maxLatency) throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -178,14 +198,14 @@ public interface DatabaseComponent {
|
|||||||
throws DbException;
|
throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a batch of raw messages for the given contact, with a total
|
* Returns a batch of messages for the given contact, with a total length
|
||||||
* length less than or equal to the given length, for transmission over a
|
* less than or equal to the given length, for transmission over a
|
||||||
* transport with the given maximum latency. Only messages that have been
|
* transport with the given maximum latency. Only messages that have been
|
||||||
* requested by the contact are returned. Returns null if there are no
|
* requested by the contact are returned. Returns null if there are no
|
||||||
* sendable messages that fit in the given length.
|
* sendable messages that fit in the given length.
|
||||||
*/
|
*/
|
||||||
@Nullable
|
@Nullable
|
||||||
Collection<byte[]> generateRequestedBatch(Transaction txn, ContactId c,
|
Collection<Message> generateRequestedBatch(Transaction txn, ContactId c,
|
||||||
int maxLength, int maxLatency) throws DbException;
|
int maxLength, int maxLatency) throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -263,6 +283,15 @@ public interface DatabaseComponent {
|
|||||||
*/
|
*/
|
||||||
Collection<LocalAuthor> getLocalAuthors(Transaction txn) throws DbException;
|
Collection<LocalAuthor> getLocalAuthors(Transaction txn) throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the message with the given ID.
|
||||||
|
* <p/>
|
||||||
|
* Read-only.
|
||||||
|
*
|
||||||
|
* @throws MessageDeletedException if the message has been deleted
|
||||||
|
*/
|
||||||
|
Message getMessage(Transaction txn, MessageId m) throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the IDs of all delivered messages in the given group.
|
* Returns the IDs of all delivered messages in the given group.
|
||||||
* <p/>
|
* <p/>
|
||||||
@@ -297,15 +326,6 @@ public interface DatabaseComponent {
|
|||||||
Collection<MessageId> getMessagesToShare(Transaction txn)
|
Collection<MessageId> getMessagesToShare(Transaction txn)
|
||||||
throws DbException;
|
throws DbException;
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the message with the given ID, in serialised form, or null if
|
|
||||||
* the message has been deleted.
|
|
||||||
* <p/>
|
|
||||||
* Read-only.
|
|
||||||
*/
|
|
||||||
@Nullable
|
|
||||||
byte[] getRawMessage(Transaction txn, MessageId m) throws DbException;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the metadata for all delivered messages in the given group.
|
* Returns the metadata for all delivered messages in the given group.
|
||||||
* <p/>
|
* <p/>
|
||||||
@@ -353,12 +373,12 @@ public interface DatabaseComponent {
|
|||||||
/**
|
/**
|
||||||
* Returns the IDs and states of all dependencies of the given message.
|
* Returns the IDs and states of all dependencies of the given message.
|
||||||
* For missing dependencies and dependencies in other groups, the state
|
* For missing dependencies and dependencies in other groups, the state
|
||||||
* {@link State UNKNOWN} is returned.
|
* {@link MessageState UNKNOWN} is returned.
|
||||||
* <p/>
|
* <p/>
|
||||||
* Read-only.
|
* Read-only.
|
||||||
*/
|
*/
|
||||||
Map<MessageId, State> getMessageDependencies(Transaction txn, MessageId m)
|
Map<MessageId, MessageState> getMessageDependencies(Transaction txn,
|
||||||
throws DbException;
|
MessageId m) throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the IDs and states of all dependents of the given message.
|
* Returns the IDs and states of all dependents of the given message.
|
||||||
@@ -367,15 +387,16 @@ public interface DatabaseComponent {
|
|||||||
* <p/>
|
* <p/>
|
||||||
* Read-only.
|
* Read-only.
|
||||||
*/
|
*/
|
||||||
Map<MessageId, State> getMessageDependents(Transaction txn, MessageId m)
|
Map<MessageId, MessageState> getMessageDependents(Transaction txn,
|
||||||
throws DbException;
|
MessageId m) throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the validation and delivery state of the given message.
|
* Gets the validation and delivery state of the given message.
|
||||||
* <p/>
|
* <p/>
|
||||||
* Read-only.
|
* Read-only.
|
||||||
*/
|
*/
|
||||||
State getMessageState(Transaction txn, MessageId m) throws DbException;
|
MessageState getMessageState(Transaction txn, MessageId m)
|
||||||
|
throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the status of the given delivered message with respect to the
|
* Returns the status of the given delivered message with respect to the
|
||||||
@@ -502,6 +523,12 @@ public interface DatabaseComponent {
|
|||||||
void setContactActive(Transaction txn, ContactId c, boolean active)
|
void setContactActive(Transaction txn, ContactId c, boolean active)
|
||||||
throws DbException;
|
throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets an alias name for the contact or unsets it if alias is null.
|
||||||
|
*/
|
||||||
|
void setContactAlias(Transaction txn, ContactId c, @Nullable String alias)
|
||||||
|
throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the given group's visibility to the given contact.
|
* Sets the given group's visibility to the given contact.
|
||||||
*/
|
*/
|
||||||
@@ -516,7 +543,7 @@ public interface DatabaseComponent {
|
|||||||
/**
|
/**
|
||||||
* Sets the validation and delivery state of the given message.
|
* Sets the validation and delivery state of the given message.
|
||||||
*/
|
*/
|
||||||
void setMessageState(Transaction txn, MessageId m, State state)
|
void setMessageState(Transaction txn, MessageId m, MessageState state)
|
||||||
throws DbException;
|
throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -0,0 +1,9 @@
|
|||||||
|
package org.briarproject.bramble.api.db;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
@NotNullByDefault
|
||||||
|
public interface DbCallable<R, E extends Exception> {
|
||||||
|
|
||||||
|
R call(Transaction txn) throws DbException, E;
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
package org.briarproject.bramble.api.db;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
@NotNullByDefault
|
||||||
|
public interface DbRunnable<E extends Exception> {
|
||||||
|
|
||||||
|
void run(Transaction txn) throws DbException, E;
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
package org.briarproject.bramble.api.db;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Thrown when a message that has been deleted is requested from the database.
|
||||||
|
* This exception may occur due to concurrent updates and does not indicate a
|
||||||
|
* database error.
|
||||||
|
*/
|
||||||
|
public class MessageDeletedException extends DbException {
|
||||||
|
}
|
||||||
@@ -6,6 +6,10 @@ public interface MigrationListener {
|
|||||||
* This is called when a migration is started while opening the database.
|
* This is called when a migration is started while opening the database.
|
||||||
* It will be called once for each migration being applied.
|
* It will be called once for each migration being applied.
|
||||||
*/
|
*/
|
||||||
void onMigrationRun();
|
void onDatabaseMigration();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is called when compaction is started while opening the database.
|
||||||
|
*/
|
||||||
|
void onDatabaseCompaction();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,12 @@
|
|||||||
|
package org.briarproject.bramble.api.db;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
@NotNullByDefault
|
||||||
|
public interface NullableDbCallable<R, E extends Exception> {
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
R call(Transaction txn) throws DbException, E;
|
||||||
|
}
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
package org.briarproject.bramble.api.identity;
|
package org.briarproject.bramble.api.identity;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.Nameable;
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
import org.briarproject.bramble.util.StringUtils;
|
import org.briarproject.bramble.util.StringUtils;
|
||||||
|
|
||||||
@@ -13,11 +14,7 @@ import static org.briarproject.bramble.api.identity.AuthorConstants.MAX_PUBLIC_K
|
|||||||
*/
|
*/
|
||||||
@Immutable
|
@Immutable
|
||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
public class Author {
|
public class Author implements Nameable {
|
||||||
|
|
||||||
public enum Status {
|
|
||||||
NONE, ANONYMOUS, UNKNOWN, UNVERIFIED, VERIFIED, OURSELVES
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The current version of the author structure.
|
* The current version of the author structure.
|
||||||
|
|||||||
@@ -0,0 +1,56 @@
|
|||||||
|
package org.briarproject.bramble.api.identity;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
|
@Immutable
|
||||||
|
@NotNullByDefault
|
||||||
|
public class AuthorInfo {
|
||||||
|
|
||||||
|
public enum Status {
|
||||||
|
NONE, ANONYMOUS, UNKNOWN, UNVERIFIED, VERIFIED, OURSELVES;
|
||||||
|
|
||||||
|
public boolean isContact() {
|
||||||
|
return this == UNVERIFIED || this == VERIFIED;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private final Status status;
|
||||||
|
@Nullable
|
||||||
|
private final String alias;
|
||||||
|
|
||||||
|
public AuthorInfo(Status status, @Nullable String alias) {
|
||||||
|
this.status = status;
|
||||||
|
this.alias = alias;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AuthorInfo(Status status) {
|
||||||
|
this(status, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Status getStatus() {
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public String getAlias() {
|
||||||
|
return alias;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
int hashCode = status.ordinal();
|
||||||
|
if (alias != null) hashCode += alias.hashCode();
|
||||||
|
return hashCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
if (!(o instanceof AuthorInfo)) return false;
|
||||||
|
AuthorInfo info = (AuthorInfo) o;
|
||||||
|
return status == info.status &&
|
||||||
|
(alias == null ? info.alias == null : alias.equals(info.alias));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -3,7 +3,6 @@ package org.briarproject.bramble.api.identity;
|
|||||||
import org.briarproject.bramble.api.crypto.CryptoExecutor;
|
import org.briarproject.bramble.api.crypto.CryptoExecutor;
|
||||||
import org.briarproject.bramble.api.db.DbException;
|
import org.briarproject.bramble.api.db.DbException;
|
||||||
import org.briarproject.bramble.api.db.Transaction;
|
import org.briarproject.bramble.api.db.Transaction;
|
||||||
import org.briarproject.bramble.api.identity.Author.Status;
|
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
@@ -37,14 +36,4 @@ public interface IdentityManager {
|
|||||||
*/
|
*/
|
||||||
LocalAuthor getLocalAuthor(Transaction txn) throws DbException;
|
LocalAuthor getLocalAuthor(Transaction txn) throws DbException;
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the {@link Status} of the given author.
|
|
||||||
*/
|
|
||||||
Status getAuthorStatus(AuthorId a) throws DbException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the {@link Status} of the given author.
|
|
||||||
*/
|
|
||||||
Status getAuthorStatus(Transaction txn, AuthorId a) throws DbException;
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,13 @@ package org.briarproject.bramble.api.keyagreement;
|
|||||||
public interface KeyAgreementConstants {
|
public interface KeyAgreementConstants {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The current version of the BQP protocol. Version number 89 is reserved.
|
* The version of the BQP protocol used in beta releases. This version
|
||||||
|
* number is reserved.
|
||||||
|
*/
|
||||||
|
byte BETA_PROTOCOL_VERSION = 89;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The current version of the BQP protocol.
|
||||||
*/
|
*/
|
||||||
byte PROTOCOL_VERSION = 4;
|
byte PROTOCOL_VERSION = 4;
|
||||||
|
|
||||||
@@ -15,7 +21,7 @@ public interface KeyAgreementConstants {
|
|||||||
/**
|
/**
|
||||||
* The connection timeout in milliseconds.
|
* The connection timeout in milliseconds.
|
||||||
*/
|
*/
|
||||||
long CONNECTION_TIMEOUT = 20 * 1000;
|
long CONNECTION_TIMEOUT = 60_000;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The transport identifier for Bluetooth.
|
* The transport identifier for Bluetooth.
|
||||||
|
|||||||
@@ -0,0 +1,20 @@
|
|||||||
|
package org.briarproject.bramble.api.keyagreement;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Thrown when a QR code that has been scanned uses a protocol version that is
|
||||||
|
* not supported.
|
||||||
|
*/
|
||||||
|
public class UnsupportedVersionException extends IOException {
|
||||||
|
|
||||||
|
private final boolean tooOld;
|
||||||
|
|
||||||
|
public UnsupportedVersionException(boolean tooOld) {
|
||||||
|
this.tooOld = tooOld;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isTooOld() {
|
||||||
|
return tooOld;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -34,7 +34,8 @@ public interface LifecycleManager {
|
|||||||
*/
|
*/
|
||||||
enum LifecycleState {
|
enum LifecycleState {
|
||||||
|
|
||||||
STARTING, MIGRATING_DATABASE, STARTING_SERVICES, RUNNING, STOPPING;
|
STARTING, MIGRATING_DATABASE, COMPACTING_DATABASE, STARTING_SERVICES,
|
||||||
|
RUNNING, STOPPING;
|
||||||
|
|
||||||
public boolean isAfter(LifecycleState state) {
|
public boolean isAfter(LifecycleState state) {
|
||||||
return ordinal() > state.ordinal();
|
return ordinal() > state.ordinal();
|
||||||
|
|||||||
@@ -0,0 +1,15 @@
|
|||||||
|
package org.briarproject.bramble.api.nullsafety;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
@NotNullByDefault
|
||||||
|
public class NullSafety {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stand-in for `Objects.requireNonNull()`.
|
||||||
|
*/
|
||||||
|
public static <T> T requireNonNull(@Nullable T t) {
|
||||||
|
if (t == null) throw new NullPointerException();
|
||||||
|
return t;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,7 +4,8 @@ public interface TorConstants {
|
|||||||
|
|
||||||
TransportId ID = new TransportId("org.briarproject.bramble.tor");
|
TransportId ID = new TransportId("org.briarproject.bramble.tor");
|
||||||
|
|
||||||
String PROP_ONION = "onion";
|
String PROP_ONION_V2 = "onion";
|
||||||
|
String PROP_ONION_V3 = "onion3";
|
||||||
|
|
||||||
int SOCKS_PORT = 59050;
|
int SOCKS_PORT = 59050;
|
||||||
int CONTROL_PORT = 59051;
|
int CONTROL_PORT = 59051;
|
||||||
@@ -12,12 +13,14 @@ public interface TorConstants {
|
|||||||
int CONNECT_TO_PROXY_TIMEOUT = 5000; // Milliseconds
|
int CONNECT_TO_PROXY_TIMEOUT = 5000; // Milliseconds
|
||||||
int EXTRA_SOCKET_TIMEOUT = 30000; // Milliseconds
|
int EXTRA_SOCKET_TIMEOUT = 30000; // Milliseconds
|
||||||
|
|
||||||
String PREF_TOR_NETWORK = "network";
|
String PREF_TOR_NETWORK = "network2";
|
||||||
String PREF_TOR_PORT = "port";
|
String PREF_TOR_PORT = "port";
|
||||||
String PREF_TOR_DISABLE_BLOCKED = "disableWhenBlocked";
|
String PREF_TOR_MOBILE = "useMobileData";
|
||||||
|
String PREF_TOR_ONLY_WHEN_CHARGING = "onlyWhenCharging";
|
||||||
|
|
||||||
int PREF_TOR_NETWORK_NEVER = 0;
|
int PREF_TOR_NETWORK_AUTOMATIC = 0;
|
||||||
int PREF_TOR_NETWORK_WIFI = 1;
|
int PREF_TOR_NETWORK_WITHOUT_BRIDGES = 1;
|
||||||
int PREF_TOR_NETWORK_ALWAYS = 2;
|
int PREF_TOR_NETWORK_WITH_BRIDGES = 2;
|
||||||
|
int PREF_TOR_NETWORK_NEVER = 3;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package org.briarproject.bramble.api.settings;
|
package org.briarproject.bramble.api.settings;
|
||||||
|
|
||||||
import org.briarproject.bramble.api.db.DbException;
|
import org.briarproject.bramble.api.db.DbException;
|
||||||
|
import org.briarproject.bramble.api.db.Transaction;
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
|
||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
@@ -11,6 +12,11 @@ public interface SettingsManager {
|
|||||||
*/
|
*/
|
||||||
Settings getSettings(String namespace) throws DbException;
|
Settings getSettings(String namespace) throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns all settings in the given namespace.
|
||||||
|
*/
|
||||||
|
Settings getSettings(Transaction txn, String namespace) throws DbException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Merges the given settings with any existing settings in the given
|
* Merges the given settings with any existing settings in the given
|
||||||
* namespace.
|
* namespace.
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package org.briarproject.bramble.api.settings.event;
|
|||||||
|
|
||||||
import org.briarproject.bramble.api.event.Event;
|
import org.briarproject.bramble.api.event.Event;
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
import org.briarproject.bramble.api.settings.Settings;
|
||||||
|
|
||||||
import javax.annotation.concurrent.Immutable;
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
@@ -13,12 +14,18 @@ import javax.annotation.concurrent.Immutable;
|
|||||||
public class SettingsUpdatedEvent extends Event {
|
public class SettingsUpdatedEvent extends Event {
|
||||||
|
|
||||||
private final String namespace;
|
private final String namespace;
|
||||||
|
private final Settings settings;
|
||||||
|
|
||||||
public SettingsUpdatedEvent(String namespace) {
|
public SettingsUpdatedEvent(String namespace, Settings settings) {
|
||||||
this.namespace = namespace;
|
this.namespace = namespace;
|
||||||
|
this.settings = settings;
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getNamespace() {
|
public String getNamespace() {
|
||||||
return namespace;
|
return namespace;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Settings getSettings() {
|
||||||
|
return settings;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
package org.briarproject.bramble.api.sync;
|
package org.briarproject.bramble.api.sync;
|
||||||
|
|
||||||
import static org.briarproject.bramble.api.sync.SyncConstants.MAX_MESSAGE_LENGTH;
|
import static org.briarproject.bramble.api.sync.SyncConstants.MAX_MESSAGE_BODY_LENGTH;
|
||||||
import static org.briarproject.bramble.api.sync.SyncConstants.MESSAGE_HEADER_LENGTH;
|
import static org.briarproject.bramble.api.sync.SyncConstants.MESSAGE_HEADER_LENGTH;
|
||||||
|
|
||||||
public class Message {
|
public class Message {
|
||||||
@@ -13,17 +13,16 @@ public class Message {
|
|||||||
private final MessageId id;
|
private final MessageId id;
|
||||||
private final GroupId groupId;
|
private final GroupId groupId;
|
||||||
private final long timestamp;
|
private final long timestamp;
|
||||||
private final byte[] raw;
|
private final byte[] body;
|
||||||
|
|
||||||
public Message(MessageId id, GroupId groupId, long timestamp, byte[] raw) {
|
public Message(MessageId id, GroupId groupId, long timestamp, byte[] body) {
|
||||||
if (raw.length <= MESSAGE_HEADER_LENGTH)
|
if (body.length == 0) throw new IllegalArgumentException();
|
||||||
throw new IllegalArgumentException();
|
if (body.length > MAX_MESSAGE_BODY_LENGTH)
|
||||||
if (raw.length > MAX_MESSAGE_LENGTH)
|
|
||||||
throw new IllegalArgumentException();
|
throw new IllegalArgumentException();
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.groupId = groupId;
|
this.groupId = groupId;
|
||||||
this.timestamp = timestamp;
|
this.timestamp = timestamp;
|
||||||
this.raw = raw;
|
this.body = body;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -50,15 +49,15 @@ public class Message {
|
|||||||
/**
|
/**
|
||||||
* Returns the length of the raw message in bytes.
|
* Returns the length of the raw message in bytes.
|
||||||
*/
|
*/
|
||||||
public int getLength() {
|
public int getRawLength() {
|
||||||
return raw.length;
|
return MESSAGE_HEADER_LENGTH + body.length;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the raw message.
|
* Returns the message body.
|
||||||
*/
|
*/
|
||||||
public byte[] getRaw() {
|
public byte[] getBody() {
|
||||||
return raw;
|
return body;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|||||||
@@ -9,5 +9,5 @@ public interface MessageFactory {
|
|||||||
|
|
||||||
Message createMessage(byte[] raw);
|
Message createMessage(byte[] raw);
|
||||||
|
|
||||||
Message createMessage(MessageId m, byte[] raw);
|
byte[] getRawMessage(Message m);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ public interface SyncRecordWriter {
|
|||||||
|
|
||||||
void writeAck(Ack a) throws IOException;
|
void writeAck(Ack a) throws IOException;
|
||||||
|
|
||||||
void writeMessage(byte[] raw) throws IOException;
|
void writeMessage(Message m) throws IOException;
|
||||||
|
|
||||||
void writeOffer(Offer o) throws IOException;
|
void writeOffer(Offer o) throws IOException;
|
||||||
|
|
||||||
|
|||||||
@@ -1,87 +0,0 @@
|
|||||||
package org.briarproject.bramble.api.sync;
|
|
||||||
|
|
||||||
import org.briarproject.bramble.api.crypto.SecretKey;
|
|
||||||
import org.briarproject.bramble.api.db.DbException;
|
|
||||||
import org.briarproject.bramble.api.db.Metadata;
|
|
||||||
import org.briarproject.bramble.api.db.Transaction;
|
|
||||||
import org.briarproject.bramble.api.lifecycle.LifecycleManager;
|
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Responsible for managing message validators and passing them messages to
|
|
||||||
* validate.
|
|
||||||
*/
|
|
||||||
@NotNullByDefault
|
|
||||||
public interface ValidationManager {
|
|
||||||
|
|
||||||
enum State {
|
|
||||||
|
|
||||||
UNKNOWN(0), INVALID(1), PENDING(2), DELIVERED(3);
|
|
||||||
|
|
||||||
private final int value;
|
|
||||||
|
|
||||||
State(int value) {
|
|
||||||
this.value = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getValue() {
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static State fromValue(int value) {
|
|
||||||
for (State s : values()) if (s.value == value) return s;
|
|
||||||
throw new IllegalArgumentException();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Registers the message validator for the given client. This method
|
|
||||||
* should be called before
|
|
||||||
* {@link LifecycleManager#startServices(SecretKey)}.
|
|
||||||
*/
|
|
||||||
void registerMessageValidator(ClientId c, int majorVersion,
|
|
||||||
MessageValidator v);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Registers the incoming message hook for the given client. The hook will
|
|
||||||
* be called once for each incoming message that passes validation. This
|
|
||||||
* method should be called before
|
|
||||||
* {@link LifecycleManager#startServices(SecretKey)}.
|
|
||||||
*/
|
|
||||||
void registerIncomingMessageHook(ClientId c, int majorVersion,
|
|
||||||
IncomingMessageHook hook);
|
|
||||||
|
|
||||||
interface MessageValidator {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Validates the given message and returns its metadata and
|
|
||||||
* dependencies.
|
|
||||||
*/
|
|
||||||
MessageContext validateMessage(Message m, Group g)
|
|
||||||
throws InvalidMessageException;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface IncomingMessageHook {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Called once for each incoming message that passes validation.
|
|
||||||
*
|
|
||||||
* @return whether or not this message should be shared
|
|
||||||
* @throws DbException Should only be used for real database errors.
|
|
||||||
* If this is thrown, delivery will be attempted again at next startup,
|
|
||||||
* whereas if an InvalidMessageException is thrown,
|
|
||||||
* the message will be permanently invalidated.
|
|
||||||
* @throws InvalidMessageException for any non-database error
|
|
||||||
* that occurs while handling remotely created data.
|
|
||||||
* This includes errors that occur while handling locally created data
|
|
||||||
* in a context controlled by remotely created data
|
|
||||||
* (for example, parsing the metadata of a dependency
|
|
||||||
* of an incoming message).
|
|
||||||
* Throwing this will delete the incoming message and its metadata
|
|
||||||
* marking it as invalid in the database.
|
|
||||||
* Never rethrow DbException as InvalidMessageException!
|
|
||||||
*/
|
|
||||||
boolean incomingMessage(Transaction txn, Message m, Metadata meta)
|
|
||||||
throws DbException, InvalidMessageException;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -3,11 +3,10 @@ package org.briarproject.bramble.api.sync.event;
|
|||||||
import org.briarproject.bramble.api.event.Event;
|
import org.briarproject.bramble.api.event.Event;
|
||||||
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
import org.briarproject.bramble.api.sync.MessageId;
|
import org.briarproject.bramble.api.sync.MessageId;
|
||||||
|
import org.briarproject.bramble.api.sync.validation.MessageState;
|
||||||
|
|
||||||
import javax.annotation.concurrent.Immutable;
|
import javax.annotation.concurrent.Immutable;
|
||||||
|
|
||||||
import static org.briarproject.bramble.api.sync.ValidationManager.State;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An event that is broadcast when a message state changed.
|
* An event that is broadcast when a message state changed.
|
||||||
*/
|
*/
|
||||||
@@ -17,10 +16,10 @@ public class MessageStateChangedEvent extends Event {
|
|||||||
|
|
||||||
private final MessageId messageId;
|
private final MessageId messageId;
|
||||||
private final boolean local;
|
private final boolean local;
|
||||||
private final State state;
|
private final MessageState state;
|
||||||
|
|
||||||
public MessageStateChangedEvent(MessageId messageId, boolean local,
|
public MessageStateChangedEvent(MessageId messageId, boolean local,
|
||||||
State state) {
|
MessageState state) {
|
||||||
this.messageId = messageId;
|
this.messageId = messageId;
|
||||||
this.local = local;
|
this.local = local;
|
||||||
this.state = state;
|
this.state = state;
|
||||||
@@ -34,7 +33,7 @@ public class MessageStateChangedEvent extends Event {
|
|||||||
return local;
|
return local;
|
||||||
}
|
}
|
||||||
|
|
||||||
public State getState() {
|
public MessageState getState() {
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,31 @@
|
|||||||
|
package org.briarproject.bramble.api.sync.validation;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.db.DbException;
|
||||||
|
import org.briarproject.bramble.api.db.Metadata;
|
||||||
|
import org.briarproject.bramble.api.db.Transaction;
|
||||||
|
import org.briarproject.bramble.api.sync.InvalidMessageException;
|
||||||
|
import org.briarproject.bramble.api.sync.Message;
|
||||||
|
|
||||||
|
public interface IncomingMessageHook {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Called once for each incoming message that passes validation.
|
||||||
|
*
|
||||||
|
* @return whether or not this message should be shared
|
||||||
|
* @throws DbException Should only be used for real database errors.
|
||||||
|
* If this is thrown, delivery will be attempted again at next startup,
|
||||||
|
* whereas if an InvalidMessageException is thrown,
|
||||||
|
* the message will be permanently invalidated.
|
||||||
|
* @throws InvalidMessageException for any non-database error
|
||||||
|
* that occurs while handling remotely created data.
|
||||||
|
* This includes errors that occur while handling locally created data
|
||||||
|
* in a context controlled by remotely created data
|
||||||
|
* (for example, parsing the metadata of a dependency
|
||||||
|
* of an incoming message).
|
||||||
|
* Throwing this will delete the incoming message and its metadata
|
||||||
|
* marking it as invalid in the database.
|
||||||
|
* Never rethrow DbException as InvalidMessageException!
|
||||||
|
*/
|
||||||
|
boolean incomingMessage(Transaction txn, Message m, Metadata meta)
|
||||||
|
throws DbException, InvalidMessageException;
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
package org.briarproject.bramble.api.sync.validation;
|
||||||
|
|
||||||
|
public enum MessageState {
|
||||||
|
|
||||||
|
UNKNOWN(0), INVALID(1), PENDING(2), DELIVERED(3);
|
||||||
|
|
||||||
|
private final int value;
|
||||||
|
|
||||||
|
MessageState(int value) {
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getValue() {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static MessageState fromValue(int value) {
|
||||||
|
for (MessageState s : values()) if (s.value == value) return s;
|
||||||
|
throw new IllegalArgumentException();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
package org.briarproject.bramble.api.sync.validation;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.sync.Group;
|
||||||
|
import org.briarproject.bramble.api.sync.InvalidMessageException;
|
||||||
|
import org.briarproject.bramble.api.sync.Message;
|
||||||
|
import org.briarproject.bramble.api.sync.MessageContext;
|
||||||
|
|
||||||
|
public interface MessageValidator {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates the given message and returns its metadata and
|
||||||
|
* dependencies.
|
||||||
|
*/
|
||||||
|
MessageContext validateMessage(Message m, Group g)
|
||||||
|
throws InvalidMessageException;
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
package org.briarproject.bramble.api.sync.validation;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.crypto.SecretKey;
|
||||||
|
import org.briarproject.bramble.api.lifecycle.LifecycleManager;
|
||||||
|
import org.briarproject.bramble.api.nullsafety.NotNullByDefault;
|
||||||
|
import org.briarproject.bramble.api.sync.ClientId;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Responsible for managing message validators and passing them messages to
|
||||||
|
* validate.
|
||||||
|
*/
|
||||||
|
@NotNullByDefault
|
||||||
|
public interface ValidationManager {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers the {@link MessageValidator} for the given client. This method
|
||||||
|
* should be called before
|
||||||
|
* {@link LifecycleManager#startServices(SecretKey)}.
|
||||||
|
*/
|
||||||
|
void registerMessageValidator(ClientId c, int majorVersion,
|
||||||
|
MessageValidator v);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers the {@link IncomingMessageHook} for the given client. The hook
|
||||||
|
* will be called once for each incoming message that passes validation.
|
||||||
|
* This method should be called before
|
||||||
|
* {@link LifecycleManager#startServices(SecretKey)}.
|
||||||
|
*/
|
||||||
|
void registerIncomingMessageHook(ClientId c, int majorVersion,
|
||||||
|
IncomingMessageHook hook);
|
||||||
|
}
|
||||||
@@ -7,5 +7,5 @@ import java.io.InputStream;
|
|||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
public interface ResourceProvider {
|
public interface ResourceProvider {
|
||||||
|
|
||||||
InputStream getResourceInputStream(String name);
|
InputStream getResourceInputStream(String name, String extension);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -38,6 +38,13 @@ public interface ClientVersioningManager {
|
|||||||
Visibility getClientVisibility(Transaction txn, ContactId contactId,
|
Visibility getClientVisibility(Transaction txn, ContactId contactId,
|
||||||
ClientId clientId, int majorVersion) throws DbException;
|
ClientId clientId, int majorVersion) throws DbException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the minor version of the given client that is supported by the
|
||||||
|
* given contact, or -1 if the contact does not support the client.
|
||||||
|
*/
|
||||||
|
int getClientMinorVersion(Transaction txn, ContactId contactId,
|
||||||
|
ClientId clientId, int majorVersion) throws DbException;
|
||||||
|
|
||||||
interface ClientVersioningHook {
|
interface ClientVersioningHook {
|
||||||
|
|
||||||
void onClientVisibilityChanging(Transaction txn, Contact c,
|
void onClientVisibilityChanging(Transaction txn, Contact c,
|
||||||
|
|||||||
@@ -8,12 +8,15 @@ import java.io.File;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
|
import java.net.ServerSocket;
|
||||||
import java.net.Socket;
|
import java.net.Socket;
|
||||||
|
import java.util.logging.Level;
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
import static java.util.logging.Level.WARNING;
|
import static java.util.logging.Level.WARNING;
|
||||||
|
import static org.briarproject.bramble.util.LogUtils.logException;
|
||||||
|
|
||||||
@NotNullByDefault
|
@NotNullByDefault
|
||||||
public class IoUtils {
|
public class IoUtils {
|
||||||
@@ -54,16 +57,35 @@ public class IoUtils {
|
|||||||
out.flush();
|
out.flush();
|
||||||
out.close();
|
out.close();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
tryToClose(in);
|
tryToClose(in, LOG, WARNING);
|
||||||
tryToClose(out);
|
tryToClose(out, LOG, WARNING);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void tryToClose(@Nullable Closeable c) {
|
public static void tryToClose(@Nullable Closeable c, Logger logger,
|
||||||
|
Level level) {
|
||||||
try {
|
try {
|
||||||
if (c != null) c.close();
|
if (c != null) c.close();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// We did our best
|
logException(logger, level, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void tryToClose(@Nullable Socket s, Logger logger,
|
||||||
|
Level level) {
|
||||||
|
try {
|
||||||
|
if (s != null) s.close();
|
||||||
|
} catch (IOException e) {
|
||||||
|
logException(logger, level, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void tryToClose(@Nullable ServerSocket ss, Logger logger,
|
||||||
|
Level level) {
|
||||||
|
try {
|
||||||
|
if (ss != null) ss.close();
|
||||||
|
} catch (IOException e) {
|
||||||
|
logException(logger, level, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -10,8 +10,6 @@ public class OsUtils {
|
|||||||
@Nullable
|
@Nullable
|
||||||
private static final String os = System.getProperty("os.name");
|
private static final String os = System.getProperty("os.name");
|
||||||
@Nullable
|
@Nullable
|
||||||
private static final String version = System.getProperty("os.version");
|
|
||||||
@Nullable
|
|
||||||
private static final String vendor = System.getProperty("java.vendor");
|
private static final String vendor = System.getProperty("java.vendor");
|
||||||
|
|
||||||
public static boolean isWindows() {
|
public static boolean isWindows() {
|
||||||
|
|||||||
@@ -47,7 +47,7 @@ public class StringUtils {
|
|||||||
try {
|
try {
|
||||||
return s.getBytes("UTF-8");
|
return s.getBytes("UTF-8");
|
||||||
} catch (UnsupportedEncodingException e) {
|
} catch (UnsupportedEncodingException e) {
|
||||||
throw new RuntimeException(e);
|
throw new AssertionError(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -63,7 +63,7 @@ public class StringUtils {
|
|||||||
try {
|
try {
|
||||||
return decoder.decode(buffer).toString();
|
return decoder.decode(buffer).toString();
|
||||||
} catch (CharacterCodingException e) {
|
} catch (CharacterCodingException e) {
|
||||||
throw new RuntimeException(e);
|
throw new AssertionError(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,42 @@
|
|||||||
|
package org.briarproject.bramble.api.identity;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.test.BrambleTestCase;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import static org.briarproject.bramble.api.identity.AuthorInfo.Status.NONE;
|
||||||
|
import static org.briarproject.bramble.api.identity.AuthorInfo.Status.VERIFIED;
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNotEquals;
|
||||||
|
|
||||||
|
public class AuthorInfoTest extends BrambleTestCase {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEquals() {
|
||||||
|
assertEquals(
|
||||||
|
new AuthorInfo(NONE),
|
||||||
|
new AuthorInfo(NONE, null)
|
||||||
|
);
|
||||||
|
assertEquals(
|
||||||
|
new AuthorInfo(NONE, "test"),
|
||||||
|
new AuthorInfo(NONE, "test")
|
||||||
|
);
|
||||||
|
|
||||||
|
assertNotEquals(
|
||||||
|
new AuthorInfo(NONE),
|
||||||
|
new AuthorInfo(VERIFIED)
|
||||||
|
);
|
||||||
|
assertNotEquals(
|
||||||
|
new AuthorInfo(NONE, "test"),
|
||||||
|
new AuthorInfo(NONE)
|
||||||
|
);
|
||||||
|
assertNotEquals(
|
||||||
|
new AuthorInfo(NONE),
|
||||||
|
new AuthorInfo(NONE, "test")
|
||||||
|
);
|
||||||
|
assertNotEquals(
|
||||||
|
new AuthorInfo(NONE, "a"),
|
||||||
|
new AuthorInfo(NONE, "b")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
package org.briarproject.bramble.test;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.system.Clock;
|
||||||
|
|
||||||
|
public class ArrayClock implements Clock {
|
||||||
|
|
||||||
|
private final long[] times;
|
||||||
|
private int index = 0;
|
||||||
|
|
||||||
|
public ArrayClock(long... times) {
|
||||||
|
this.times = times;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long currentTimeMillis() {
|
||||||
|
return times[index++];
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void sleep(long milliseconds) throws InterruptedException {
|
||||||
|
Thread.sleep(milliseconds);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
package org.briarproject.bramble.test;
|
||||||
|
|
||||||
|
import org.briarproject.bramble.api.system.Clock;
|
||||||
|
|
||||||
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
|
||||||
|
public class SettableClock implements Clock {
|
||||||
|
|
||||||
|
private final AtomicLong time;
|
||||||
|
|
||||||
|
public SettableClock(AtomicLong time) {
|
||||||
|
this.time = time;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long currentTimeMillis() {
|
||||||
|
return time.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void sleep(long milliseconds) throws InterruptedException {
|
||||||
|
Thread.sleep(milliseconds);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -24,6 +24,7 @@ import java.util.Map;
|
|||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
|
import static java.util.Arrays.asList;
|
||||||
import static org.briarproject.bramble.api.identity.Author.FORMAT_VERSION;
|
import static org.briarproject.bramble.api.identity.Author.FORMAT_VERSION;
|
||||||
import static org.briarproject.bramble.api.identity.AuthorConstants.MAX_AUTHOR_NAME_LENGTH;
|
import static org.briarproject.bramble.api.identity.AuthorConstants.MAX_AUTHOR_NAME_LENGTH;
|
||||||
import static org.briarproject.bramble.api.identity.AuthorConstants.MAX_PUBLIC_KEY_LENGTH;
|
import static org.briarproject.bramble.api.identity.AuthorConstants.MAX_PUBLIC_KEY_LENGTH;
|
||||||
@@ -32,7 +33,6 @@ import static org.briarproject.bramble.api.properties.TransportPropertyConstants
|
|||||||
import static org.briarproject.bramble.api.sync.ClientId.MAX_CLIENT_ID_LENGTH;
|
import static org.briarproject.bramble.api.sync.ClientId.MAX_CLIENT_ID_LENGTH;
|
||||||
import static org.briarproject.bramble.api.sync.SyncConstants.MAX_GROUP_DESCRIPTOR_LENGTH;
|
import static org.briarproject.bramble.api.sync.SyncConstants.MAX_GROUP_DESCRIPTOR_LENGTH;
|
||||||
import static org.briarproject.bramble.api.sync.SyncConstants.MAX_MESSAGE_BODY_LENGTH;
|
import static org.briarproject.bramble.api.sync.SyncConstants.MAX_MESSAGE_BODY_LENGTH;
|
||||||
import static org.briarproject.bramble.api.sync.SyncConstants.MESSAGE_HEADER_LENGTH;
|
|
||||||
import static org.briarproject.bramble.util.StringUtils.getRandomString;
|
import static org.briarproject.bramble.util.StringUtils.getRandomString;
|
||||||
|
|
||||||
public class TestUtils {
|
public class TestUtils {
|
||||||
@@ -40,6 +40,7 @@ public class TestUtils {
|
|||||||
private static final AtomicInteger nextTestDir =
|
private static final AtomicInteger nextTestDir =
|
||||||
new AtomicInteger((int) (Math.random() * 1000 * 1000));
|
new AtomicInteger((int) (Math.random() * 1000 * 1000));
|
||||||
private static final Random random = new Random();
|
private static final Random random = new Random();
|
||||||
|
private static final long timestamp = System.currentTimeMillis();
|
||||||
|
|
||||||
public static File getTestDirectory() {
|
public static File getTestDirectory() {
|
||||||
int name = nextTestDir.getAndIncrement();
|
int name = nextTestDir.getAndIncrement();
|
||||||
@@ -101,9 +102,8 @@ public class TestUtils {
|
|||||||
String name = getRandomString(nameLength);
|
String name = getRandomString(nameLength);
|
||||||
byte[] publicKey = getRandomBytes(MAX_PUBLIC_KEY_LENGTH);
|
byte[] publicKey = getRandomBytes(MAX_PUBLIC_KEY_LENGTH);
|
||||||
byte[] privateKey = getRandomBytes(MAX_PUBLIC_KEY_LENGTH);
|
byte[] privateKey = getRandomBytes(MAX_PUBLIC_KEY_LENGTH);
|
||||||
long created = System.currentTimeMillis();
|
|
||||||
return new LocalAuthor(id, FORMAT_VERSION, name, publicKey, privateKey,
|
return new LocalAuthor(id, FORMAT_VERSION, name, publicKey, privateKey,
|
||||||
created);
|
timestamp);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Author getAuthor() {
|
public static Author getAuthor() {
|
||||||
@@ -131,14 +131,13 @@ public class TestUtils {
|
|||||||
|
|
||||||
public static Message getMessage(GroupId groupId) {
|
public static Message getMessage(GroupId groupId) {
|
||||||
int bodyLength = 1 + random.nextInt(MAX_MESSAGE_BODY_LENGTH);
|
int bodyLength = 1 + random.nextInt(MAX_MESSAGE_BODY_LENGTH);
|
||||||
return getMessage(groupId, MESSAGE_HEADER_LENGTH + bodyLength);
|
return getMessage(groupId, bodyLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Message getMessage(GroupId groupId, int rawLength) {
|
public static Message getMessage(GroupId groupId, int bodyLength) {
|
||||||
MessageId id = new MessageId(getRandomId());
|
MessageId id = new MessageId(getRandomId());
|
||||||
byte[] raw = getRandomBytes(rawLength);
|
byte[] body = getRandomBytes(bodyLength);
|
||||||
long timestamp = System.currentTimeMillis();
|
return new Message(id, groupId, timestamp, body);
|
||||||
return new Message(id, groupId, timestamp, raw);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static double getMedian(Collection<? extends Number> samples) {
|
public static double getMedian(Collection<? extends Number> samples) {
|
||||||
@@ -174,4 +173,10 @@ public class TestUtils {
|
|||||||
Collection<? extends Number> samples) {
|
Collection<? extends Number> samples) {
|
||||||
return Math.sqrt(getVariance(samples));
|
return Math.sqrt(getVariance(samples));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static boolean isOptionalTestEnabled(Class testClass) {
|
||||||
|
String optionalTests = System.getenv("OPTIONAL_TESTS");
|
||||||
|
return optionalTests != null &&
|
||||||
|
asList(optionalTests.split(",")).contains(testClass.getName());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ dependencyVerification {
|
|||||||
verify = [
|
verify = [
|
||||||
'cglib:cglib:3.2.0:cglib-3.2.0.jar:adb13bab79712ad6bdf1bd59f2a3918018a8016e722e8a357065afb9e6690861',
|
'cglib:cglib:3.2.0:cglib-3.2.0.jar:adb13bab79712ad6bdf1bd59f2a3918018a8016e722e8a357065afb9e6690861',
|
||||||
'com.google.code.findbugs:jsr305:3.0.2:jsr305-3.0.2.jar:766ad2a0783f2687962c8ad74ceecc38a28b9f72a2d085ee438b7813e928d0c7',
|
'com.google.code.findbugs:jsr305:3.0.2:jsr305-3.0.2.jar:766ad2a0783f2687962c8ad74ceecc38a28b9f72a2d085ee438b7813e928d0c7',
|
||||||
'com.google.dagger:dagger:2.0.2:dagger-2.0.2.jar:84c0282ed8be73a29e0475d639da030b55dee72369e58dd35ae7d4fe6243dcf9',
|
'com.google.dagger:dagger:2.19:dagger-2.19.jar:514b6f1e0727c6572e1d65cb27e4ae668b7aeaeb93a29515182965265b609939',
|
||||||
'javax.inject:javax.inject:1:javax.inject-1.jar:91c77044a50c481636c32d916fd89c9118a72195390452c81065080f957de7ff',
|
'javax.inject:javax.inject:1:javax.inject-1.jar:91c77044a50c481636c32d916fd89c9118a72195390452c81065080f957de7ff',
|
||||||
'junit:junit:4.12:junit-4.12.jar:59721f0805e223d84b90677887d9ff567dc534d7c502ca903c0c2b17f05c116a',
|
'junit:junit:4.12:junit-4.12.jar:59721f0805e223d84b90677887d9ff567dc534d7c502ca903c0c2b17f05c116a',
|
||||||
'org.apache.ant:ant-launcher:1.9.4:ant-launcher-1.9.4.jar:7bccea20b41801ca17bcbc909a78c835d0f443f12d639c77bd6ae3d05861608d',
|
'org.apache.ant:ant-launcher:1.9.4:ant-launcher-1.9.4.jar:7bccea20b41801ca17bcbc909a78c835d0f443f12d639c77bd6ae3d05861608d',
|
||||||
|
|||||||
@@ -3,10 +3,10 @@ sourceCompatibility = 1.8
|
|||||||
targetCompatibility = 1.8
|
targetCompatibility = 1.8
|
||||||
|
|
||||||
apply plugin: 'ru.vyarus.animalsniffer'
|
apply plugin: 'ru.vyarus.animalsniffer'
|
||||||
apply plugin: 'net.ltgt.apt'
|
|
||||||
apply plugin: 'idea'
|
apply plugin: 'idea'
|
||||||
apply plugin: 'witness'
|
apply plugin: 'witness'
|
||||||
apply from: 'witness.gradle'
|
apply from: 'witness.gradle'
|
||||||
|
apply from: '../dagger.gradle'
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation project(path: ':bramble-api', configuration: 'default')
|
implementation project(path: ':bramble-api', configuration: 'default')
|
||||||
@@ -14,10 +14,9 @@ dependencies {
|
|||||||
implementation 'com.h2database:h2:1.4.192' // The last version that supports Java 1.6
|
implementation 'com.h2database:h2:1.4.192' // The last version that supports Java 1.6
|
||||||
implementation 'org.bitlet:weupnp:0.1.4'
|
implementation 'org.bitlet:weupnp:0.1.4'
|
||||||
implementation 'net.i2p.crypto:eddsa:0.2.0'
|
implementation 'net.i2p.crypto:eddsa:0.2.0'
|
||||||
implementation 'org.whispersystems:curve25519-java:0.4.1'
|
implementation 'org.whispersystems:curve25519-java:0.5.0'
|
||||||
implementation 'org.briarproject:jtorctl:0.3'
|
|
||||||
|
|
||||||
apt 'com.google.dagger:dagger-compiler:2.0.2'
|
annotationProcessor 'com.google.dagger:dagger-compiler:2.19'
|
||||||
|
|
||||||
testImplementation project(path: ':bramble-api', configuration: 'testOutput')
|
testImplementation project(path: ':bramble-api', configuration: 'testOutput')
|
||||||
testImplementation 'org.hsqldb:hsqldb:2.3.5' // The last version that supports Java 1.6
|
testImplementation 'org.hsqldb:hsqldb:2.3.5' // The last version that supports Java 1.6
|
||||||
@@ -25,15 +24,13 @@ dependencies {
|
|||||||
testImplementation "org.jmock:jmock:2.8.2"
|
testImplementation "org.jmock:jmock:2.8.2"
|
||||||
testImplementation "org.jmock:jmock-junit4:2.8.2"
|
testImplementation "org.jmock:jmock-junit4:2.8.2"
|
||||||
testImplementation "org.jmock:jmock-legacy:2.8.2"
|
testImplementation "org.jmock:jmock-legacy:2.8.2"
|
||||||
testImplementation "org.hamcrest:hamcrest-library:1.3"
|
|
||||||
testImplementation "org.hamcrest:hamcrest-core:1.3"
|
|
||||||
|
|
||||||
testApt 'com.google.dagger:dagger-compiler:2.0.2'
|
testAnnotationProcessor 'com.google.dagger:dagger-compiler:2.19'
|
||||||
|
|
||||||
signature 'org.codehaus.mojo.signature:java16:1.1@signature'
|
signature 'org.codehaus.mojo.signature:java16:1.1@signature'
|
||||||
}
|
}
|
||||||
|
|
||||||
// needed to make test output available to bramble-j2se
|
// needed to make test output available to bramble-java
|
||||||
configurations {
|
configurations {
|
||||||
testOutput.extendsFrom(testCompile)
|
testOutput.extendsFrom(testCompile)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1 @@
|
|||||||
|
*.class
|
||||||
114
bramble-core/src/main/java/net/freehaven/tor/control/Bytes.java
Normal file
114
bramble-core/src/main/java/net/freehaven/tor/control/Bytes.java
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Static class to do bytewise structure manipulation in Java.
|
||||||
|
*/
|
||||||
|
/* XXXX There must be a better way to do most of this.
|
||||||
|
* XXXX The string logic here uses default encoding, which is stupid.
|
||||||
|
*/
|
||||||
|
final class Bytes {
|
||||||
|
|
||||||
|
/** Write the two-byte value in 's' into the byte array 'ba', starting at
|
||||||
|
* the index 'pos'. */
|
||||||
|
public static void setU16(byte[] ba, int pos, short s) {
|
||||||
|
ba[pos] = (byte)((s >> 8) & 0xff);
|
||||||
|
ba[pos+1] = (byte)((s ) & 0xff);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Write the four-byte value in 'i' into the byte array 'ba', starting at
|
||||||
|
* the index 'pos'. */
|
||||||
|
public static void setU32(byte[] ba, int pos, int i) {
|
||||||
|
ba[pos] = (byte)((i >> 24) & 0xff);
|
||||||
|
ba[pos+1] = (byte)((i >> 16) & 0xff);
|
||||||
|
ba[pos+2] = (byte)((i >> 8) & 0xff);
|
||||||
|
ba[pos+3] = (byte)((i ) & 0xff);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Return the four-byte value starting at index 'pos' within 'ba' */
|
||||||
|
public static int getU32(byte[] ba, int pos) {
|
||||||
|
return
|
||||||
|
((ba[pos ]&0xff)<<24) |
|
||||||
|
((ba[pos+1]&0xff)<<16) |
|
||||||
|
((ba[pos+2]&0xff)<< 8) |
|
||||||
|
((ba[pos+3]&0xff));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String getU32S(byte[] ba, int pos) {
|
||||||
|
return String.valueOf( (getU32(ba,pos))&0xffffffffL );
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Return the two-byte value starting at index 'pos' within 'ba' */
|
||||||
|
public static int getU16(byte[] ba, int pos) {
|
||||||
|
return
|
||||||
|
((ba[pos ]&0xff)<<8) |
|
||||||
|
((ba[pos+1]&0xff));
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Return the string starting at position 'pos' of ba and extending
|
||||||
|
* until a zero byte or the end of the string. */
|
||||||
|
public static String getNulTerminatedStr(byte[] ba, int pos) {
|
||||||
|
int len, maxlen = ba.length-pos;
|
||||||
|
for (len=0; len<maxlen; ++len) {
|
||||||
|
if (ba[pos+len] == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return new String(ba, pos, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read bytes from 'ba' starting at 'pos', dividing them into strings
|
||||||
|
* along the character in 'split' and writing them into 'lst'
|
||||||
|
*/
|
||||||
|
public static void splitStr(List<String> lst, byte[] ba, int pos, byte split) {
|
||||||
|
while (pos < ba.length && ba[pos] != 0) {
|
||||||
|
int len;
|
||||||
|
for (len=0; pos+len < ba.length; ++len) {
|
||||||
|
if (ba[pos+len] == 0 || ba[pos+len] == split)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (len>0)
|
||||||
|
lst.add(new String(ba, pos, len));
|
||||||
|
pos += len;
|
||||||
|
if (ba[pos] == split)
|
||||||
|
++pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read bytes from 'ba' starting at 'pos', dividing them into strings
|
||||||
|
* along the character in 'split' and writing them into 'lst'
|
||||||
|
*/
|
||||||
|
public static List<String> splitStr(List<String> lst, String str) {
|
||||||
|
// split string on spaces, include trailing/leading
|
||||||
|
String[] tokenArray = str.split(" ", -1);
|
||||||
|
if (lst == null) {
|
||||||
|
lst = Arrays.asList( tokenArray );
|
||||||
|
} else {
|
||||||
|
lst.addAll( Arrays.asList( tokenArray ) );
|
||||||
|
}
|
||||||
|
return lst;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final char[] NYBBLES = {
|
||||||
|
'0', '1', '2', '3', '4', '5', '6', '7',
|
||||||
|
'8', '9', 'A', 'B', 'C', 'D', 'E', 'F'
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final String hex(byte[] ba) {
|
||||||
|
StringBuffer buf = new StringBuffer();
|
||||||
|
for (int i = 0; i < ba.length; ++i) {
|
||||||
|
int b = (ba[i]) & 0xff;
|
||||||
|
buf.append(NYBBLES[b >> 4]);
|
||||||
|
buf.append(NYBBLES[b&0x0f]);
|
||||||
|
}
|
||||||
|
return buf.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Bytes() {};
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
/** A single key-value pair from Tor's configuration. */
|
||||||
|
public class ConfigEntry {
|
||||||
|
public ConfigEntry(String k, String v) {
|
||||||
|
key = k;
|
||||||
|
value = v;
|
||||||
|
is_default = false;
|
||||||
|
}
|
||||||
|
public ConfigEntry(String k) {
|
||||||
|
key = k;
|
||||||
|
value = "";
|
||||||
|
is_default = true;
|
||||||
|
}
|
||||||
|
public final String key;
|
||||||
|
public final String value;
|
||||||
|
public final boolean is_default;
|
||||||
|
}
|
||||||
@@ -0,0 +1,75 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Abstract interface whose methods are invoked when Tor sends us an event.
|
||||||
|
*
|
||||||
|
* @see TorControlConnection#setEventHandler
|
||||||
|
* @see TorControlConnection#setEvents
|
||||||
|
*/
|
||||||
|
public interface EventHandler {
|
||||||
|
/**
|
||||||
|
* Invoked when a circuit's status has changed.
|
||||||
|
* Possible values for <b>status</b> are:
|
||||||
|
* <ul>
|
||||||
|
* <li>"LAUNCHED" : circuit ID assigned to new circuit</li>
|
||||||
|
* <li>"BUILT" : all hops finished, can now accept streams</li>
|
||||||
|
* <li>"EXTENDED" : one more hop has been completed</li>
|
||||||
|
* <li>"FAILED" : circuit closed (was not built)</li>
|
||||||
|
* <li>"CLOSED" : circuit closed (was built)</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* <b>circID</b> is the alphanumeric identifier of the affected circuit,
|
||||||
|
* and <b>path</b> is a comma-separated list of alphanumeric ServerIDs.
|
||||||
|
*/
|
||||||
|
public void circuitStatus(String status, String circID, String path);
|
||||||
|
/**
|
||||||
|
* Invoked when a stream's status has changed.
|
||||||
|
* Possible values for <b>status</b> are:
|
||||||
|
* <ul>
|
||||||
|
* <li>"NEW" : New request to connect</li>
|
||||||
|
* <li>"NEWRESOLVE" : New request to resolve an address</li>
|
||||||
|
* <li>"SENTCONNECT" : Sent a connect cell along a circuit</li>
|
||||||
|
* <li>"SENTRESOLVE" : Sent a resolve cell along a circuit</li>
|
||||||
|
* <li>"SUCCEEDED" : Received a reply; stream established</li>
|
||||||
|
* <li>"FAILED" : Stream failed and not retriable.</li>
|
||||||
|
* <li>"CLOSED" : Stream closed</li>
|
||||||
|
* <li>"DETACHED" : Detached from circuit; still retriable.</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* <b>streamID</b> is the alphanumeric identifier of the affected stream,
|
||||||
|
* and its <b>target</b> is specified as address:port.
|
||||||
|
*/
|
||||||
|
public void streamStatus(String status, String streamID, String target);
|
||||||
|
/**
|
||||||
|
* Invoked when the status of a connection to an OR has changed.
|
||||||
|
* Possible values for <b>status</b> are ["LAUNCHED" | "CONNECTED" | "FAILED" | "CLOSED"].
|
||||||
|
* <b>orName</b> is the alphanumeric identifier of the OR affected.
|
||||||
|
*/
|
||||||
|
public void orConnStatus(String status, String orName);
|
||||||
|
/**
|
||||||
|
* Invoked once per second. <b>read</b> and <b>written</b> are
|
||||||
|
* the number of bytes read and written, respectively, in
|
||||||
|
* the last second.
|
||||||
|
*/
|
||||||
|
public void bandwidthUsed(long read, long written);
|
||||||
|
/**
|
||||||
|
* Invoked whenever Tor learns about new ORs. The <b>orList</b> object
|
||||||
|
* contains the alphanumeric ServerIDs associated with the new ORs.
|
||||||
|
*/
|
||||||
|
public void newDescriptors(java.util.List<String> orList);
|
||||||
|
/**
|
||||||
|
* Invoked when Tor logs a message.
|
||||||
|
* <b>severity</b> is one of ["DEBUG" | "INFO" | "NOTICE" | "WARN" | "ERR"],
|
||||||
|
* and <b>msg</b> is the message string.
|
||||||
|
*/
|
||||||
|
public void message(String severity, String msg);
|
||||||
|
/**
|
||||||
|
* Invoked when an unspecified message is received.
|
||||||
|
* <type> is the message type, and <msg> is the message string.
|
||||||
|
*/
|
||||||
|
public void unrecognized(String type, String msg);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Implementation of EventHandler that ignores all events. Useful
|
||||||
|
* when you only want to override one method.
|
||||||
|
*/
|
||||||
|
public class NullEventHandler implements EventHandler {
|
||||||
|
public void circuitStatus(String status, String circID, String path) {}
|
||||||
|
public void streamStatus(String status, String streamID, String target) {}
|
||||||
|
public void orConnStatus(String status, String orName) {}
|
||||||
|
public void bandwidthUsed(long read, long written) {}
|
||||||
|
public void newDescriptors(java.util.List<String> orList) {}
|
||||||
|
public void message(String severity, String msg) {}
|
||||||
|
public void unrecognized(String type, String msg) {}
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
import java.security.MessageDigest;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
|
import java.security.SecureRandom;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A hashed digest of a secret password (used to set control connection
|
||||||
|
* security.)
|
||||||
|
*
|
||||||
|
* For the actual hashing algorithm, see RFC2440's secret-to-key conversion.
|
||||||
|
*/
|
||||||
|
public class PasswordDigest {
|
||||||
|
|
||||||
|
private final byte[] secret;
|
||||||
|
private final String hashedKey;
|
||||||
|
|
||||||
|
/** Return a new password digest with a random secret and salt. */
|
||||||
|
public static PasswordDigest generateDigest() {
|
||||||
|
byte[] secret = new byte[20];
|
||||||
|
SecureRandom rng = new SecureRandom();
|
||||||
|
rng.nextBytes(secret);
|
||||||
|
return new PasswordDigest(secret);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Construct a new password digest with a given secret and random salt */
|
||||||
|
public PasswordDigest(byte[] secret) {
|
||||||
|
this(secret, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Construct a new password digest with a given secret and random salt.
|
||||||
|
* Note that the 9th byte of the specifier determines the number of hash
|
||||||
|
* iterations as in RFC2440.
|
||||||
|
*/
|
||||||
|
public PasswordDigest(byte[] secret, byte[] specifier) {
|
||||||
|
this.secret = secret.clone();
|
||||||
|
if (specifier == null) {
|
||||||
|
specifier = new byte[9];
|
||||||
|
SecureRandom rng = new SecureRandom();
|
||||||
|
rng.nextBytes(specifier);
|
||||||
|
specifier[8] = 96;
|
||||||
|
}
|
||||||
|
hashedKey = "16:"+encodeBytes(secretToKey(secret, specifier));
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Return the secret used to generate this password hash.
|
||||||
|
*/
|
||||||
|
public byte[] getSecret() {
|
||||||
|
return secret.clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Return the hashed password in the format used by Tor. */
|
||||||
|
public String getHashedPassword() {
|
||||||
|
return hashedKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Parameter used by RFC2440's s2k algorithm. */
|
||||||
|
private static final int EXPBIAS = 6;
|
||||||
|
|
||||||
|
/** Implement rfc2440 s2k */
|
||||||
|
public static byte[] secretToKey(byte[] secret, byte[] specifier) {
|
||||||
|
MessageDigest d;
|
||||||
|
try {
|
||||||
|
d = MessageDigest.getInstance("SHA-1");
|
||||||
|
} catch (NoSuchAlgorithmException ex) {
|
||||||
|
throw new RuntimeException("Can't run without sha-1.");
|
||||||
|
}
|
||||||
|
int c = (specifier[8])&0xff;
|
||||||
|
int count = (16 + (c&15)) << ((c>>4) + EXPBIAS);
|
||||||
|
|
||||||
|
byte[] tmp = new byte[8+secret.length];
|
||||||
|
System.arraycopy(specifier, 0, tmp, 0, 8);
|
||||||
|
System.arraycopy(secret, 0, tmp, 8, secret.length);
|
||||||
|
while (count > 0) {
|
||||||
|
if (count >= tmp.length) {
|
||||||
|
d.update(tmp);
|
||||||
|
count -= tmp.length;
|
||||||
|
} else {
|
||||||
|
d.update(tmp, 0, count);
|
||||||
|
count = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
byte[] key = new byte[20+9];
|
||||||
|
System.arraycopy(d.digest(), 0, key, 9, 20);
|
||||||
|
System.arraycopy(specifier, 0, key, 0, 9);
|
||||||
|
return key;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Return a hexadecimal encoding of a byte array. */
|
||||||
|
// XXX There must be a better way to do this in Java.
|
||||||
|
private static final String encodeBytes(byte[] ba) {
|
||||||
|
return Bytes.hex(ba);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
We broke the version detection stuff in Tor 0.1.2.16 / 0.2.0.4-alpha.
|
||||||
|
Somebody should rip out the v0 control protocol stuff from here, and
|
||||||
|
it should start working again. -RD
|
||||||
|
|
||||||
@@ -0,0 +1,151 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
/** Interface defining constants used by the Tor controller protocol.
|
||||||
|
*/
|
||||||
|
// XXXX Take documentation for these from control-spec.txt
|
||||||
|
public interface TorControlCommands {
|
||||||
|
|
||||||
|
public static final short CMD_ERROR = 0x0000;
|
||||||
|
public static final short CMD_DONE = 0x0001;
|
||||||
|
public static final short CMD_SETCONF = 0x0002;
|
||||||
|
public static final short CMD_GETCONF = 0x0003;
|
||||||
|
public static final short CMD_CONFVALUE = 0x0004;
|
||||||
|
public static final short CMD_SETEVENTS = 0x0005;
|
||||||
|
public static final short CMD_EVENT = 0x0006;
|
||||||
|
public static final short CMD_AUTH = 0x0007;
|
||||||
|
public static final short CMD_SAVECONF = 0x0008;
|
||||||
|
public static final short CMD_SIGNAL = 0x0009;
|
||||||
|
public static final short CMD_MAPADDRESS = 0x000A;
|
||||||
|
public static final short CMD_GETINFO = 0x000B;
|
||||||
|
public static final short CMD_INFOVALUE = 0x000C;
|
||||||
|
public static final short CMD_EXTENDCIRCUIT = 0x000D;
|
||||||
|
public static final short CMD_ATTACHSTREAM = 0x000E;
|
||||||
|
public static final short CMD_POSTDESCRIPTOR = 0x000F;
|
||||||
|
public static final short CMD_FRAGMENTHEADER = 0x0010;
|
||||||
|
public static final short CMD_FRAGMENT = 0x0011;
|
||||||
|
public static final short CMD_REDIRECTSTREAM = 0x0012;
|
||||||
|
public static final short CMD_CLOSESTREAM = 0x0013;
|
||||||
|
public static final short CMD_CLOSECIRCUIT = 0x0014;
|
||||||
|
|
||||||
|
public static final String[] CMD_NAMES = {
|
||||||
|
"ERROR",
|
||||||
|
"DONE",
|
||||||
|
"SETCONF",
|
||||||
|
"GETCONF",
|
||||||
|
"CONFVALUE",
|
||||||
|
"SETEVENTS",
|
||||||
|
"EVENT",
|
||||||
|
"AUTH",
|
||||||
|
"SAVECONF",
|
||||||
|
"SIGNAL",
|
||||||
|
"MAPADDRESS",
|
||||||
|
"GETINFO",
|
||||||
|
"INFOVALUE",
|
||||||
|
"EXTENDCIRCUIT",
|
||||||
|
"ATTACHSTREAM",
|
||||||
|
"POSTDESCRIPTOR",
|
||||||
|
"FRAGMENTHEADER",
|
||||||
|
"FRAGMENT",
|
||||||
|
"REDIRECTSTREAM",
|
||||||
|
"CLOSESTREAM",
|
||||||
|
"CLOSECIRCUIT",
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final short EVENT_CIRCSTATUS = 0x0001;
|
||||||
|
public static final short EVENT_STREAMSTATUS = 0x0002;
|
||||||
|
public static final short EVENT_ORCONNSTATUS = 0x0003;
|
||||||
|
public static final short EVENT_BANDWIDTH = 0x0004;
|
||||||
|
public static final short EVENT_NEWDESCRIPTOR = 0x0006;
|
||||||
|
public static final short EVENT_MSG_DEBUG = 0x0007;
|
||||||
|
public static final short EVENT_MSG_INFO = 0x0008;
|
||||||
|
public static final short EVENT_MSG_NOTICE = 0x0009;
|
||||||
|
public static final short EVENT_MSG_WARN = 0x000A;
|
||||||
|
public static final short EVENT_MSG_ERROR = 0x000B;
|
||||||
|
|
||||||
|
public static final String[] EVENT_NAMES = {
|
||||||
|
"(0)",
|
||||||
|
"CIRC",
|
||||||
|
"STREAM",
|
||||||
|
"ORCONN",
|
||||||
|
"BW",
|
||||||
|
"OLDLOG",
|
||||||
|
"NEWDESC",
|
||||||
|
"DEBUG",
|
||||||
|
"INFO",
|
||||||
|
"NOTICE",
|
||||||
|
"WARN",
|
||||||
|
"ERR",
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final byte CIRC_STATUS_LAUNCHED = 0x01;
|
||||||
|
public static final byte CIRC_STATUS_BUILT = 0x02;
|
||||||
|
public static final byte CIRC_STATUS_EXTENDED = 0x03;
|
||||||
|
public static final byte CIRC_STATUS_FAILED = 0x04;
|
||||||
|
public static final byte CIRC_STATUS_CLOSED = 0x05;
|
||||||
|
|
||||||
|
public static final String[] CIRC_STATUS_NAMES = {
|
||||||
|
"LAUNCHED",
|
||||||
|
"BUILT",
|
||||||
|
"EXTENDED",
|
||||||
|
"FAILED",
|
||||||
|
"CLOSED",
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final byte STREAM_STATUS_SENT_CONNECT = 0x00;
|
||||||
|
public static final byte STREAM_STATUS_SENT_RESOLVE = 0x01;
|
||||||
|
public static final byte STREAM_STATUS_SUCCEEDED = 0x02;
|
||||||
|
public static final byte STREAM_STATUS_FAILED = 0x03;
|
||||||
|
public static final byte STREAM_STATUS_CLOSED = 0x04;
|
||||||
|
public static final byte STREAM_STATUS_NEW_CONNECT = 0x05;
|
||||||
|
public static final byte STREAM_STATUS_NEW_RESOLVE = 0x06;
|
||||||
|
public static final byte STREAM_STATUS_DETACHED = 0x07;
|
||||||
|
|
||||||
|
public static final String[] STREAM_STATUS_NAMES = {
|
||||||
|
"SENT_CONNECT",
|
||||||
|
"SENT_RESOLVE",
|
||||||
|
"SUCCEEDED",
|
||||||
|
"FAILED",
|
||||||
|
"CLOSED",
|
||||||
|
"NEW_CONNECT",
|
||||||
|
"NEW_RESOLVE",
|
||||||
|
"DETACHED"
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final byte OR_CONN_STATUS_LAUNCHED = 0x00;
|
||||||
|
public static final byte OR_CONN_STATUS_CONNECTED = 0x01;
|
||||||
|
public static final byte OR_CONN_STATUS_FAILED = 0x02;
|
||||||
|
public static final byte OR_CONN_STATUS_CLOSED = 0x03;
|
||||||
|
|
||||||
|
public static final String[] OR_CONN_STATUS_NAMES = {
|
||||||
|
"LAUNCHED","CONNECTED","FAILED","CLOSED"
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final byte SIGNAL_HUP = 0x01;
|
||||||
|
public static final byte SIGNAL_INT = 0x02;
|
||||||
|
public static final byte SIGNAL_USR1 = 0x0A;
|
||||||
|
public static final byte SIGNAL_USR2 = 0x0C;
|
||||||
|
public static final byte SIGNAL_TERM = 0x0F;
|
||||||
|
|
||||||
|
public static final String ERROR_MSGS[] = {
|
||||||
|
"Unspecified error",
|
||||||
|
"Internal error",
|
||||||
|
"Unrecognized message type",
|
||||||
|
"Syntax error",
|
||||||
|
"Unrecognized configuration key",
|
||||||
|
"Invalid configuration value",
|
||||||
|
"Unrecognized byte code",
|
||||||
|
"Unauthorized",
|
||||||
|
"Failed authentication attempt",
|
||||||
|
"Resource exhausted",
|
||||||
|
"No such stream",
|
||||||
|
"No such circuit",
|
||||||
|
"No such OR",
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final String HS_ADDRESS = "onionAddress";
|
||||||
|
public static final String HS_PRIVKEY = "onionPrivKey";
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,998 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.io.OutputStreamWriter;
|
||||||
|
import java.io.PrintStream;
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.io.Writer;
|
||||||
|
import java.net.Socket;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.StringTokenizer;
|
||||||
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
|
import static java.util.logging.Logger.getLogger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A connection to a running Tor process as specified in control-spec.txt.
|
||||||
|
*/
|
||||||
|
public class TorControlConnection implements TorControlCommands {
|
||||||
|
|
||||||
|
private static final Logger LOG =
|
||||||
|
getLogger(TorControlConnection.class.getName());
|
||||||
|
|
||||||
|
private final LinkedList<Waiter> waiters;
|
||||||
|
private final BufferedReader input;
|
||||||
|
private final Writer output;
|
||||||
|
|
||||||
|
private ControlParseThread thread; // Locking: this
|
||||||
|
|
||||||
|
private volatile EventHandler handler;
|
||||||
|
private volatile PrintWriter debugOutput;
|
||||||
|
private volatile IOException parseThreadException;
|
||||||
|
|
||||||
|
static class Waiter {
|
||||||
|
|
||||||
|
List<ReplyLine> response; // Locking: this
|
||||||
|
boolean interrupted;
|
||||||
|
|
||||||
|
List<ReplyLine> getResponse() throws InterruptedException {
|
||||||
|
LOG.info("Entering synchronized (waiter " + hashCode() + ")");
|
||||||
|
synchronized (this) {
|
||||||
|
LOG.info("Entered synchronized (waiter " + hashCode() + ")");
|
||||||
|
while (response == null) {
|
||||||
|
LOG.info("Waiter " + hashCode() + " waiting for response");
|
||||||
|
wait();
|
||||||
|
if (interrupted) {
|
||||||
|
LOG.info("Waiter " + hashCode() + " interrupted");
|
||||||
|
throw new InterruptedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
LOG.info("Waiter " + hashCode() + " got response " + response);
|
||||||
|
LOG.info("Leaving synchronized (waiter " + hashCode() + ")");
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void setResponse(List<ReplyLine> response) {
|
||||||
|
LOG.info("Entering synchronized (waiter " + hashCode() + ")");
|
||||||
|
synchronized (this) {
|
||||||
|
LOG.info("Entered synchronized (waiter " + hashCode() + ")");
|
||||||
|
LOG.info("Setting response for waiter " + hashCode() + ": "
|
||||||
|
+ response);
|
||||||
|
this.response = response;
|
||||||
|
notifyAll();
|
||||||
|
LOG.info("Leaving synchronized (waiter " + hashCode() + ")");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void interrupt() {
|
||||||
|
LOG.info("Entering synchronized (waiter " + hashCode() + ")");
|
||||||
|
synchronized (this) {
|
||||||
|
LOG.info("Entered synchronized (waiter " + hashCode() + ")");
|
||||||
|
LOG.info("Interrupting waiter " + hashCode());
|
||||||
|
interrupted = true;
|
||||||
|
notifyAll();
|
||||||
|
LOG.info("Leaving synchronized (waiter " + hashCode() + ")");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static class ReplyLine {
|
||||||
|
|
||||||
|
final String status;
|
||||||
|
final String msg;
|
||||||
|
final String rest;
|
||||||
|
|
||||||
|
ReplyLine(String status, String msg, String rest) {
|
||||||
|
this.status = status;
|
||||||
|
this.msg = msg;
|
||||||
|
this.rest = rest;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return status + " " + msg + " " + rest;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new TorControlConnection to communicate with Tor over
|
||||||
|
* a given socket. After calling this constructor, it is typical to
|
||||||
|
* call launchThread and authenticate.
|
||||||
|
*/
|
||||||
|
public TorControlConnection(Socket connection) throws IOException {
|
||||||
|
this(connection.getInputStream(), connection.getOutputStream());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new TorControlConnection to communicate with Tor over
|
||||||
|
* an arbitrary pair of data streams.
|
||||||
|
*/
|
||||||
|
public TorControlConnection(InputStream i, OutputStream o) {
|
||||||
|
this(new InputStreamReader(i), new OutputStreamWriter(o));
|
||||||
|
}
|
||||||
|
|
||||||
|
public TorControlConnection(Reader i, Writer o) {
|
||||||
|
this.output = o;
|
||||||
|
if (i instanceof BufferedReader)
|
||||||
|
this.input = (BufferedReader) i;
|
||||||
|
else
|
||||||
|
this.input = new BufferedReader(i);
|
||||||
|
this.waiters = new LinkedList<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected final void writeEscaped(String s) throws IOException {
|
||||||
|
StringTokenizer st = new StringTokenizer(s, "\n");
|
||||||
|
while (st.hasMoreTokens()) {
|
||||||
|
String line = st.nextToken();
|
||||||
|
if (line.startsWith("."))
|
||||||
|
line = "." + line;
|
||||||
|
if (line.endsWith("\r"))
|
||||||
|
line += "\n";
|
||||||
|
else
|
||||||
|
line += "\r\n";
|
||||||
|
if (debugOutput != null)
|
||||||
|
debugOutput.print(">> " + line);
|
||||||
|
output.write(line);
|
||||||
|
}
|
||||||
|
output.write(".\r\n");
|
||||||
|
if (debugOutput != null)
|
||||||
|
debugOutput.print(">> .\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static String quote(String s) {
|
||||||
|
StringBuffer sb = new StringBuffer("\"");
|
||||||
|
for (int i = 0; i < s.length(); ++i) {
|
||||||
|
char c = s.charAt(i);
|
||||||
|
switch (c) {
|
||||||
|
case '\r':
|
||||||
|
case '\n':
|
||||||
|
case '\\':
|
||||||
|
case '\"':
|
||||||
|
sb.append('\\');
|
||||||
|
}
|
||||||
|
sb.append(c);
|
||||||
|
}
|
||||||
|
sb.append('\"');
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected final ArrayList<ReplyLine> readReply() throws IOException {
|
||||||
|
ArrayList<ReplyLine> reply = new ArrayList<>();
|
||||||
|
char c;
|
||||||
|
do {
|
||||||
|
String line = input.readLine();
|
||||||
|
if (line == null) {
|
||||||
|
// if line is null, the end of the stream has been reached, i.e.
|
||||||
|
// the connection to Tor has been closed!
|
||||||
|
if (reply.isEmpty()) {
|
||||||
|
// nothing received so far, can exit cleanly
|
||||||
|
return reply;
|
||||||
|
}
|
||||||
|
// received half of a reply before the connection broke down
|
||||||
|
throw new TorControlSyntaxError("Connection to Tor " +
|
||||||
|
" broke down while receiving reply!");
|
||||||
|
}
|
||||||
|
if (debugOutput != null)
|
||||||
|
debugOutput.println("<< " + line);
|
||||||
|
if (line.length() < 4)
|
||||||
|
throw new TorControlSyntaxError(
|
||||||
|
"Line (\"" + line + "\") too short");
|
||||||
|
String status = line.substring(0, 3);
|
||||||
|
c = line.charAt(3);
|
||||||
|
String msg = line.substring(4);
|
||||||
|
String rest = null;
|
||||||
|
if (c == '+') {
|
||||||
|
StringBuffer data = new StringBuffer();
|
||||||
|
while (true) {
|
||||||
|
line = input.readLine();
|
||||||
|
if (debugOutput != null)
|
||||||
|
debugOutput.print("<< " + line);
|
||||||
|
if (line.equals("."))
|
||||||
|
break;
|
||||||
|
else if (line.startsWith("."))
|
||||||
|
line = line.substring(1);
|
||||||
|
data.append(line).append('\n');
|
||||||
|
}
|
||||||
|
rest = data.toString();
|
||||||
|
}
|
||||||
|
reply.add(new ReplyLine(status, msg, rest));
|
||||||
|
} while (c != ' ');
|
||||||
|
|
||||||
|
return reply;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected List<ReplyLine> sendAndWaitForResponse(String s,
|
||||||
|
String rest) throws IOException {
|
||||||
|
LOG.info("Entering synchronized (connection)");
|
||||||
|
synchronized (this) {
|
||||||
|
LOG.info("Entered synchronized (connection)");
|
||||||
|
LOG.info("Sending '" + s + "', '" + rest +
|
||||||
|
"' and waiting for response");
|
||||||
|
if (parseThreadException != null) {
|
||||||
|
LOG.info("Throwing previously caught exception "
|
||||||
|
+ parseThreadException);
|
||||||
|
throw parseThreadException;
|
||||||
|
}
|
||||||
|
checkThread();
|
||||||
|
Waiter w = new Waiter();
|
||||||
|
LOG.info("Created waiter " + w.hashCode());
|
||||||
|
if (debugOutput != null)
|
||||||
|
debugOutput.print(">> " + s);
|
||||||
|
LOG.info("Entering synchronized (waiters)");
|
||||||
|
synchronized (waiters) {
|
||||||
|
LOG.info("Entered synchronized (waiters)");
|
||||||
|
output.write(s);
|
||||||
|
LOG.info("Wrote '" + s + "'");
|
||||||
|
if (rest != null) {
|
||||||
|
writeEscaped(rest);
|
||||||
|
LOG.info("Wrote escaped '" + rest + "'");
|
||||||
|
}
|
||||||
|
output.flush();
|
||||||
|
LOG.info("Flushed output");
|
||||||
|
waiters.addLast(w);
|
||||||
|
LOG.info("Added waiter, " + waiters.size() + " waiting");
|
||||||
|
LOG.info("Leaving synchronized (waiters)");
|
||||||
|
}
|
||||||
|
List<ReplyLine> lst;
|
||||||
|
try {
|
||||||
|
LOG.info("Getting response from waiter " + w.hashCode());
|
||||||
|
lst = w.getResponse();
|
||||||
|
LOG.info("Got response from waiter " + w.hashCode() + ": " +
|
||||||
|
lst);
|
||||||
|
} catch (InterruptedException ex) {
|
||||||
|
throw new IOException("Interrupted");
|
||||||
|
}
|
||||||
|
for (Iterator<ReplyLine> i = lst.iterator(); i.hasNext(); ) {
|
||||||
|
ReplyLine c = i.next();
|
||||||
|
if (!c.status.startsWith("2"))
|
||||||
|
throw new TorControlError("Error reply: " + c.msg);
|
||||||
|
}
|
||||||
|
LOG.info("Leaving synchronized (connection)");
|
||||||
|
return lst;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper: decode a CMD_EVENT command and dispatch it to our
|
||||||
|
* EventHandler (if any).
|
||||||
|
*/
|
||||||
|
protected void handleEvent(ArrayList<ReplyLine> events) {
|
||||||
|
if (handler == null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (Iterator<ReplyLine> i = events.iterator(); i.hasNext(); ) {
|
||||||
|
ReplyLine line = i.next();
|
||||||
|
int idx = line.msg.indexOf(' ');
|
||||||
|
String tp = line.msg.substring(0, idx).toUpperCase();
|
||||||
|
String rest = line.msg.substring(idx + 1);
|
||||||
|
if (tp.equals("CIRC")) {
|
||||||
|
List<String> lst = Bytes.splitStr(null, rest);
|
||||||
|
handler.circuitStatus(lst.get(1),
|
||||||
|
lst.get(0),
|
||||||
|
lst.get(1).equals("LAUNCHED")
|
||||||
|
|| lst.size() < 3 ? ""
|
||||||
|
: lst.get(2));
|
||||||
|
} else if (tp.equals("STREAM")) {
|
||||||
|
List<String> lst = Bytes.splitStr(null, rest);
|
||||||
|
handler.streamStatus(lst.get(1),
|
||||||
|
lst.get(0),
|
||||||
|
lst.get(3));
|
||||||
|
// XXXX circID.
|
||||||
|
} else if (tp.equals("ORCONN")) {
|
||||||
|
List<String> lst = Bytes.splitStr(null, rest);
|
||||||
|
handler.orConnStatus(lst.get(1), lst.get(0));
|
||||||
|
} else if (tp.equals("BW")) {
|
||||||
|
List<String> lst = Bytes.splitStr(null, rest);
|
||||||
|
handler.bandwidthUsed(Integer.parseInt(lst.get(0)),
|
||||||
|
Integer.parseInt(lst.get(1)));
|
||||||
|
} else if (tp.equals("NEWDESC")) {
|
||||||
|
List<String> lst = Bytes.splitStr(null, rest);
|
||||||
|
handler.newDescriptors(lst);
|
||||||
|
} else if (tp.equals("DEBUG") ||
|
||||||
|
tp.equals("INFO") ||
|
||||||
|
tp.equals("NOTICE") ||
|
||||||
|
tp.equals("WARN") ||
|
||||||
|
tp.equals("ERR")) {
|
||||||
|
handler.message(tp, rest);
|
||||||
|
} else {
|
||||||
|
handler.unrecognized(tp, rest);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets <b>w</b> as the PrintWriter for debugging output,
|
||||||
|
* which writes out all messages passed between Tor and the controller.
|
||||||
|
* Outgoing messages are preceded by "\>\>" and incoming messages are preceded
|
||||||
|
* by "\<\<"
|
||||||
|
*/
|
||||||
|
public void setDebugging(PrintWriter w) {
|
||||||
|
debugOutput = w;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets <b>s</b> as the PrintStream for debugging output,
|
||||||
|
* which writes out all messages passed between Tor and the controller.
|
||||||
|
* Outgoing messages are preceded by "\>\>" and incoming messages are preceded
|
||||||
|
* by "\<\<"
|
||||||
|
*/
|
||||||
|
public void setDebugging(PrintStream s) {
|
||||||
|
debugOutput = new PrintWriter(s, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the EventHandler object that will be notified of any
|
||||||
|
* events Tor delivers to this connection. To make Tor send us
|
||||||
|
* events, call setEvents().
|
||||||
|
*/
|
||||||
|
public void setEventHandler(EventHandler handler) {
|
||||||
|
this.handler = handler;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start a thread to react to Tor's responses in the background.
|
||||||
|
* This is necessary to handle asynchronous events and synchronous
|
||||||
|
* responses that arrive independantly over the same socket.
|
||||||
|
*/
|
||||||
|
public Thread launchThread(boolean daemon) {
|
||||||
|
LOG.info("Entering synchronized (connection)");
|
||||||
|
synchronized (this) {
|
||||||
|
LOG.info("Entered synchronized (connection)");
|
||||||
|
ControlParseThread th = new ControlParseThread();
|
||||||
|
LOG.info("Launching parse thread " + th.hashCode());
|
||||||
|
if (daemon)
|
||||||
|
th.setDaemon(true);
|
||||||
|
th.start();
|
||||||
|
this.thread = th;
|
||||||
|
LOG.info("Leaving synchronized (connection)");
|
||||||
|
return th;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected class ControlParseThread extends Thread {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
react();
|
||||||
|
} catch (IOException ex) {
|
||||||
|
LOG.info("Parse thread " + hashCode()
|
||||||
|
+ " caught exception " + ex);
|
||||||
|
parseThreadException = ex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void checkThread() {
|
||||||
|
LOG.info("Entering synchronized (connection)");
|
||||||
|
synchronized (this) {
|
||||||
|
LOG.info("Entered synchronized (connection)");
|
||||||
|
if (thread == null)
|
||||||
|
launchThread(true);
|
||||||
|
LOG.info("Leaving synchronized (connection)");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* helper: implement the main background loop.
|
||||||
|
*/
|
||||||
|
protected void react() throws IOException {
|
||||||
|
while (true) {
|
||||||
|
ArrayList<ReplyLine> lst = readReply();
|
||||||
|
LOG.info("Read reply: " + lst);
|
||||||
|
if (lst.isEmpty()) {
|
||||||
|
// interrupted queued waiters, there won't be any response.
|
||||||
|
LOG.info("Entering synchronized (waiters)");
|
||||||
|
synchronized (waiters) {
|
||||||
|
LOG.info("Entered synchronized (waiters)");
|
||||||
|
if (!waiters.isEmpty()) {
|
||||||
|
for (Waiter w : waiters) {
|
||||||
|
LOG.info("Interrupting waiter " + w.hashCode());
|
||||||
|
w.interrupt();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
LOG.info("No waiters");
|
||||||
|
}
|
||||||
|
LOG.info("Leaving synchronized (waiters)");
|
||||||
|
}
|
||||||
|
throw new IOException("Tor is no longer running");
|
||||||
|
}
|
||||||
|
if ((lst.get(0)).status.startsWith("6")) {
|
||||||
|
LOG.info("Reply is an event");
|
||||||
|
handleEvent(lst);
|
||||||
|
} else {
|
||||||
|
LOG.info("Entering synchronized (waiters)");
|
||||||
|
synchronized (waiters) {
|
||||||
|
LOG.info("Entered synchronized (waiters)");
|
||||||
|
if (!waiters.isEmpty()) {
|
||||||
|
Waiter w;
|
||||||
|
w = waiters.removeFirst();
|
||||||
|
LOG.info("Setting response for waiter " + w.hashCode());
|
||||||
|
w.setResponse(lst);
|
||||||
|
} else {
|
||||||
|
LOG.info("No waiters");
|
||||||
|
}
|
||||||
|
LOG.info("Leaving synchronized (waiters)");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Change the value of the configuration option 'key' to 'val'.
|
||||||
|
*/
|
||||||
|
public void setConf(String key, String value) throws IOException {
|
||||||
|
List<String> lst = new ArrayList<>();
|
||||||
|
lst.add(key + " " + value);
|
||||||
|
setConf(lst);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Change the values of the configuration options stored in kvMap.
|
||||||
|
*/
|
||||||
|
public void setConf(Map<String, String> kvMap) throws IOException {
|
||||||
|
List<String> lst = new ArrayList<>();
|
||||||
|
for (Iterator<Map.Entry<String, String>> it =
|
||||||
|
kvMap.entrySet().iterator(); it.hasNext(); ) {
|
||||||
|
Map.Entry<String, String> ent = it.next();
|
||||||
|
lst.add(ent.getKey() + " " + ent.getValue() + "\n");
|
||||||
|
}
|
||||||
|
setConf(lst);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Changes the values of the configuration options stored in
|
||||||
|
* <b>kvList</b>. Each list element in <b>kvList</b> is expected to be
|
||||||
|
* String of the format "key value".
|
||||||
|
* <p>
|
||||||
|
* Tor behaves as though it had just read each of the key-value pairs
|
||||||
|
* from its configuration file. Keywords with no corresponding values have
|
||||||
|
* their configuration values reset to their defaults. setConf is
|
||||||
|
* all-or-nothing: if there is an error in any of the configuration settings,
|
||||||
|
* Tor sets none of them.
|
||||||
|
* <p>
|
||||||
|
* When a configuration option takes multiple values, or when multiple
|
||||||
|
* configuration keys form a context-sensitive group (see getConf below), then
|
||||||
|
* setting any of the options in a setConf command is taken to reset all of
|
||||||
|
* the others. For example, if two ORBindAddress values are configured, and a
|
||||||
|
* command arrives containing a single ORBindAddress value, the new
|
||||||
|
* command's value replaces the two old values.
|
||||||
|
* <p>
|
||||||
|
* To remove all settings for a given option entirely (and go back to its
|
||||||
|
* default value), include a String in <b>kvList</b> containing the key and no value.
|
||||||
|
*/
|
||||||
|
public void setConf(Collection<String> kvList) throws IOException {
|
||||||
|
if (kvList.size() == 0)
|
||||||
|
return;
|
||||||
|
StringBuffer b = new StringBuffer("SETCONF");
|
||||||
|
for (Iterator<String> it = kvList.iterator(); it.hasNext(); ) {
|
||||||
|
String kv = it.next();
|
||||||
|
int i = kv.indexOf(' ');
|
||||||
|
if (i == -1)
|
||||||
|
b.append(" ").append(kv);
|
||||||
|
b.append(" ").append(kv.substring(0, i)).append("=")
|
||||||
|
.append(quote(kv.substring(i + 1)));
|
||||||
|
}
|
||||||
|
b.append("\r\n");
|
||||||
|
sendAndWaitForResponse(b.toString(), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Try to reset the values listed in the collection 'keys' to their
|
||||||
|
* default values.
|
||||||
|
**/
|
||||||
|
public void resetConf(Collection<String> keys) throws IOException {
|
||||||
|
if (keys.size() == 0)
|
||||||
|
return;
|
||||||
|
StringBuffer b = new StringBuffer("RESETCONF");
|
||||||
|
for (Iterator<String> it = keys.iterator(); it.hasNext(); ) {
|
||||||
|
String key = it.next();
|
||||||
|
b.append(" ").append(key);
|
||||||
|
}
|
||||||
|
b.append("\r\n");
|
||||||
|
sendAndWaitForResponse(b.toString(), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the value of the configuration option 'key'
|
||||||
|
*/
|
||||||
|
public List<ConfigEntry> getConf(String key) throws IOException {
|
||||||
|
List<String> lst = new ArrayList<>();
|
||||||
|
lst.add(key);
|
||||||
|
return getConf(lst);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Requests the values of the configuration variables listed in <b>keys</b>.
|
||||||
|
* Results are returned as a list of ConfigEntry objects.
|
||||||
|
* <p>
|
||||||
|
* If an option appears multiple times in the configuration, all of its
|
||||||
|
* key-value pairs are returned in order.
|
||||||
|
* <p>
|
||||||
|
* Some options are context-sensitive, and depend on other options with
|
||||||
|
* different keywords. These cannot be fetched directly. Currently there
|
||||||
|
* is only one such option: clients should use the "HiddenServiceOptions"
|
||||||
|
* virtual keyword to get all HiddenServiceDir, HiddenServicePort,
|
||||||
|
* HiddenServiceNodes, and HiddenServiceExcludeNodes option settings.
|
||||||
|
*/
|
||||||
|
public List<ConfigEntry> getConf(Collection<String> keys)
|
||||||
|
throws IOException {
|
||||||
|
StringBuffer sb = new StringBuffer("GETCONF");
|
||||||
|
for (Iterator<String> it = keys.iterator(); it.hasNext(); ) {
|
||||||
|
String key = it.next();
|
||||||
|
sb.append(" ").append(key);
|
||||||
|
}
|
||||||
|
sb.append("\r\n");
|
||||||
|
List<ReplyLine> lst = sendAndWaitForResponse(sb.toString(), null);
|
||||||
|
List<ConfigEntry> result = new ArrayList<>();
|
||||||
|
for (Iterator<ReplyLine> it = lst.iterator(); it.hasNext(); ) {
|
||||||
|
String kv = (it.next()).msg;
|
||||||
|
int idx = kv.indexOf('=');
|
||||||
|
if (idx >= 0)
|
||||||
|
result.add(new ConfigEntry(kv.substring(0, idx),
|
||||||
|
kv.substring(idx + 1)));
|
||||||
|
else
|
||||||
|
result.add(new ConfigEntry(kv));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Request that the server inform the client about interesting events.
|
||||||
|
* Each element of <b>events</b> is one of the following Strings:
|
||||||
|
* ["CIRC" | "STREAM" | "ORCONN" | "BW" | "DEBUG" |
|
||||||
|
* "INFO" | "NOTICE" | "WARN" | "ERR" | "NEWDESC" | "ADDRMAP"] .
|
||||||
|
* <p>
|
||||||
|
* Any events not listed in the <b>events</b> are turned off; thus, calling
|
||||||
|
* setEvents with an empty <b>events</b> argument turns off all event reporting.
|
||||||
|
*/
|
||||||
|
public void setEvents(List<String> events) throws IOException {
|
||||||
|
StringBuffer sb = new StringBuffer("SETEVENTS");
|
||||||
|
for (Iterator<String> it = events.iterator(); it.hasNext(); ) {
|
||||||
|
sb.append(" ").append(it.next());
|
||||||
|
}
|
||||||
|
sb.append("\r\n");
|
||||||
|
sendAndWaitForResponse(sb.toString(), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Authenticates the controller to the Tor server.
|
||||||
|
* <p>
|
||||||
|
* By default, the current Tor implementation trusts all local users, and
|
||||||
|
* the controller can authenticate itself by calling authenticate(new byte[0]).
|
||||||
|
* <p>
|
||||||
|
* If the 'CookieAuthentication' option is true, Tor writes a "magic cookie"
|
||||||
|
* file named "control_auth_cookie" into its data directory. To authenticate,
|
||||||
|
* the controller must send the contents of this file in <b>auth</b>.
|
||||||
|
* <p>
|
||||||
|
* If the 'HashedControlPassword' option is set, <b>auth</b> must contain the salted
|
||||||
|
* hash of a secret password. The salted hash is computed according to the
|
||||||
|
* S2K algorithm in RFC 2440 (OpenPGP), and prefixed with the s2k specifier.
|
||||||
|
* This is then encoded in hexadecimal, prefixed by the indicator sequence
|
||||||
|
* "16:".
|
||||||
|
* <p>
|
||||||
|
* You can generate the salt of a password by calling
|
||||||
|
* 'tor --hash-password <password>'
|
||||||
|
* or by using the provided PasswordDigest class.
|
||||||
|
* To authenticate under this scheme, the controller sends Tor the original
|
||||||
|
* secret that was used to generate the password.
|
||||||
|
*/
|
||||||
|
public void authenticate(byte[] auth) throws IOException {
|
||||||
|
String cmd = "AUTHENTICATE " + Bytes.hex(auth) + "\r\n";
|
||||||
|
sendAndWaitForResponse(cmd, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Instructs the server to write out its configuration options into its torrc.
|
||||||
|
*/
|
||||||
|
public void saveConf() throws IOException {
|
||||||
|
sendAndWaitForResponse("SAVECONF\r\n", null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a signal from the controller to the Tor server.
|
||||||
|
* <b>signal</b> is one of the following Strings:
|
||||||
|
* <ul>
|
||||||
|
* <li>"RELOAD" or "HUP" : Reload config items, refetch directory</li>
|
||||||
|
* <li>"SHUTDOWN" or "INT" : Controlled shutdown: if server is an OP, exit immediately.
|
||||||
|
* If it's an OR, close listeners and exit after 30 seconds</li>
|
||||||
|
* <li>"DUMP" or "USR1" : Dump stats: log information about open connections and circuits</li>
|
||||||
|
* <li>"DEBUG" or "USR2" : Debug: switch all open logs to loglevel debug</li>
|
||||||
|
* <li>"HALT" or "TERM" : Immediate shutdown: clean up and exit now</li>
|
||||||
|
* </ul>
|
||||||
|
*/
|
||||||
|
public void signal(String signal) throws IOException {
|
||||||
|
String cmd = "SIGNAL " + signal + "\r\n";
|
||||||
|
sendAndWaitForResponse(cmd, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send a signal to the Tor process to shut it down or halt it.
|
||||||
|
* Does not wait for a response.
|
||||||
|
*/
|
||||||
|
public void shutdownTor(String signal) throws IOException {
|
||||||
|
String s = "SIGNAL " + signal + "\r\n";
|
||||||
|
Waiter w = new Waiter();
|
||||||
|
if (debugOutput != null)
|
||||||
|
debugOutput.print(">> " + s);
|
||||||
|
LOG.info("Entering synchronized (waiters)");
|
||||||
|
synchronized (waiters) {
|
||||||
|
LOG.info("Entered synchronized (waiters)");
|
||||||
|
output.write(s);
|
||||||
|
output.flush();
|
||||||
|
LOG.info("Leaving synchronized (waiters)");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells the Tor server that future SOCKS requests for connections to a set of original
|
||||||
|
* addresses should be replaced with connections to the specified replacement
|
||||||
|
* addresses. Each element of <b>kvLines</b> is a String of the form
|
||||||
|
* "old-address new-address". This function returns the new address mapping.
|
||||||
|
* <p>
|
||||||
|
* The client may decline to provide a body for the original address, and
|
||||||
|
* instead send a special null address ("0.0.0.0" for IPv4, "::0" for IPv6, or
|
||||||
|
* "." for hostname), signifying that the server should choose the original
|
||||||
|
* address itself, and return that address in the reply. The server
|
||||||
|
* should ensure that it returns an element of address space that is unlikely
|
||||||
|
* to be in actual use. If there is already an address mapped to the
|
||||||
|
* destination address, the server may reuse that mapping.
|
||||||
|
* <p>
|
||||||
|
* If the original address is already mapped to a different address, the old
|
||||||
|
* mapping is removed. If the original address and the destination address
|
||||||
|
* are the same, the server removes any mapping in place for the original
|
||||||
|
* address.
|
||||||
|
* <p>
|
||||||
|
* Mappings set by the controller last until the Tor process exits:
|
||||||
|
* they never expire. If the controller wants the mapping to last only
|
||||||
|
* a certain time, then it must explicitly un-map the address when that
|
||||||
|
* time has elapsed.
|
||||||
|
*/
|
||||||
|
public Map<String, String> mapAddresses(Collection<String> kvLines)
|
||||||
|
throws IOException {
|
||||||
|
StringBuffer sb = new StringBuffer("MAPADDRESS");
|
||||||
|
for (Iterator<String> it = kvLines.iterator(); it.hasNext(); ) {
|
||||||
|
String kv = it.next();
|
||||||
|
int i = kv.indexOf(' ');
|
||||||
|
sb.append(" ").append(kv.substring(0, i)).append("=")
|
||||||
|
.append(quote(kv.substring(i + 1)));
|
||||||
|
}
|
||||||
|
sb.append("\r\n");
|
||||||
|
List<ReplyLine> lst = sendAndWaitForResponse(sb.toString(), null);
|
||||||
|
Map<String, String> result = new HashMap<>();
|
||||||
|
for (Iterator<ReplyLine> it = lst.iterator(); it.hasNext(); ) {
|
||||||
|
String kv = (it.next()).msg;
|
||||||
|
int idx = kv.indexOf('=');
|
||||||
|
result.put(kv.substring(0, idx),
|
||||||
|
kv.substring(idx + 1));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, String> mapAddresses(Map<String, String> addresses)
|
||||||
|
throws IOException {
|
||||||
|
List<String> kvList = new ArrayList<>();
|
||||||
|
for (Iterator<Map.Entry<String, String>> it =
|
||||||
|
addresses.entrySet().iterator(); it.hasNext(); ) {
|
||||||
|
Map.Entry<String, String> e = it.next();
|
||||||
|
kvList.add(e.getKey() + " " + e.getValue());
|
||||||
|
}
|
||||||
|
return mapAddresses(kvList);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String mapAddress(String fromAddr, String toAddr)
|
||||||
|
throws IOException {
|
||||||
|
List<String> lst = new ArrayList<>();
|
||||||
|
lst.add(fromAddr + " " + toAddr + "\n");
|
||||||
|
Map<String, String> m = mapAddresses(lst);
|
||||||
|
return m.get(fromAddr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Queries the Tor server for keyed values that are not stored in the torrc
|
||||||
|
* configuration file. Returns a map of keys to values.
|
||||||
|
* <p>
|
||||||
|
* Recognized keys include:
|
||||||
|
* <ul>
|
||||||
|
* <li>"version" : The version of the server's software, including the name
|
||||||
|
* of the software. (example: "Tor 0.0.9.4")</li>
|
||||||
|
* <li>"desc/id/<OR identity>" or "desc/name/<OR nickname>" : the latest server
|
||||||
|
* descriptor for a given OR, NUL-terminated. If no such OR is known, the
|
||||||
|
* corresponding value is an empty string.</li>
|
||||||
|
* <li>"network-status" : a space-separated list of all known OR identities.
|
||||||
|
* This is in the same format as the router-status line in directories;
|
||||||
|
* see tor-spec.txt for details.</li>
|
||||||
|
* <li>"addr-mappings/all"</li>
|
||||||
|
* <li>"addr-mappings/config"</li>
|
||||||
|
* <li>"addr-mappings/cache"</li>
|
||||||
|
* <li>"addr-mappings/control" : a space-separated list of address mappings, each
|
||||||
|
* in the form of "from-address=to-address". The 'config' key
|
||||||
|
* returns those address mappings set in the configuration; the 'cache'
|
||||||
|
* key returns the mappings in the client-side DNS cache; the 'control'
|
||||||
|
* key returns the mappings set via the control interface; the 'all'
|
||||||
|
* target returns the mappings set through any mechanism.</li>
|
||||||
|
* <li>"circuit-status" : A series of lines as for a circuit status event. Each line is of the form:
|
||||||
|
* "CircuitID CircStatus Path"</li>
|
||||||
|
* <li>"stream-status" : A series of lines as for a stream status event. Each is of the form:
|
||||||
|
* "StreamID StreamStatus CircID Target"</li>
|
||||||
|
* <li>"orconn-status" : A series of lines as for an OR connection status event. Each is of the
|
||||||
|
* form: "ServerID ORStatus"</li>
|
||||||
|
* </ul>
|
||||||
|
*/
|
||||||
|
public Map<String, String> getInfo(Collection<String> keys)
|
||||||
|
throws IOException {
|
||||||
|
StringBuffer sb = new StringBuffer("GETINFO");
|
||||||
|
for (Iterator<String> it = keys.iterator(); it.hasNext(); ) {
|
||||||
|
sb.append(" ").append(it.next());
|
||||||
|
}
|
||||||
|
sb.append("\r\n");
|
||||||
|
List<ReplyLine> lst = sendAndWaitForResponse(sb.toString(), null);
|
||||||
|
Map<String, String> m = new HashMap<>();
|
||||||
|
for (Iterator<ReplyLine> it = lst.iterator(); it.hasNext(); ) {
|
||||||
|
ReplyLine line = it.next();
|
||||||
|
int idx = line.msg.indexOf('=');
|
||||||
|
if (idx < 0)
|
||||||
|
break;
|
||||||
|
String k = line.msg.substring(0, idx);
|
||||||
|
String v;
|
||||||
|
if (line.rest != null) {
|
||||||
|
v = line.rest;
|
||||||
|
} else {
|
||||||
|
v = line.msg.substring(idx + 1);
|
||||||
|
}
|
||||||
|
m.put(k, v);
|
||||||
|
}
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the value of the information field 'key'
|
||||||
|
*/
|
||||||
|
public String getInfo(String key) throws IOException {
|
||||||
|
List<String> lst = new ArrayList<>();
|
||||||
|
lst.add(key);
|
||||||
|
Map<String, String> m = getInfo(lst);
|
||||||
|
return m.get(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An extendCircuit request takes one of two forms: either the <b>circID</b> is zero, in
|
||||||
|
* which case it is a request for the server to build a new circuit according
|
||||||
|
* to the specified path, or the <b>circID</b> is nonzero, in which case it is a
|
||||||
|
* request for the server to extend an existing circuit with that ID according
|
||||||
|
* to the specified <b>path</b>.
|
||||||
|
* <p>
|
||||||
|
* If successful, returns the Circuit ID of the (maybe newly created) circuit.
|
||||||
|
*/
|
||||||
|
public String extendCircuit(String circID, String path) throws IOException {
|
||||||
|
List<ReplyLine> lst = sendAndWaitForResponse(
|
||||||
|
"EXTENDCIRCUIT " + circID + " " + path + "\r\n", null);
|
||||||
|
return (lst.get(0)).msg;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Informs the Tor server that the stream specified by <b>streamID</b> should be
|
||||||
|
* associated with the circuit specified by <b>circID</b>.
|
||||||
|
* <p>
|
||||||
|
* Each stream may be associated with
|
||||||
|
* at most one circuit, and multiple streams may share the same circuit.
|
||||||
|
* Streams can only be attached to completed circuits (that is, circuits that
|
||||||
|
* have sent a circuit status "BUILT" event or are listed as built in a
|
||||||
|
* getInfo circuit-status request).
|
||||||
|
* <p>
|
||||||
|
* If <b>circID</b> is 0, responsibility for attaching the given stream is
|
||||||
|
* returned to Tor.
|
||||||
|
* <p>
|
||||||
|
* By default, Tor automatically attaches streams to
|
||||||
|
* circuits itself, unless the configuration variable
|
||||||
|
* "__LeaveStreamsUnattached" is set to "1". Attempting to attach streams
|
||||||
|
* via TC when "__LeaveStreamsUnattached" is false may cause a race between
|
||||||
|
* Tor and the controller, as both attempt to attach streams to circuits.
|
||||||
|
*/
|
||||||
|
public void attachStream(String streamID, String circID)
|
||||||
|
throws IOException {
|
||||||
|
sendAndWaitForResponse(
|
||||||
|
"ATTACHSTREAM " + streamID + " " + circID + "\r\n", null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor about the server descriptor in <b>desc</b>.
|
||||||
|
* <p>
|
||||||
|
* The descriptor, when parsed, must contain a number of well-specified
|
||||||
|
* fields, including fields for its nickname and identity.
|
||||||
|
*/
|
||||||
|
// More documentation here on format of desc?
|
||||||
|
// No need for return value? control-spec.txt says reply is merely "250 OK" on success...
|
||||||
|
public String postDescriptor(String desc) throws IOException {
|
||||||
|
List<ReplyLine> lst =
|
||||||
|
sendAndWaitForResponse("+POSTDESCRIPTOR\r\n", desc);
|
||||||
|
return (lst.get(0)).msg;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to change the exit address of the stream identified by <b>streamID</b>
|
||||||
|
* to <b>address</b>. No remapping is performed on the new provided address.
|
||||||
|
* <p>
|
||||||
|
* To be sure that the modified address will be used, this event must be sent
|
||||||
|
* after a new stream event is received, and before attaching this stream to
|
||||||
|
* a circuit.
|
||||||
|
*/
|
||||||
|
public void redirectStream(String streamID, String address)
|
||||||
|
throws IOException {
|
||||||
|
sendAndWaitForResponse(
|
||||||
|
"REDIRECTSTREAM " + streamID + " " + address + "\r\n",
|
||||||
|
null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to close the stream identified by <b>streamID</b>.
|
||||||
|
* <b>reason</b> should be one of the Tor RELAY_END reasons given in tor-spec.txt, as a decimal:
|
||||||
|
* <ul>
|
||||||
|
* <li>1 -- REASON_MISC (catch-all for unlisted reasons)</li>
|
||||||
|
* <li>2 -- REASON_RESOLVEFAILED (couldn't look up hostname)</li>
|
||||||
|
* <li>3 -- REASON_CONNECTREFUSED (remote host refused connection)</li>
|
||||||
|
* <li>4 -- REASON_EXITPOLICY (OR refuses to connect to host or port)</li>
|
||||||
|
* <li>5 -- REASON_DESTROY (Circuit is being destroyed)</li>
|
||||||
|
* <li>6 -- REASON_DONE (Anonymized TCP connection was closed)</li>
|
||||||
|
* <li>7 -- REASON_TIMEOUT (Connection timed out, or OR timed out while connecting)</li>
|
||||||
|
* <li>8 -- (unallocated)</li>
|
||||||
|
* <li>9 -- REASON_HIBERNATING (OR is temporarily hibernating)</li>
|
||||||
|
* <li>10 -- REASON_INTERNAL (Internal error at the OR)</li>
|
||||||
|
* <li>11 -- REASON_RESOURCELIMIT (OR has no resources to fulfill request)</li>
|
||||||
|
* <li>12 -- REASON_CONNRESET (Connection was unexpectedly reset)</li>
|
||||||
|
* <li>13 -- REASON_TORPROTOCOL (Sent when closing connection because of Tor protocol violations)</li>
|
||||||
|
* </ul>
|
||||||
|
* <p>
|
||||||
|
* Tor may hold the stream open for a while to flush any data that is pending.
|
||||||
|
*/
|
||||||
|
public void closeStream(String streamID, byte reason)
|
||||||
|
throws IOException {
|
||||||
|
sendAndWaitForResponse(
|
||||||
|
"CLOSESTREAM " + streamID + " " + reason + "\r\n", null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to close the circuit identified by <b>circID</b>.
|
||||||
|
* If <b>ifUnused</b> is true, do not close the circuit unless it is unused.
|
||||||
|
*/
|
||||||
|
public void closeCircuit(String circID, boolean ifUnused)
|
||||||
|
throws IOException {
|
||||||
|
sendAndWaitForResponse("CLOSECIRCUIT " + circID +
|
||||||
|
(ifUnused ? " IFUNUSED" : "") + "\r\n", null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to exit when this control connection is closed. This command
|
||||||
|
* was added in Tor 0.2.2.28-beta.
|
||||||
|
*/
|
||||||
|
public void takeOwnership() throws IOException {
|
||||||
|
sendAndWaitForResponse("TAKEOWNERSHIP\r\n", null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to generate and set up a new onion service using the best
|
||||||
|
* supported algorithm.
|
||||||
|
* <p/>
|
||||||
|
* ADD_ONION was added in Tor 0.2.7.1-alpha.
|
||||||
|
*/
|
||||||
|
public Map<String, String> addOnion(Map<Integer, String> portLines)
|
||||||
|
throws IOException {
|
||||||
|
return addOnion("NEW:BEST", portLines, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to generate and set up a new onion service using the best
|
||||||
|
* supported algorithm.
|
||||||
|
* <p/>
|
||||||
|
* ADD_ONION was added in Tor 0.2.7.1-alpha.
|
||||||
|
*/
|
||||||
|
public Map<String, String> addOnion(Map<Integer, String> portLines,
|
||||||
|
boolean ephemeral, boolean detach)
|
||||||
|
throws IOException {
|
||||||
|
return addOnion("NEW:BEST", portLines, ephemeral, detach);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to set up an onion service using the provided private key.
|
||||||
|
* <p/>
|
||||||
|
* ADD_ONION was added in Tor 0.2.7.1-alpha.
|
||||||
|
*/
|
||||||
|
public Map<String, String> addOnion(String privKey,
|
||||||
|
Map<Integer, String> portLines)
|
||||||
|
throws IOException {
|
||||||
|
return addOnion(privKey, portLines, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to set up an onion service using the provided private key.
|
||||||
|
* <p/>
|
||||||
|
* ADD_ONION was added in Tor 0.2.7.1-alpha.
|
||||||
|
*/
|
||||||
|
public Map<String, String> addOnion(String privKey,
|
||||||
|
Map<Integer, String> portLines,
|
||||||
|
boolean ephemeral, boolean detach)
|
||||||
|
throws IOException {
|
||||||
|
List<String> flags = new ArrayList<>();
|
||||||
|
if (ephemeral)
|
||||||
|
flags.add("DiscardPK");
|
||||||
|
if (detach)
|
||||||
|
flags.add("Detach");
|
||||||
|
return addOnion(privKey, portLines, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to set up an onion service.
|
||||||
|
* <p/>
|
||||||
|
* ADD_ONION was added in Tor 0.2.7.1-alpha.
|
||||||
|
*/
|
||||||
|
public Map<String, String> addOnion(String privKey,
|
||||||
|
Map<Integer, String> portLines,
|
||||||
|
List<String> flags)
|
||||||
|
throws IOException {
|
||||||
|
if (privKey.indexOf(':') < 0)
|
||||||
|
throw new IllegalArgumentException("Invalid privKey");
|
||||||
|
if (portLines == null || portLines.size() < 1)
|
||||||
|
throw new IllegalArgumentException(
|
||||||
|
"Must provide at least one port line");
|
||||||
|
StringBuilder b = new StringBuilder();
|
||||||
|
b.append("ADD_ONION ").append(privKey);
|
||||||
|
if (flags != null && flags.size() > 0) {
|
||||||
|
b.append(" Flags=");
|
||||||
|
String separator = "";
|
||||||
|
for (String flag : flags) {
|
||||||
|
b.append(separator).append(flag);
|
||||||
|
separator = ",";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (Map.Entry<Integer, String> portLine : portLines.entrySet()) {
|
||||||
|
int virtPort = portLine.getKey();
|
||||||
|
String target = portLine.getValue();
|
||||||
|
b.append(" Port=").append(virtPort);
|
||||||
|
if (target != null && target.length() > 0)
|
||||||
|
b.append(",").append(target);
|
||||||
|
}
|
||||||
|
b.append("\r\n");
|
||||||
|
List<ReplyLine> lst = sendAndWaitForResponse(b.toString(), null);
|
||||||
|
Map<String, String> ret = new HashMap<>();
|
||||||
|
ret.put(HS_ADDRESS, (lst.get(0)).msg.split("=", 2)[1]);
|
||||||
|
if (lst.size() > 2)
|
||||||
|
ret.put(HS_PRIVKEY, (lst.get(1)).msg.split("=", 2)[1]);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to take down an onion service previously set up with
|
||||||
|
* addOnion(). The hostname excludes the .onion extension.
|
||||||
|
* <p/>
|
||||||
|
* DEL_ONION was added in Tor 0.2.7.1-alpha.
|
||||||
|
*/
|
||||||
|
public void delOnion(String hostname) throws IOException {
|
||||||
|
sendAndWaitForResponse("DEL_ONION " + hostname + "\r\n", null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tells Tor to forget any cached client state relating to the hidden
|
||||||
|
* service with the given hostname (excluding the .onion extension).
|
||||||
|
*/
|
||||||
|
public void forgetHiddenService(String hostname) throws IOException {
|
||||||
|
sendAndWaitForResponse("HSFORGET " + hostname + "\r\n", null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An exception raised when Tor tells us about an error.
|
||||||
|
*/
|
||||||
|
public class TorControlError extends IOException {
|
||||||
|
|
||||||
|
static final long serialVersionUID = 3;
|
||||||
|
|
||||||
|
private final int errorType;
|
||||||
|
|
||||||
|
public TorControlError(int type, String s) {
|
||||||
|
super(s);
|
||||||
|
errorType = type;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TorControlError(String s) {
|
||||||
|
this(-1, s);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getErrorType() {
|
||||||
|
return errorType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getErrorMsg() {
|
||||||
|
try {
|
||||||
|
if (errorType == -1)
|
||||||
|
return null;
|
||||||
|
return TorControlCommands.ERROR_MSGS[errorType];
|
||||||
|
} catch (ArrayIndexOutOfBoundsException ex) {
|
||||||
|
return "Unrecongized error #"+errorType;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An exception raised when Tor behaves in an unexpected way.
|
||||||
|
*/
|
||||||
|
public class TorControlSyntaxError extends IOException {
|
||||||
|
|
||||||
|
static final long serialVersionUID = 3;
|
||||||
|
|
||||||
|
public TorControlSyntaxError(String s) { super(s); }
|
||||||
|
}
|
||||||
|
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
*.class
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
// Copyright 2005 Nick Mathewson, Roger Dingledine
|
||||||
|
// See LICENSE file for copying information
|
||||||
|
package net.freehaven.tor.control.examples;
|
||||||
|
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import net.freehaven.tor.control.EventHandler;
|
||||||
|
|
||||||
|
public class DebuggingEventHandler implements EventHandler {
|
||||||
|
|
||||||
|
private final PrintWriter out;
|
||||||
|
|
||||||
|
public DebuggingEventHandler(PrintWriter p) {
|
||||||
|
out = p;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void circuitStatus(String status, String circID, String path) {
|
||||||
|
out.println("Circuit "+circID+" is now "+status+" (path="+path+")");
|
||||||
|
}
|
||||||
|
public void streamStatus(String status, String streamID, String target) {
|
||||||
|
out.println("Stream "+streamID+" is now "+status+" (target="+target+")");
|
||||||
|
}
|
||||||
|
public void orConnStatus(String status, String orName) {
|
||||||
|
out.println("OR connection to "+orName+" is now "+status);
|
||||||
|
}
|
||||||
|
public void bandwidthUsed(long read, long written) {
|
||||||
|
out.println("Bandwidth usage: "+read+" bytes read; "+
|
||||||
|
written+" bytes written.");
|
||||||
|
}
|
||||||
|
public void newDescriptors(java.util.List<String> orList) {
|
||||||
|
out.println("New descriptors for routers:");
|
||||||
|
for (Iterator<String> i = orList.iterator(); i.hasNext(); )
|
||||||
|
out.println(" "+i.next());
|
||||||
|
}
|
||||||
|
public void message(String type, String msg) {
|
||||||
|
out.println("["+type+"] "+msg.trim());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void unrecognized(String type, String msg) {
|
||||||
|
out.println("unrecognized event ["+type+"] "+msg.trim());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user