mirror of
https://github.com/chatmail/core.git
synced 2026-04-17 21:46:35 +03:00
Compare commits
675 Commits
v1.157.3
...
iequidoo/h
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
267026b44e | ||
|
|
22da92c563 | ||
|
|
357f7107a6 | ||
|
|
d96b4beff1 | ||
|
|
56c605fa0a | ||
|
|
2e9fd1c25d | ||
|
|
1b1a5f170e | ||
|
|
1946603be6 | ||
|
|
c43b622c23 | ||
|
|
73bf6983b9 | ||
|
|
aaa0f8e245 | ||
|
|
5a1e0e8824 | ||
|
|
cf5b145ce0 | ||
|
|
dd11a0e29a | ||
|
|
3d86cb5953 | ||
|
|
75eb94e44f | ||
|
|
7fef812b1e | ||
|
|
5f174ceaf2 | ||
|
|
06b038ab5d | ||
|
|
b20da3cb0e | ||
|
|
a3328ea2de | ||
|
|
ee75094bef | ||
|
|
a40fd288fc | ||
|
|
81ba2d20d6 | ||
|
|
f04c881b8c | ||
|
|
ee6b9075aa | ||
|
|
9c2a13b88e | ||
|
|
1db6ea70cc | ||
|
|
da2d9620cd | ||
|
|
d1dcb739f2 | ||
|
|
e34687ba42 | ||
|
|
5034449009 | ||
|
|
997e8216bf | ||
|
|
7f059140be | ||
|
|
c9b3da4a1a | ||
|
|
098084b9a7 | ||
|
|
9bc2aeebb8 | ||
|
|
56370c2f90 | ||
|
|
59959259bf | ||
|
|
08f8f488b1 | ||
|
|
f34311d5c4 | ||
|
|
885a5efa39 | ||
|
|
8b4c718b6b | ||
|
|
2ada3cd613 | ||
|
|
b920552fc3 | ||
|
|
92c31903c6 | ||
|
|
145145f0fb | ||
|
|
05ba206c5a | ||
|
|
9f0d106818 | ||
|
|
21caf87119 | ||
|
|
4abc695790 | ||
|
|
df1a7ca386 | ||
|
|
a06ba35ce1 | ||
|
|
18445c09c2 | ||
|
|
f428033d95 | ||
|
|
0e30dd895f | ||
|
|
c001a9a983 | ||
|
|
5f3948b462 | ||
|
|
45a1d81805 | ||
|
|
19d7799324 | ||
|
|
24e18c1485 | ||
|
|
3eb1a7dfac | ||
|
|
2f2a147efb | ||
|
|
f4938465c3 | ||
|
|
129137b5de | ||
|
|
ec3f765727 | ||
|
|
a743ad9490 | ||
|
|
89315b8ef2 | ||
|
|
e7348a4fd8 | ||
|
|
c68244692d | ||
|
|
3c93f61b4d | ||
|
|
51b9e86d71 | ||
|
|
347938a9f9 | ||
|
|
9897ef2e9b | ||
|
|
2f34a740c7 | ||
|
|
fc81cef113 | ||
|
|
04c2585c27 | ||
|
|
59fac54f7b | ||
|
|
65b61efb31 | ||
|
|
afc74b0829 | ||
|
|
2481a0f48e | ||
|
|
6c24edb40d | ||
|
|
e4178789da | ||
|
|
b417ba86bc | ||
|
|
498a831873 | ||
|
|
c6722d36de | ||
|
|
90f0d5c060 | ||
|
|
90ec2f2518 | ||
|
|
5b66535134 | ||
|
|
eea848f72b | ||
|
|
214a1d3e2d | ||
|
|
e270a502d1 | ||
|
|
b863345600 | ||
|
|
61b49a9339 | ||
|
|
41c80cf3f2 | ||
|
|
6fd3645360 | ||
|
|
b812d0a7f7 | ||
|
|
e8a4c9237d | ||
|
|
5256013615 | ||
|
|
9826c28581 | ||
|
|
9ceceebdc3 | ||
|
|
187d913f84 | ||
|
|
4a0b180d86 | ||
|
|
6fa6055912 | ||
|
|
667995cde4 | ||
|
|
1e0def87fd | ||
|
|
a219e5ee8c | ||
|
|
8070dfcc82 | ||
|
|
176a89bd03 | ||
|
|
dd8dd2f95c | ||
|
|
eb1bd1d200 | ||
|
|
460d2f3c2a | ||
|
|
0ab10f99fd | ||
|
|
377f57f1c3 | ||
|
|
caf5f1f619 | ||
|
|
d9ff85a202 | ||
|
|
f180a7c024 | ||
|
|
7fac9332e1 | ||
|
|
8dd7c42f69 | ||
|
|
b542eeecc0 | ||
|
|
bee8295daa | ||
|
|
ab9fd3d5ed | ||
|
|
cc54a3feda | ||
|
|
94984f35ec | ||
|
|
0e47e89d63 | ||
|
|
2d7dc7a1be | ||
|
|
4d76a5b599 | ||
|
|
87035ff744 | ||
|
|
e0d123f732 | ||
|
|
8eddcfc9d2 | ||
|
|
af58b86b60 | ||
|
|
00ae7ce33c | ||
|
|
0bc9fe841a | ||
|
|
e37920ed4e | ||
|
|
6a7466df93 | ||
|
|
1bb966e5a8 | ||
|
|
34e631395f | ||
|
|
080ddde68d | ||
|
|
209a8026fb | ||
|
|
23bfa4fc43 | ||
|
|
58d40c118c | ||
|
|
9d39769445 | ||
|
|
bfc08abe88 | ||
|
|
6a7b097273 | ||
|
|
8f2390ac99 | ||
|
|
481f5cae22 | ||
|
|
b9068b95b8 | ||
|
|
df2c35b551 | ||
|
|
3cd4152a3c | ||
|
|
2534510f0b | ||
|
|
3f8aa4635e | ||
|
|
ada59e8205 | ||
|
|
9ec0332483 | ||
|
|
d509b0cf5c | ||
|
|
4d624d8c3a | ||
|
|
9f0ba4b9c2 | ||
|
|
a930ae27be | ||
|
|
38e4919be1 | ||
|
|
a668047f75 | ||
|
|
c2ea2cda4c | ||
|
|
f3c3a2c301 | ||
|
|
0da7e587a7 | ||
|
|
e6e686aaf4 | ||
|
|
58e1fa5c36 | ||
|
|
42549526c7 | ||
|
|
9fe1c8fe80 | ||
|
|
b8dbcb3dbd | ||
|
|
7c5675670a | ||
|
|
291945a4fd | ||
|
|
439e8827bd | ||
|
|
a745cf78ee | ||
|
|
af69756df0 | ||
|
|
46c42ab6e4 | ||
|
|
33a127187b | ||
|
|
24ddbdd251 | ||
|
|
0122a98eea | ||
|
|
406545c1f1 | ||
|
|
a1b593027b | ||
|
|
eae1ba258a | ||
|
|
d2db30eabc | ||
|
|
9fb7c52217 | ||
|
|
6cab1786d3 | ||
|
|
362328167c | ||
|
|
570a9993f7 | ||
|
|
5adc68cf0b | ||
|
|
1b1757ebf2 | ||
|
|
d8950fb7d1 | ||
|
|
ba2e573c23 | ||
|
|
31391fc074 | ||
|
|
f94b2c3794 | ||
|
|
eb0a5fed8e | ||
|
|
eaa47d175f | ||
|
|
e968000a89 | ||
|
|
1ba448fe19 | ||
|
|
a5c82425f4 | ||
|
|
1bd31f6b8e | ||
|
|
c0ea0e52b3 | ||
|
|
e6a3daacb3 | ||
|
|
09dabda4a3 | ||
|
|
f523d912af | ||
|
|
90b0ca79ea | ||
|
|
a506e2d5a2 | ||
|
|
4c66518a68 | ||
|
|
42b4b83f8e | ||
|
|
7477ebbdd7 | ||
|
|
738dc5ce19 | ||
|
|
3680467e14 | ||
|
|
c5ada9b203 | ||
|
|
3d2805bc78 | ||
|
|
2dde286d68 | ||
|
|
2260156c40 | ||
|
|
129e970727 | ||
|
|
66271db8c0 | ||
|
|
09d33e62bd | ||
|
|
bf3dfa4ab6 | ||
|
|
40b866117e | ||
|
|
cb5f9f3051 | ||
|
|
80f97cf9bd | ||
|
|
6d860f7eae | ||
|
|
545643b610 | ||
|
|
7ee6f2c36a | ||
|
|
5d9b887624 | ||
|
|
12c0e298f5 | ||
|
|
f9aec7af0d | ||
|
|
b181d78dd5 | ||
|
|
b9ff40c6b5 | ||
|
|
0684810d38 | ||
|
|
1cc7ce6e27 | ||
|
|
82bc1bf0b1 | ||
|
|
75bcf8660b | ||
|
|
5e1d945198 | ||
|
|
e047184ede | ||
|
|
307a2eb6ec | ||
|
|
ab8aedf06e | ||
|
|
b6ab13f1de | ||
|
|
53a3e51920 | ||
|
|
4033566b4a | ||
|
|
bed1623dcb | ||
|
|
d4704977bc | ||
|
|
838eed94bc | ||
|
|
9870725d1f | ||
|
|
ba827283be | ||
|
|
1e37cb8c3c | ||
|
|
1991e01641 | ||
|
|
d7e87b6336 | ||
|
|
fde490ba15 | ||
|
|
cf5a16d967 | ||
|
|
e8dde9c63d | ||
|
|
667a935665 | ||
|
|
28cea706fa | ||
|
|
209a990444 | ||
|
|
6365a46fac | ||
|
|
a81496e9ab | ||
|
|
ca05733b9d | ||
|
|
dfb5348a78 | ||
|
|
602e52490c | ||
|
|
740b24e8a4 | ||
|
|
44a09ffd12 | ||
|
|
054c42cbc2 | ||
|
|
34263a70e2 | ||
|
|
7ea6ca35d7 | ||
|
|
a9aad497fc | ||
|
|
7da8489635 | ||
|
|
683561374d | ||
|
|
66c9982822 | ||
|
|
1b6450b210 | ||
|
|
aa8a13adb2 | ||
|
|
5888541c05 | ||
|
|
f893487dc0 | ||
|
|
b84beaf974 | ||
|
|
75a3c55e70 | ||
|
|
854a09e12f | ||
|
|
40412fd4a9 | ||
|
|
57fc084795 | ||
|
|
143ba6d5e7 | ||
|
|
6b338a923c | ||
|
|
e6ab1e3df5 | ||
|
|
5da6976bf9 | ||
|
|
bd15d90e77 | ||
|
|
61633cf23b | ||
|
|
9f1107c0e7 | ||
|
|
ff0d5ce179 | ||
|
|
0bbd910883 | ||
|
|
4258088fb4 | ||
|
|
6372b677d2 | ||
|
|
9af00af70f | ||
|
|
4010c60e7b | ||
|
|
aaa83a8f52 | ||
|
|
776408c564 | ||
|
|
d0cb2110e6 | ||
|
|
11e3480fe8 | ||
|
|
2cd54b72b0 | ||
|
|
c34ccafb2e | ||
|
|
6837874d43 | ||
|
|
3656337d41 | ||
|
|
a89b6321f1 | ||
|
|
ac10103b18 | ||
|
|
b696a242fc | ||
|
|
7e4822c8ca | ||
|
|
a955cb5400 | ||
|
|
2e2cfc4cb3 | ||
|
|
4157d1986f | ||
|
|
d13eb2f580 | ||
|
|
5476f69179 | ||
|
|
dcdf30da35 | ||
|
|
55746c8c19 | ||
|
|
dbdf5f2746 | ||
|
|
b4e28deed3 | ||
|
|
f4a604dcfb | ||
|
|
b3c5787ec8 | ||
|
|
471d0469dd | ||
|
|
113eda575f | ||
|
|
45f1da82fe | ||
|
|
5f45ff77e4 | ||
|
|
1c0201ee3d | ||
|
|
c7340e04ec | ||
|
|
0a32476dc5 | ||
|
|
e02bc6ffb5 | ||
|
|
f41a3970b2 | ||
|
|
6c536f3a9b | ||
|
|
4b24b6a848 | ||
|
|
5f254a929f | ||
|
|
8df1a01ace | ||
|
|
27b5ffb34f | ||
|
|
80af012962 | ||
|
|
615c80bef4 | ||
|
|
f5f4026dbb | ||
|
|
b431206ede | ||
|
|
c4878e9b49 | ||
|
|
aa452971a6 | ||
|
|
2d798f7cfe | ||
|
|
08bb0484eb | ||
|
|
b0b7337f5a | ||
|
|
93241a4beb | ||
|
|
4f1bf1f13c | ||
|
|
2d0b7b5bd8 | ||
|
|
8fe3ce5cab | ||
|
|
59a0f1d94f | ||
|
|
5175dc3450 | ||
|
|
9a22ccd058 | ||
|
|
c06ed49a2a | ||
|
|
2e51a5a454 | ||
|
|
75cc353528 | ||
|
|
3977580426 | ||
|
|
3a1370e174 | ||
|
|
c218c05b96 | ||
|
|
db247d9f9a | ||
|
|
78b7715ea6 | ||
|
|
ba76944d75 | ||
|
|
4a1a2122f0 | ||
|
|
d80b749dec | ||
|
|
039a8b7c36 | ||
|
|
779f58ab16 | ||
|
|
b9183fe5eb | ||
|
|
9d342671d5 | ||
|
|
4e47ebd5fc | ||
|
|
d5c418e909 | ||
|
|
85414558c5 | ||
|
|
d6af8d2526 | ||
|
|
1209e95e34 | ||
|
|
51f9279e67 | ||
|
|
f27d54f7fa | ||
|
|
7f3648f8ae | ||
|
|
49fc258578 | ||
|
|
0c51b4fe41 | ||
|
|
dbad714539 | ||
|
|
edd8008650 | ||
|
|
615a1b3f4e | ||
|
|
fe6044e1aa | ||
|
|
46b275bfab | ||
|
|
25f44c517a | ||
|
|
cac04f8ee4 | ||
|
|
45d8566ec0 | ||
|
|
29a98ba13b | ||
|
|
e3973f6448 | ||
|
|
7b41425fe4 | ||
|
|
2c7d51f98f | ||
|
|
a2df29515a | ||
|
|
6df1d165dd | ||
|
|
e03e2d9a68 | ||
|
|
8fc6ea19b4 | ||
|
|
c5c947e175 | ||
|
|
6d8dff54a7 | ||
|
|
a0f6bdffeb | ||
|
|
e6fd52afff | ||
|
|
0142515887 | ||
|
|
d45ec7f34d | ||
|
|
752f45f0f0 | ||
|
|
0299543a86 | ||
|
|
d3908d6b36 | ||
|
|
2cf979de53 | ||
|
|
f5e8c8083d | ||
|
|
58b99f59f7 | ||
|
|
402e42f858 | ||
|
|
fbae0739a6 | ||
|
|
0359481ba4 | ||
|
|
6406f305b8 | ||
|
|
e5e0f0cdd7 | ||
|
|
0bac4acdd8 | ||
|
|
ce5697c5f7 | ||
|
|
22258f7269 | ||
|
|
5ab107866a | ||
|
|
374a5ef687 | ||
|
|
1a2e355bb8 | ||
|
|
192a6a2b9d | ||
|
|
4ca0ce2fb2 | ||
|
|
ab4cb01065 | ||
|
|
661a8864b9 | ||
|
|
67f00fbb84 | ||
|
|
389649ea8a | ||
|
|
a87ee030fc | ||
|
|
3f66ae91cd | ||
|
|
75b7bea78f | ||
|
|
acba27a328 | ||
|
|
cba9eb98d6 | ||
|
|
da9b24d191 | ||
|
|
c9c5d94666 | ||
|
|
aad8f698dd | ||
|
|
35e107e87d | ||
|
|
d9b361f066 | ||
|
|
94e75cb3b8 | ||
|
|
c7fb64e2f3 | ||
|
|
ebddabe958 | ||
|
|
b81f7cfcab | ||
|
|
9197ef04f7 | ||
|
|
7e4d4cf680 | ||
|
|
0a73c2b7ab | ||
|
|
2ee3f58b69 | ||
|
|
f60af72a5e | ||
|
|
95125d30ef | ||
|
|
48a9fafe6c | ||
|
|
c4cc2fe731 | ||
|
|
3df0bd8890 | ||
|
|
2a5a0717aa | ||
|
|
ee8364913b | ||
|
|
3267126a33 | ||
|
|
2ee3675ba2 | ||
|
|
faf4fd1ca6 | ||
|
|
53ebf2ca27 | ||
|
|
f3eea9937c | ||
|
|
5c3de759d3 | ||
|
|
0ffd4d9f87 | ||
|
|
416131b4a2 | ||
|
|
7ac04d0204 | ||
|
|
a40337f4e0 | ||
|
|
b45d9aa464 | ||
|
|
48b2e2bc1f | ||
|
|
545007aca5 | ||
|
|
07ce319839 | ||
|
|
0d36c85568 | ||
|
|
139fbfae85 | ||
|
|
0568393157 | ||
|
|
7ec732977a | ||
|
|
a8a7cec376 | ||
|
|
7f6beeeecb | ||
|
|
15092407ea | ||
|
|
bd70d48cdf | ||
|
|
ce04e904e2 | ||
|
|
026ddbf9f1 | ||
|
|
628b178076 | ||
|
|
823a16e8e9 | ||
|
|
407ec1311e | ||
|
|
b9667aae6b | ||
|
|
806b437209 | ||
|
|
1a5232f863 | ||
|
|
7ad119f126 | ||
|
|
1682f4b252 | ||
|
|
6a320d545b | ||
|
|
e7aebd6fbc | ||
|
|
8189abd660 | ||
|
|
5ded153ae4 | ||
|
|
2fd5507c00 | ||
|
|
becb83faf1 | ||
|
|
32263b4574 | ||
|
|
fd3e48dcb2 | ||
|
|
69573cd735 | ||
|
|
5c2af42cdd | ||
|
|
42975b2ff3 | ||
|
|
c7063c00f7 | ||
|
|
89df9536e9 | ||
|
|
0e45c2246f | ||
|
|
81a6afde15 | ||
|
|
adcc8a919c | ||
|
|
a24e6d4278 | ||
|
|
776b2247dd | ||
|
|
37dc1f5ca0 | ||
|
|
a68ddab703 | ||
|
|
877f873910 | ||
|
|
53fa0147ae | ||
|
|
7655c5b150 | ||
|
|
235b625f71 | ||
|
|
014b0024a0 | ||
|
|
b0508e661a | ||
|
|
ab3cd6a8f7 | ||
|
|
85461204c5 | ||
|
|
3abf2b5227 | ||
|
|
0d5d7032fe | ||
|
|
c48b04ab99 | ||
|
|
eaa30dbe21 | ||
|
|
bb0f812f71 | ||
|
|
4c287075da | ||
|
|
09d18f9097 | ||
|
|
47b9bfc8bf | ||
|
|
21d13e8a9c | ||
|
|
079260a7cf | ||
|
|
fdec78c092 | ||
|
|
259ffef0bb | ||
|
|
6661a0803e | ||
|
|
c1471bdbd9 | ||
|
|
a981573e48 | ||
|
|
8fb3a7514e | ||
|
|
846c8e7f1b | ||
|
|
98a1b9e373 | ||
|
|
ba55dd339e | ||
|
|
5a2ce60392 | ||
|
|
7ebcee14e7 | ||
|
|
ccf829fe8c | ||
|
|
a274f5fb86 | ||
|
|
5421a555f4 | ||
|
|
b1233b2b07 | ||
|
|
e55ac59846 | ||
|
|
cd6cd6ba47 | ||
|
|
026b06003b | ||
|
|
02141b86c2 | ||
|
|
9bb2600d73 | ||
|
|
33ea13daf4 | ||
|
|
10b6019e7e | ||
|
|
727f0ab6ce | ||
|
|
31752e9674 | ||
|
|
841d4e6e1e | ||
|
|
7dc890119d | ||
|
|
293a683484 | ||
|
|
737bc15382 | ||
|
|
1a72711999 | ||
|
|
3fea829340 | ||
|
|
6dba14158a | ||
|
|
83bc497f0d | ||
|
|
990a13fd96 | ||
|
|
29b84424f4 | ||
|
|
ef798cd86d | ||
|
|
9d3450f50c | ||
|
|
1db9b77711 | ||
|
|
a6713630b9 | ||
|
|
4168985869 | ||
|
|
1ea8647018 | ||
|
|
f311cae5ad | ||
|
|
7e8e4d2f39 | ||
|
|
1379821b03 | ||
|
|
1722cb8851 | ||
|
|
49c300d2ac | ||
|
|
0e3277bc5a | ||
|
|
9f5e608c61 | ||
|
|
0b82b42128 | ||
|
|
b4828c251f | ||
|
|
7a4f0eed23 | ||
|
|
54a6b0efcb | ||
|
|
9229eae4e0 | ||
|
|
3e8987b460 | ||
|
|
634cbd14f0 | ||
|
|
31cf663f8b | ||
|
|
175145969c | ||
|
|
8db1a01d9a | ||
|
|
b3c5f64315 | ||
|
|
35e717dd49 | ||
|
|
203e668928 | ||
|
|
de38b413f1 | ||
|
|
21010f4de6 | ||
|
|
b03edabb11 | ||
|
|
4001d79e4b | ||
|
|
3513a97a3d | ||
|
|
072855daef | ||
|
|
dc87ba87c9 | ||
|
|
2b3f030d6a | ||
|
|
a3bbdf0bec | ||
|
|
7a7f95f5ef | ||
|
|
746b071be0 | ||
|
|
d307e75b2f | ||
|
|
de5cbd3de3 | ||
|
|
5210b37601 | ||
|
|
4ad9fa144d | ||
|
|
a2d5a10f84 | ||
|
|
b056314fd0 | ||
|
|
3b35d5e0ea | ||
|
|
5e95a70eca | ||
|
|
8d1e43b9d3 | ||
|
|
ed2cf0a9d1 | ||
|
|
ab0b4cad52 | ||
|
|
11469ace78 | ||
|
|
0ab54da2eb | ||
|
|
953eb90e87 | ||
|
|
d2803c4305 | ||
|
|
ab47d6f611 | ||
|
|
07946a18c3 | ||
|
|
0e874735ac | ||
|
|
b56cf72c87 | ||
|
|
9c5cf84c9f | ||
|
|
c19197a960 | ||
|
|
cecd3a2956 | ||
|
|
c8c6beb1b6 | ||
|
|
03635c8d7f | ||
|
|
f942a63c5d | ||
|
|
211badee41 | ||
|
|
c239da542c | ||
|
|
9ea3f23fef | ||
|
|
4681b33eac | ||
|
|
28a6ff3270 | ||
|
|
71bd3aefd6 | ||
|
|
ba15591c22 | ||
|
|
e5b79bf405 | ||
|
|
cfaa8ceba2 | ||
|
|
89a73d775e | ||
|
|
66e3dc7226 | ||
|
|
fce14ebc99 | ||
|
|
5806cadad6 | ||
|
|
68ce6491a5 | ||
|
|
01638ce99e | ||
|
|
c26e43630c | ||
|
|
011779dcf7 | ||
|
|
18e5d5b67a | ||
|
|
0c1afa527b | ||
|
|
159068c772 | ||
|
|
f8841a85d7 | ||
|
|
92620d9c82 | ||
|
|
1cc03ca264 | ||
|
|
5cf8864066 | ||
|
|
c16c6f3ad6 | ||
|
|
b0fa413aa9 | ||
|
|
9c974b40ac | ||
|
|
5b47c4947f | ||
|
|
c62bab3fe5 | ||
|
|
7776060d68 | ||
|
|
3aea6884ac | ||
|
|
1ba0dd503c | ||
|
|
a1837aeb8c | ||
|
|
ee079ce021 | ||
|
|
70563867a6 | ||
|
|
f72d27f7de | ||
|
|
ddc2f55a6f | ||
|
|
8f3fc10625 | ||
|
|
97b0d09ed2 | ||
|
|
e2f9c80cd5 | ||
|
|
394cba3c78 | ||
|
|
f472c05120 | ||
|
|
3efd94914c | ||
|
|
99a6756d28 | ||
|
|
3310315865 | ||
|
|
a7729e3548 | ||
|
|
dc2e4df286 | ||
|
|
386b91a9a7 | ||
|
|
d4847206cf | ||
|
|
7624a50cb1 | ||
|
|
568c044a90 | ||
|
|
a8f8d34c25 | ||
|
|
a308766e47 | ||
|
|
0df86b6308 | ||
|
|
e951a697ec | ||
|
|
1ebaa2a718 | ||
|
|
6cb6daaab2 | ||
|
|
d25fb4770c | ||
|
|
e4e738ec5f | ||
|
|
8a5a67d6f2 | ||
|
|
ee68b9c7ba | ||
|
|
a51b2fa751 | ||
|
|
4c4646e72c | ||
|
|
2ca866b644 | ||
|
|
ed7dfd6b65 | ||
|
|
de79cd1583 | ||
|
|
0e84cfd8ad | ||
|
|
8a9e60afc3 | ||
|
|
b5fa6553af | ||
|
|
5280448cd3 | ||
|
|
891e166996 | ||
|
|
df24532503 | ||
|
|
b82fa19c6f | ||
|
|
8cb136ab9d |
35
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
35
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: Report something that isn't working.
|
||||||
|
title: ''
|
||||||
|
assignees: ''
|
||||||
|
labels: bug
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
This is the chatmail core's bug report tracker.
|
||||||
|
For Delta Chat feature requests and support, please go to the forum: https://support.delta.chat
|
||||||
|
Please fill out as much of this form as you can (leaving out stuff that is not applicable is ok).
|
||||||
|
-->
|
||||||
|
|
||||||
|
- Operating System (Linux/Mac/Windows/iOS/Android):
|
||||||
|
- Core Version:
|
||||||
|
- Client Version:
|
||||||
|
|
||||||
|
## Expected behavior
|
||||||
|
|
||||||
|
*What did you try to achieve?*
|
||||||
|
|
||||||
|
## Actual behavior
|
||||||
|
|
||||||
|
*What happened instead?*
|
||||||
|
|
||||||
|
### Steps to reproduce the problem
|
||||||
|
|
||||||
|
1.
|
||||||
|
2.
|
||||||
|
|
||||||
|
### Screenshots
|
||||||
|
|
||||||
|
### Logs
|
||||||
|
|
||||||
4
.github/dependabot.yml
vendored
4
.github/dependabot.yml
vendored
@@ -7,6 +7,8 @@ updates:
|
|||||||
commit-message:
|
commit-message:
|
||||||
prefix: "chore(cargo)"
|
prefix: "chore(cargo)"
|
||||||
open-pull-requests-limit: 50
|
open-pull-requests-limit: 50
|
||||||
|
cooldown:
|
||||||
|
default-days: 7
|
||||||
|
|
||||||
# Keep GitHub Actions up to date.
|
# Keep GitHub Actions up to date.
|
||||||
# <https://docs.github.com/en/code-security/dependabot/working-with-dependabot/keeping-your-actions-up-to-date-with-dependabot>
|
# <https://docs.github.com/en/code-security/dependabot/working-with-dependabot/keeping-your-actions-up-to-date-with-dependabot>
|
||||||
@@ -14,3 +16,5 @@ updates:
|
|||||||
directory: "/"
|
directory: "/"
|
||||||
schedule:
|
schedule:
|
||||||
interval: "weekly"
|
interval: "weekly"
|
||||||
|
cooldown:
|
||||||
|
default-days: 7
|
||||||
|
|||||||
83
.github/workflows/ci.yml
vendored
83
.github/workflows/ci.yml
vendored
@@ -20,20 +20,24 @@ permissions: {}
|
|||||||
|
|
||||||
env:
|
env:
|
||||||
RUSTFLAGS: -Dwarnings
|
RUSTFLAGS: -Dwarnings
|
||||||
|
RUST_VERSION: 1.91.0
|
||||||
|
|
||||||
|
# Minimum Supported Rust Version
|
||||||
|
MSRV: 1.85.0
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
lint_rust:
|
lint_rust:
|
||||||
name: Lint Rust
|
name: Lint Rust
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
env:
|
|
||||||
RUSTUP_TOOLCHAIN: 1.84.1
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- name: Install rustfmt and clippy
|
- name: Install rustfmt and clippy
|
||||||
run: rustup toolchain install $RUSTUP_TOOLCHAIN --profile minimal --component rustfmt --component clippy
|
run: rustup toolchain install $RUST_VERSION --profile minimal --component rustfmt --component clippy
|
||||||
|
- run: rustup override set $RUST_VERSION
|
||||||
|
shell: bash
|
||||||
- name: Cache rust cargo artifacts
|
- name: Cache rust cargo artifacts
|
||||||
uses: swatinem/rust-cache@v2
|
uses: swatinem/rust-cache@v2
|
||||||
- name: Run rustfmt
|
- name: Run rustfmt
|
||||||
@@ -49,7 +53,7 @@ jobs:
|
|||||||
name: cargo deny
|
name: cargo deny
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
@@ -63,10 +67,12 @@ jobs:
|
|||||||
name: Check provider database
|
name: Check provider database
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
|
- name: Install rustfmt
|
||||||
|
run: rustup component add --toolchain stable-x86_64-unknown-linux-gnu rustfmt
|
||||||
- name: Check provider database
|
- name: Check provider database
|
||||||
run: scripts/update-provider-database.sh
|
run: scripts/update-provider-database.sh
|
||||||
|
|
||||||
@@ -76,7 +82,7 @@ jobs:
|
|||||||
env:
|
env:
|
||||||
RUSTDOCFLAGS: -Dwarnings
|
RUSTDOCFLAGS: -Dwarnings
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
@@ -91,25 +97,36 @@ jobs:
|
|||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
- os: ubuntu-latest
|
- os: ubuntu-latest
|
||||||
rust: 1.84.1
|
rust: latest
|
||||||
- os: windows-latest
|
- os: windows-latest
|
||||||
rust: 1.84.1
|
rust: latest
|
||||||
- os: macos-latest
|
- os: macos-latest
|
||||||
rust: 1.84.1
|
rust: latest
|
||||||
|
|
||||||
# Minimum Supported Rust Version = 1.81.0
|
# Minimum Supported Rust Version
|
||||||
- os: ubuntu-latest
|
- os: ubuntu-latest
|
||||||
rust: 1.81.0
|
rust: minimum
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- run:
|
||||||
|
echo "RUSTUP_TOOLCHAIN=$MSRV" >> $GITHUB_ENV
|
||||||
|
shell: bash
|
||||||
|
if: matrix.rust == 'minimum'
|
||||||
|
- run:
|
||||||
|
echo "RUSTUP_TOOLCHAIN=$RUST_VERSION" >> $GITHUB_ENV
|
||||||
|
shell: bash
|
||||||
|
if: matrix.rust == 'latest'
|
||||||
|
|
||||||
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
|
|
||||||
- name: Install Rust ${{ matrix.rust }}
|
- name: Install Rust ${{ matrix.rust }}
|
||||||
run: rustup toolchain install --profile minimal ${{ matrix.rust }}
|
run: rustup toolchain install --profile minimal $RUSTUP_TOOLCHAIN
|
||||||
- run: rustup override set ${{ matrix.rust }}
|
shell: bash
|
||||||
|
- run: rustup override set $RUSTUP_TOOLCHAIN
|
||||||
|
shell: bash
|
||||||
|
|
||||||
- name: Cache rust cargo artifacts
|
- name: Cache rust cargo artifacts
|
||||||
uses: swatinem/rust-cache@v2
|
uses: swatinem/rust-cache@v2
|
||||||
@@ -122,12 +139,12 @@ jobs:
|
|||||||
- name: Tests
|
- name: Tests
|
||||||
env:
|
env:
|
||||||
RUST_BACKTRACE: 1
|
RUST_BACKTRACE: 1
|
||||||
run: cargo nextest run --workspace
|
run: cargo nextest run --workspace --locked
|
||||||
|
|
||||||
- name: Doc-Tests
|
- name: Doc-Tests
|
||||||
env:
|
env:
|
||||||
RUST_BACKTRACE: 1
|
RUST_BACKTRACE: 1
|
||||||
run: cargo test --workspace --doc
|
run: cargo test --workspace --locked --doc
|
||||||
|
|
||||||
- name: Test cargo vendor
|
- name: Test cargo vendor
|
||||||
run: cargo vendor
|
run: cargo vendor
|
||||||
@@ -139,7 +156,7 @@ jobs:
|
|||||||
os: [ubuntu-latest, macos-latest]
|
os: [ubuntu-latest, macos-latest]
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
@@ -151,7 +168,7 @@ jobs:
|
|||||||
run: cargo build -p deltachat_ffi
|
run: cargo build -p deltachat_ffi
|
||||||
|
|
||||||
- name: Upload C library
|
- name: Upload C library
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: ${{ matrix.os }}-libdeltachat.a
|
name: ${{ matrix.os }}-libdeltachat.a
|
||||||
path: target/debug/libdeltachat.a
|
path: target/debug/libdeltachat.a
|
||||||
@@ -164,7 +181,7 @@ jobs:
|
|||||||
os: [ubuntu-latest, macos-latest, windows-latest]
|
os: [ubuntu-latest, macos-latest, windows-latest]
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
@@ -176,7 +193,7 @@ jobs:
|
|||||||
run: cargo build -p deltachat-rpc-server
|
run: cargo build -p deltachat-rpc-server
|
||||||
|
|
||||||
- name: Upload deltachat-rpc-server
|
- name: Upload deltachat-rpc-server
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: ${{ matrix.os }}-deltachat-rpc-server
|
name: ${{ matrix.os }}-deltachat-rpc-server
|
||||||
path: ${{ matrix.os == 'windows-latest' && 'target/debug/deltachat-rpc-server.exe' || 'target/debug/deltachat-rpc-server' }}
|
path: ${{ matrix.os == 'windows-latest' && 'target/debug/deltachat-rpc-server.exe' || 'target/debug/deltachat-rpc-server' }}
|
||||||
@@ -186,7 +203,7 @@ jobs:
|
|||||||
name: Python lint
|
name: Python lint
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
@@ -211,9 +228,9 @@ jobs:
|
|||||||
include:
|
include:
|
||||||
# Currently used Rust version.
|
# Currently used Rust version.
|
||||||
- os: ubuntu-latest
|
- os: ubuntu-latest
|
||||||
python: 3.13
|
python: 3.14
|
||||||
- os: macos-latest
|
- os: macos-latest
|
||||||
python: 3.13
|
python: 3.14
|
||||||
|
|
||||||
# PyPy tests
|
# PyPy tests
|
||||||
- os: ubuntu-latest
|
- os: ubuntu-latest
|
||||||
@@ -229,19 +246,19 @@ jobs:
|
|||||||
|
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
|
|
||||||
- name: Download libdeltachat.a
|
- name: Download libdeltachat.a
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: ${{ matrix.os }}-libdeltachat.a
|
name: ${{ matrix.os }}-libdeltachat.a
|
||||||
path: target/debug
|
path: target/debug
|
||||||
|
|
||||||
- name: Install python
|
- name: Install python
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v6
|
||||||
with:
|
with:
|
||||||
python-version: ${{ matrix.python }}
|
python-version: ${{ matrix.python }}
|
||||||
|
|
||||||
@@ -264,11 +281,11 @@ jobs:
|
|||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
- os: ubuntu-latest
|
- os: ubuntu-latest
|
||||||
python: 3.13
|
python: 3.14
|
||||||
- os: macos-latest
|
- os: macos-latest
|
||||||
python: 3.13
|
python: 3.14
|
||||||
- os: windows-latest
|
- os: windows-latest
|
||||||
python: 3.13
|
python: 3.14
|
||||||
|
|
||||||
# PyPy tests
|
# PyPy tests
|
||||||
- os: ubuntu-latest
|
- os: ubuntu-latest
|
||||||
@@ -282,13 +299,13 @@ jobs:
|
|||||||
|
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
|
|
||||||
- name: Install python
|
- name: Install python
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v6
|
||||||
with:
|
with:
|
||||||
python-version: ${{ matrix.python }}
|
python-version: ${{ matrix.python }}
|
||||||
|
|
||||||
@@ -296,7 +313,7 @@ jobs:
|
|||||||
run: pip install tox
|
run: pip install tox
|
||||||
|
|
||||||
- name: Download deltachat-rpc-server
|
- name: Download deltachat-rpc-server
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: ${{ matrix.os }}-deltachat-rpc-server
|
name: ${{ matrix.os }}-deltachat-rpc-server
|
||||||
path: target/debug
|
path: target/debug
|
||||||
|
|||||||
80
.github/workflows/deltachat-rpc-server.yml
vendored
80
.github/workflows/deltachat-rpc-server.yml
vendored
@@ -30,17 +30,17 @@ jobs:
|
|||||||
arch: [aarch64, armv7l, armv6l, i686, x86_64]
|
arch: [aarch64, armv7l, armv6l, i686, x86_64]
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
|
|
||||||
- name: Build deltachat-rpc-server binaries
|
- name: Build deltachat-rpc-server binaries
|
||||||
run: nix build .#deltachat-rpc-server-${{ matrix.arch }}-linux
|
run: nix build .#deltachat-rpc-server-${{ matrix.arch }}-linux
|
||||||
|
|
||||||
- name: Upload binary
|
- name: Upload binary
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-${{ matrix.arch }}-linux
|
name: deltachat-rpc-server-${{ matrix.arch }}-linux
|
||||||
path: result/bin/deltachat-rpc-server
|
path: result/bin/deltachat-rpc-server
|
||||||
@@ -54,17 +54,17 @@ jobs:
|
|||||||
arch: [win32, win64]
|
arch: [win32, win64]
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
|
|
||||||
- name: Build deltachat-rpc-server binaries
|
- name: Build deltachat-rpc-server binaries
|
||||||
run: nix build .#deltachat-rpc-server-${{ matrix.arch }}
|
run: nix build .#deltachat-rpc-server-${{ matrix.arch }}
|
||||||
|
|
||||||
- name: Upload binary
|
- name: Upload binary
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-${{ matrix.arch }}
|
name: deltachat-rpc-server-${{ matrix.arch }}
|
||||||
path: result/bin/deltachat-rpc-server.exe
|
path: result/bin/deltachat-rpc-server.exe
|
||||||
@@ -79,7 +79,7 @@ jobs:
|
|||||||
|
|
||||||
runs-on: macos-latest
|
runs-on: macos-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
@@ -91,7 +91,7 @@ jobs:
|
|||||||
run: cargo build --release --package deltachat-rpc-server --target ${{ matrix.arch }}-apple-darwin --features vendored
|
run: cargo build --release --package deltachat-rpc-server --target ${{ matrix.arch }}-apple-darwin --features vendored
|
||||||
|
|
||||||
- name: Upload binary
|
- name: Upload binary
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-${{ matrix.arch }}-macos
|
name: deltachat-rpc-server-${{ matrix.arch }}-macos
|
||||||
path: target/${{ matrix.arch }}-apple-darwin/release/deltachat-rpc-server
|
path: target/${{ matrix.arch }}-apple-darwin/release/deltachat-rpc-server
|
||||||
@@ -105,17 +105,17 @@ jobs:
|
|||||||
arch: [arm64-v8a, armeabi-v7a]
|
arch: [arm64-v8a, armeabi-v7a]
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
|
|
||||||
- name: Build deltachat-rpc-server binaries
|
- name: Build deltachat-rpc-server binaries
|
||||||
run: nix build .#deltachat-rpc-server-${{ matrix.arch }}-android
|
run: nix build .#deltachat-rpc-server-${{ matrix.arch }}-android
|
||||||
|
|
||||||
- name: Upload binary
|
- name: Upload binary
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-${{ matrix.arch }}-android
|
name: deltachat-rpc-server-${{ matrix.arch }}-android
|
||||||
path: result/bin/deltachat-rpc-server
|
path: result/bin/deltachat-rpc-server
|
||||||
@@ -132,74 +132,74 @@ jobs:
|
|||||||
contents: write
|
contents: write
|
||||||
runs-on: "ubuntu-latest"
|
runs-on: "ubuntu-latest"
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
|
|
||||||
- name: Download Linux aarch64 binary
|
- name: Download Linux aarch64 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-aarch64-linux
|
name: deltachat-rpc-server-aarch64-linux
|
||||||
path: deltachat-rpc-server-aarch64-linux.d
|
path: deltachat-rpc-server-aarch64-linux.d
|
||||||
|
|
||||||
- name: Download Linux armv7l binary
|
- name: Download Linux armv7l binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-armv7l-linux
|
name: deltachat-rpc-server-armv7l-linux
|
||||||
path: deltachat-rpc-server-armv7l-linux.d
|
path: deltachat-rpc-server-armv7l-linux.d
|
||||||
|
|
||||||
- name: Download Linux armv6l binary
|
- name: Download Linux armv6l binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-armv6l-linux
|
name: deltachat-rpc-server-armv6l-linux
|
||||||
path: deltachat-rpc-server-armv6l-linux.d
|
path: deltachat-rpc-server-armv6l-linux.d
|
||||||
|
|
||||||
- name: Download Linux i686 binary
|
- name: Download Linux i686 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-i686-linux
|
name: deltachat-rpc-server-i686-linux
|
||||||
path: deltachat-rpc-server-i686-linux.d
|
path: deltachat-rpc-server-i686-linux.d
|
||||||
|
|
||||||
- name: Download Linux x86_64 binary
|
- name: Download Linux x86_64 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-x86_64-linux
|
name: deltachat-rpc-server-x86_64-linux
|
||||||
path: deltachat-rpc-server-x86_64-linux.d
|
path: deltachat-rpc-server-x86_64-linux.d
|
||||||
|
|
||||||
- name: Download Win32 binary
|
- name: Download Win32 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-win32
|
name: deltachat-rpc-server-win32
|
||||||
path: deltachat-rpc-server-win32.d
|
path: deltachat-rpc-server-win32.d
|
||||||
|
|
||||||
- name: Download Win64 binary
|
- name: Download Win64 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-win64
|
name: deltachat-rpc-server-win64
|
||||||
path: deltachat-rpc-server-win64.d
|
path: deltachat-rpc-server-win64.d
|
||||||
|
|
||||||
- name: Download macOS binary for x86_64
|
- name: Download macOS binary for x86_64
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-x86_64-macos
|
name: deltachat-rpc-server-x86_64-macos
|
||||||
path: deltachat-rpc-server-x86_64-macos.d
|
path: deltachat-rpc-server-x86_64-macos.d
|
||||||
|
|
||||||
- name: Download macOS binary for aarch64
|
- name: Download macOS binary for aarch64
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-aarch64-macos
|
name: deltachat-rpc-server-aarch64-macos
|
||||||
path: deltachat-rpc-server-aarch64-macos.d
|
path: deltachat-rpc-server-aarch64-macos.d
|
||||||
|
|
||||||
- name: Download Android binary for arm64-v8a
|
- name: Download Android binary for arm64-v8a
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-arm64-v8a-android
|
name: deltachat-rpc-server-arm64-v8a-android
|
||||||
path: deltachat-rpc-server-arm64-v8a-android.d
|
path: deltachat-rpc-server-arm64-v8a-android.d
|
||||||
|
|
||||||
- name: Download Android binary for armeabi-v7a
|
- name: Download Android binary for armeabi-v7a
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-armeabi-v7a-android
|
name: deltachat-rpc-server-armeabi-v7a-android
|
||||||
path: deltachat-rpc-server-armeabi-v7a-android.d
|
path: deltachat-rpc-server-armeabi-v7a-android.d
|
||||||
@@ -224,7 +224,7 @@ jobs:
|
|||||||
|
|
||||||
# Python 3.11 is needed for tomllib used in scripts/wheel-rpc-server.py
|
# Python 3.11 is needed for tomllib used in scripts/wheel-rpc-server.py
|
||||||
- name: Install python 3.12
|
- name: Install python 3.12
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v6
|
||||||
with:
|
with:
|
||||||
python-version: 3.12
|
python-version: 3.12
|
||||||
|
|
||||||
@@ -285,76 +285,76 @@ jobs:
|
|||||||
# Needed to publish the binaries to the release.
|
# Needed to publish the binaries to the release.
|
||||||
contents: write
|
contents: write
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: actions/setup-python@v5
|
- uses: actions/setup-python@v6
|
||||||
with:
|
with:
|
||||||
python-version: "3.11"
|
python-version: "3.11"
|
||||||
|
|
||||||
- name: Download Linux aarch64 binary
|
- name: Download Linux aarch64 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-aarch64-linux
|
name: deltachat-rpc-server-aarch64-linux
|
||||||
path: deltachat-rpc-server-aarch64-linux.d
|
path: deltachat-rpc-server-aarch64-linux.d
|
||||||
|
|
||||||
- name: Download Linux armv7l binary
|
- name: Download Linux armv7l binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-armv7l-linux
|
name: deltachat-rpc-server-armv7l-linux
|
||||||
path: deltachat-rpc-server-armv7l-linux.d
|
path: deltachat-rpc-server-armv7l-linux.d
|
||||||
|
|
||||||
- name: Download Linux armv6l binary
|
- name: Download Linux armv6l binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-armv6l-linux
|
name: deltachat-rpc-server-armv6l-linux
|
||||||
path: deltachat-rpc-server-armv6l-linux.d
|
path: deltachat-rpc-server-armv6l-linux.d
|
||||||
|
|
||||||
- name: Download Linux i686 binary
|
- name: Download Linux i686 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-i686-linux
|
name: deltachat-rpc-server-i686-linux
|
||||||
path: deltachat-rpc-server-i686-linux.d
|
path: deltachat-rpc-server-i686-linux.d
|
||||||
|
|
||||||
- name: Download Linux x86_64 binary
|
- name: Download Linux x86_64 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-x86_64-linux
|
name: deltachat-rpc-server-x86_64-linux
|
||||||
path: deltachat-rpc-server-x86_64-linux.d
|
path: deltachat-rpc-server-x86_64-linux.d
|
||||||
|
|
||||||
- name: Download Win32 binary
|
- name: Download Win32 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-win32
|
name: deltachat-rpc-server-win32
|
||||||
path: deltachat-rpc-server-win32.d
|
path: deltachat-rpc-server-win32.d
|
||||||
|
|
||||||
- name: Download Win64 binary
|
- name: Download Win64 binary
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-win64
|
name: deltachat-rpc-server-win64
|
||||||
path: deltachat-rpc-server-win64.d
|
path: deltachat-rpc-server-win64.d
|
||||||
|
|
||||||
- name: Download macOS binary for x86_64
|
- name: Download macOS binary for x86_64
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-x86_64-macos
|
name: deltachat-rpc-server-x86_64-macos
|
||||||
path: deltachat-rpc-server-x86_64-macos.d
|
path: deltachat-rpc-server-x86_64-macos.d
|
||||||
|
|
||||||
- name: Download macOS binary for aarch64
|
- name: Download macOS binary for aarch64
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-aarch64-macos
|
name: deltachat-rpc-server-aarch64-macos
|
||||||
path: deltachat-rpc-server-aarch64-macos.d
|
path: deltachat-rpc-server-aarch64-macos.d
|
||||||
|
|
||||||
- name: Download Android binary for arm64-v8a
|
- name: Download Android binary for arm64-v8a
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-arm64-v8a-android
|
name: deltachat-rpc-server-arm64-v8a-android
|
||||||
path: deltachat-rpc-server-arm64-v8a-android.d
|
path: deltachat-rpc-server-arm64-v8a-android.d
|
||||||
|
|
||||||
- name: Download Android binary for armeabi-v7a
|
- name: Download Android binary for armeabi-v7a
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-armeabi-v7a-android
|
name: deltachat-rpc-server-armeabi-v7a-android
|
||||||
path: deltachat-rpc-server-armeabi-v7a-android.d
|
path: deltachat-rpc-server-armeabi-v7a-android.d
|
||||||
@@ -384,7 +384,7 @@ jobs:
|
|||||||
ls -lah
|
ls -lah
|
||||||
|
|
||||||
- name: Upload to artifacts
|
- name: Upload to artifacts
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: deltachat-rpc-server-npm-package
|
name: deltachat-rpc-server-npm-package
|
||||||
path: deltachat-rpc-server/npm-package/*.tgz
|
path: deltachat-rpc-server/npm-package/*.tgz
|
||||||
@@ -401,7 +401,7 @@ jobs:
|
|||||||
deltachat-rpc-server/npm-package/*.tgz
|
deltachat-rpc-server/npm-package/*.tgz
|
||||||
|
|
||||||
# Configure Node.js for publishing.
|
# Configure Node.js for publishing.
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
registry-url: "https://registry.npmjs.org"
|
registry-url: "https://registry.npmjs.org"
|
||||||
|
|||||||
2
.github/workflows/dependabot.yml
vendored
2
.github/workflows/dependabot.yml
vendored
@@ -14,7 +14,7 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Dependabot metadata
|
- name: Dependabot metadata
|
||||||
id: metadata
|
id: metadata
|
||||||
uses: dependabot/fetch-metadata@v2.3.0
|
uses: dependabot/fetch-metadata@v2.4.0
|
||||||
with:
|
with:
|
||||||
github-token: "${{ secrets.GITHUB_TOKEN }}"
|
github-token: "${{ secrets.GITHUB_TOKEN }}"
|
||||||
- name: Approve a PR
|
- name: Approve a PR
|
||||||
|
|||||||
@@ -9,17 +9,17 @@ permissions: {}
|
|||||||
jobs:
|
jobs:
|
||||||
pack-module:
|
pack-module:
|
||||||
name: "Publish @deltachat/jsonrpc-client"
|
name: "Publish @deltachat/jsonrpc-client"
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-latest
|
||||||
permissions:
|
permissions:
|
||||||
id-token: write
|
id-token: write
|
||||||
contents: read
|
contents: read
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
|
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
registry-url: "https://registry.npmjs.org"
|
registry-url: "https://registry.npmjs.org"
|
||||||
|
|||||||
4
.github/workflows/jsonrpc.yml
vendored
4
.github/workflows/jsonrpc.yml
vendored
@@ -16,12 +16,12 @@ jobs:
|
|||||||
build_and_test:
|
build_and_test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- name: Use Node.js 18.x
|
- name: Use Node.js 18.x
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version: 18.x
|
node-version: 18.x
|
||||||
- name: Add Rust cache
|
- name: Add Rust cache
|
||||||
|
|||||||
27
.github/workflows/nix.yml
vendored
27
.github/workflows/nix.yml
vendored
@@ -5,10 +5,12 @@ on:
|
|||||||
paths:
|
paths:
|
||||||
- flake.nix
|
- flake.nix
|
||||||
- flake.lock
|
- flake.lock
|
||||||
|
- .github/workflows/nix.yml
|
||||||
push:
|
push:
|
||||||
paths:
|
paths:
|
||||||
- flake.nix
|
- flake.nix
|
||||||
- flake.lock
|
- flake.lock
|
||||||
|
- .github/workflows/nix.yml
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
|
|
||||||
@@ -19,15 +21,12 @@ jobs:
|
|||||||
name: check flake formatting
|
name: check flake formatting
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
- run: nix fmt
|
- run: nix fmt flake.nix -- --check
|
||||||
|
|
||||||
# Check that formatting does not change anything.
|
|
||||||
- run: git diff --exit-code
|
|
||||||
|
|
||||||
build:
|
build:
|
||||||
name: nix build
|
name: nix build
|
||||||
@@ -81,11 +80,11 @@ jobs:
|
|||||||
#- deltachat-rpc-server-x86_64-android
|
#- deltachat-rpc-server-x86_64-android
|
||||||
#- deltachat-rpc-server-x86-android
|
#- deltachat-rpc-server-x86-android
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
- run: nix build .#${{ matrix.installable }}
|
- run: nix build .#${{ matrix.installable }}
|
||||||
|
|
||||||
build-macos:
|
build-macos:
|
||||||
@@ -95,14 +94,16 @@ jobs:
|
|||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
installable:
|
installable:
|
||||||
- deltachat-rpc-server-aarch64-darwin
|
- deltachat-rpc-server
|
||||||
|
- deltachat-rpc-server-x86_64-darwin
|
||||||
|
|
||||||
# Fails to bulid
|
# Fails to build
|
||||||
# - deltachat-rpc-server-x86_64-darwin
|
# because of <https://github.com/NixOS/nixpkgs/issues/413910>.
|
||||||
|
# - deltachat-rpc-server-aarch64-darwin
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
- run: nix build .#${{ matrix.installable }}
|
- run: nix build .#${{ matrix.installable }}
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
@@ -23,7 +23,7 @@ jobs:
|
|||||||
working-directory: deltachat-rpc-client
|
working-directory: deltachat-rpc-client
|
||||||
run: python3 -m build
|
run: python3 -m build
|
||||||
- name: Store the distribution packages
|
- name: Store the distribution packages
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: python-package-distributions
|
name: python-package-distributions
|
||||||
path: deltachat-rpc-client/dist/
|
path: deltachat-rpc-client/dist/
|
||||||
@@ -42,7 +42,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Download all the dists
|
- name: Download all the dists
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v6
|
||||||
with:
|
with:
|
||||||
name: python-package-distributions
|
name: python-package-distributions
|
||||||
path: dist/
|
path: dist/
|
||||||
|
|||||||
6
.github/workflows/repl.yml
vendored
6
.github/workflows/repl.yml
vendored
@@ -14,15 +14,15 @@ jobs:
|
|||||||
name: Build REPL example
|
name: Build REPL example
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
- name: Build
|
- name: Build
|
||||||
run: nix build .#deltachat-repl-win64
|
run: nix build .#deltachat-repl-win64
|
||||||
- name: Upload binary
|
- name: Upload binary
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v5
|
||||||
with:
|
with:
|
||||||
name: repl.exe
|
name: repl.exe
|
||||||
path: "result/bin/deltachat-repl.exe"
|
path: "result/bin/deltachat-repl.exe"
|
||||||
|
|||||||
14
.github/workflows/upload-docs.yml
vendored
14
.github/workflows/upload-docs.yml
vendored
@@ -13,7 +13,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
@@ -31,12 +31,12 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
fetch-depth: 0 # Fetch history to calculate VCS version number.
|
fetch-depth: 0 # Fetch history to calculate VCS version number.
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
- name: Build Python documentation
|
- name: Build Python documentation
|
||||||
run: nix build .#python-docs
|
run: nix build .#python-docs
|
||||||
- name: Upload to py.delta.chat
|
- name: Upload to py.delta.chat
|
||||||
@@ -50,12 +50,12 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
fetch-depth: 0 # Fetch history to calculate VCS version number.
|
fetch-depth: 0 # Fetch history to calculate VCS version number.
|
||||||
- uses: DeterminateSystems/nix-installer-action@main
|
- uses: cachix/install-nix-action@fd24c48048070c1be9acd18c9d369a83f0fe94d7 # v31
|
||||||
- name: Build C documentation
|
- name: Build C documentation
|
||||||
run: nix build .#docs
|
run: nix build .#docs
|
||||||
- name: Upload to c.delta.chat
|
- name: Upload to c.delta.chat
|
||||||
@@ -72,13 +72,13 @@ jobs:
|
|||||||
working-directory: ./deltachat-jsonrpc/typescript
|
working-directory: ./deltachat-jsonrpc/typescript
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
fetch-depth: 0 # Fetch history to calculate VCS version number.
|
fetch-depth: 0 # Fetch history to calculate VCS version number.
|
||||||
- name: Use Node.js
|
- name: Use Node.js
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version: '18'
|
node-version: '18'
|
||||||
- name: npm install
|
- name: npm install
|
||||||
|
|||||||
2
.github/workflows/upload-ffi-docs.yml
vendored
2
.github/workflows/upload-ffi-docs.yml
vendored
@@ -16,7 +16,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
show-progress: false
|
show-progress: false
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
|
|||||||
6
.github/workflows/zizmor-scan.yml
vendored
6
.github/workflows/zizmor-scan.yml
vendored
@@ -14,18 +14,18 @@ jobs:
|
|||||||
security-events: write
|
security-events: write
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
|
|
||||||
- name: Install the latest version of uv
|
- name: Install the latest version of uv
|
||||||
uses: astral-sh/setup-uv@v5
|
uses: astral-sh/setup-uv@85856786d1ce8acfbcc2f13a5f3fbd6b938f9f41
|
||||||
|
|
||||||
- name: Run zizmor
|
- name: Run zizmor
|
||||||
run: uvx zizmor --format sarif . > results.sarif
|
run: uvx zizmor --format sarif . > results.sarif
|
||||||
|
|
||||||
- name: Upload SARIF file
|
- name: Upload SARIF file
|
||||||
uses: github/codeql-action/upload-sarif@v3
|
uses: github/codeql-action/upload-sarif@v4
|
||||||
with:
|
with:
|
||||||
sarif_file: results.sarif
|
sarif_file: results.sarif
|
||||||
category: zizmor
|
category: zizmor
|
||||||
|
|||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -36,6 +36,7 @@ deltachat-ffi/xml
|
|||||||
coverage/
|
coverage/
|
||||||
.DS_Store
|
.DS_Store
|
||||||
.vscode
|
.vscode
|
||||||
|
.zed
|
||||||
python/accounts.txt
|
python/accounts.txt
|
||||||
python/all-testaccounts.txt
|
python/all-testaccounts.txt
|
||||||
tmp/
|
tmp/
|
||||||
@@ -53,3 +54,4 @@ result
|
|||||||
# direnv
|
# direnv
|
||||||
.envrc
|
.envrc
|
||||||
.direnv
|
.direnv
|
||||||
|
.aider*
|
||||||
|
|||||||
1060
CHANGELOG.md
1060
CHANGELOG.md
File diff suppressed because it is too large
Load Diff
@@ -44,7 +44,7 @@ If you want to contribute a code, follow this guide.
|
|||||||
|
|
||||||
The following prefix types are used:
|
The following prefix types are used:
|
||||||
- `feat`: Features, e.g. "feat: Pause IO for BackupProvider". If you are unsure what's the category of your commit, you can often just use `feat`.
|
- `feat`: Features, e.g. "feat: Pause IO for BackupProvider". If you are unsure what's the category of your commit, you can often just use `feat`.
|
||||||
- `fix`: Bug fixes, e.g. "fix: delete `smtp` rows when message sending is cancelled"
|
- `fix`: Bug fixes, e.g. "fix: delete `smtp` rows when message sending is canceled"
|
||||||
- `api`: API changes, e.g. "api(rust): add `get_msg_read_receipts(context, msg_id)`"
|
- `api`: API changes, e.g. "api(rust): add `get_msg_read_receipts(context, msg_id)`"
|
||||||
- `refactor`: Refactorings, e.g. "refactor: iterate over `msg_ids` without `.iter()`"
|
- `refactor`: Refactorings, e.g. "refactor: iterate over `msg_ids` without `.iter()`"
|
||||||
- `perf`: Performance improvements, e.g. "perf: improve SQLite performance with `PRAGMA synchronous=normal`"
|
- `perf`: Performance improvements, e.g. "perf: improve SQLite performance with `PRAGMA synchronous=normal`"
|
||||||
|
|||||||
2190
Cargo.lock
generated
2190
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
93
Cargo.toml
93
Cargo.toml
@@ -1,9 +1,9 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "deltachat"
|
name = "deltachat"
|
||||||
version = "1.157.3"
|
version = "2.25.0"
|
||||||
edition = "2021"
|
edition = "2024"
|
||||||
license = "MPL-2.0"
|
license = "MPL-2.0"
|
||||||
rust-version = "1.81"
|
rust-version = "1.85"
|
||||||
repository = "https://github.com/chatmail/core"
|
repository = "https://github.com/chatmail/core"
|
||||||
|
|
||||||
[profile.dev]
|
[profile.dev]
|
||||||
@@ -18,6 +18,9 @@ opt-level = 1
|
|||||||
debug = 1
|
debug = 1
|
||||||
opt-level = 0
|
opt-level = 0
|
||||||
|
|
||||||
|
[profile.fuzz]
|
||||||
|
inherits = "test"
|
||||||
|
|
||||||
# Always optimize dependencies.
|
# Always optimize dependencies.
|
||||||
# This does not apply to crates in the workspace.
|
# This does not apply to crates in the workspace.
|
||||||
# <https://doc.rust-lang.org/cargo/reference/profiles.html#overrides>
|
# <https://doc.rust-lang.org/cargo/reference/profiles.html#overrides>
|
||||||
@@ -41,79 +44,78 @@ ratelimit = { path = "./deltachat-ratelimit" }
|
|||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
async-broadcast = "0.7.2"
|
async-broadcast = "0.7.2"
|
||||||
async-channel = { workspace = true }
|
async-channel = { workspace = true }
|
||||||
async-imap = { version = "0.10.3", default-features = false, features = ["runtime-tokio", "compress"] }
|
async-imap = { version = "0.11.1", default-features = false, features = ["runtime-tokio", "compress"] }
|
||||||
async-native-tls = { version = "0.5", default-features = false, features = ["runtime-tokio"] }
|
async-native-tls = { version = "0.5", default-features = false, features = ["runtime-tokio"] }
|
||||||
async-smtp = { version = "0.10", default-features = false, features = ["runtime-tokio"] }
|
async-smtp = { version = "0.10.2", default-features = false, features = ["runtime-tokio"] }
|
||||||
async_zip = { version = "0.0.17", default-features = false, features = ["deflate", "tokio-fs"] }
|
async_zip = { version = "0.0.18", default-features = false, features = ["deflate", "tokio-fs"] }
|
||||||
base64 = { workspace = true }
|
base64 = { workspace = true }
|
||||||
brotli = { version = "7", default-features=false, features = ["std"] }
|
blake3 = "1.8.2"
|
||||||
|
brotli = { version = "8", default-features=false, features = ["std"] }
|
||||||
bytes = "1"
|
bytes = "1"
|
||||||
chrono = { workspace = true, features = ["alloc", "clock", "std"] }
|
chrono = { workspace = true, features = ["alloc", "clock", "std"] }
|
||||||
data-encoding = "2.7.0"
|
colorutils-rs = { version = "0.7.5", default-features = false }
|
||||||
|
data-encoding = "2.9.0"
|
||||||
escaper = "0.1"
|
escaper = "0.1"
|
||||||
fast-socks5 = "0.10"
|
fast-socks5 = "0.10"
|
||||||
fd-lock = "4"
|
fd-lock = "4"
|
||||||
futures-lite = { workspace = true }
|
futures-lite = { workspace = true }
|
||||||
futures = { workspace = true }
|
futures = { workspace = true }
|
||||||
hex = "0.4.0"
|
hex = "0.4.0"
|
||||||
hickory-resolver = "=0.25.0-alpha.5"
|
http-body-util = "0.1.3"
|
||||||
http-body-util = "0.1.2"
|
|
||||||
humansize = "2"
|
humansize = "2"
|
||||||
hyper = "1"
|
hyper = "1"
|
||||||
hyper-util = "0.1.10"
|
hyper-util = "0.1.16"
|
||||||
image = { version = "0.25.5", default-features=false, features = ["gif", "jpeg", "ico", "png", "pnm", "webp", "bmp"] }
|
image = { version = "0.25.6", default-features=false, features = ["gif", "jpeg", "ico", "png", "pnm", "webp", "bmp"] }
|
||||||
iroh-gossip = { version = "0.33", default-features = false, features = ["net"] }
|
iroh-gossip = { version = "0.35", default-features = false, features = ["net"] }
|
||||||
iroh = { version = "0.33", default-features = false }
|
iroh = { version = "0.35", default-features = false }
|
||||||
kamadak-exif = "0.6.1"
|
kamadak-exif = "0.6.1"
|
||||||
libc = { workspace = true }
|
libc = { workspace = true }
|
||||||
mail-builder = { version = "0.4.2", default-features = false }
|
mail-builder = { version = "0.4.4", default-features = false }
|
||||||
mailparse = { workspace = true }
|
mailparse = { workspace = true }
|
||||||
mime = "0.3.17"
|
mime = "0.3.17"
|
||||||
num_cpus = "1.16"
|
num_cpus = "1.17"
|
||||||
num-derive = "0.4"
|
num-derive = "0.4"
|
||||||
num-traits = { workspace = true }
|
num-traits = { workspace = true }
|
||||||
once_cell = { workspace = true }
|
parking_lot = "0.12.4"
|
||||||
parking_lot = "0.12"
|
|
||||||
percent-encoding = "2.3"
|
percent-encoding = "2.3"
|
||||||
pgp = { version = "0.15.0", default-features = false }
|
pgp = { version = "0.17.0", default-features = false }
|
||||||
pin-project = "1"
|
pin-project = "1"
|
||||||
qrcodegen = "1.7.0"
|
qrcodegen = "1.7.0"
|
||||||
quick-xml = "0.37"
|
quick-xml = { version = "0.38", features = ["escape-html"] }
|
||||||
quoted_printable = "0.5"
|
rand-old = { package = "rand", version = "0.8" }
|
||||||
rand = { workspace = true }
|
rand = { workspace = true }
|
||||||
regex = { workspace = true }
|
regex = { workspace = true }
|
||||||
rusqlite = { workspace = true, features = ["sqlcipher"] }
|
rusqlite = { workspace = true, features = ["sqlcipher"] }
|
||||||
rust-hsluv = "0.1"
|
rustls-pki-types = "1.12.0"
|
||||||
rustls-pki-types = "1.11.0"
|
|
||||||
rustls = { version = "0.23.22", default-features = false }
|
|
||||||
sanitize-filename = { workspace = true }
|
sanitize-filename = { workspace = true }
|
||||||
|
sdp = "0.8.0"
|
||||||
serde_json = { workspace = true }
|
serde_json = { workspace = true }
|
||||||
serde_urlencoded = "0.7.1"
|
serde_urlencoded = "0.7.1"
|
||||||
serde = { workspace = true, features = ["derive"] }
|
serde = { workspace = true, features = ["derive"] }
|
||||||
sha-1 = "0.10"
|
sha-1 = "0.10"
|
||||||
sha2 = "0.10"
|
sha2 = "0.10"
|
||||||
shadowsocks = { version = "1.22.0", default-features = false, features = ["aead-cipher", "aead-cipher-2022"] }
|
shadowsocks = { version = "1.23.1", default-features = false, features = ["aead-cipher", "aead-cipher-2022"] }
|
||||||
smallvec = "1.14.0"
|
smallvec = "1.15.1"
|
||||||
strum = "0.27"
|
strum = "0.27"
|
||||||
strum_macros = "0.27"
|
strum_macros = "0.27"
|
||||||
tagger = "4.3.4"
|
tagger = "4.3.4"
|
||||||
textwrap = "0.16.1"
|
textwrap = "0.16.2"
|
||||||
thiserror = { workspace = true }
|
thiserror = { workspace = true }
|
||||||
tokio-io-timeout = "1.2.0"
|
tokio-io-timeout = "1.2.1"
|
||||||
tokio-rustls = { version = "0.26.2", default-features = false }
|
tokio-rustls = { version = "0.26.2", default-features = false }
|
||||||
tokio-stream = { version = "0.1.17", features = ["fs"] }
|
tokio-stream = { version = "0.1.17", features = ["fs"] }
|
||||||
tokio-tar = { version = "0.3" } # TODO: integrate tokio into async-tar
|
astral-tokio-tar = { version = "0.5.6", default-features = false }
|
||||||
tokio-util = { workspace = true }
|
tokio-util = { workspace = true }
|
||||||
tokio = { workspace = true, features = ["fs", "rt-multi-thread", "macros"] }
|
tokio = { workspace = true, features = ["fs", "rt-multi-thread", "macros"] }
|
||||||
toml = "0.8"
|
toml = "0.9"
|
||||||
|
tracing = "0.1.41"
|
||||||
url = "2"
|
url = "2"
|
||||||
uuid = { version = "1", features = ["serde", "v4"] }
|
uuid = { version = "1", features = ["serde", "v4"] }
|
||||||
webpki-roots = "0.26.8"
|
webpki-roots = "0.26.8"
|
||||||
blake3 = "1.6.1"
|
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
anyhow = { workspace = true, features = ["backtrace"] } # Enable `backtrace` feature in tests.
|
anyhow = { workspace = true, features = ["backtrace"] } # Enable `backtrace` feature in tests.
|
||||||
criterion = { version = "0.5.1", features = ["async_tokio"] }
|
criterion = { version = "0.7.0", features = ["async_tokio"] }
|
||||||
futures-lite = { workspace = true }
|
futures-lite = { workspace = true }
|
||||||
log = { workspace = true }
|
log = { workspace = true }
|
||||||
nu-ansi-term = { workspace = true }
|
nu-ansi-term = { workspace = true }
|
||||||
@@ -154,6 +156,11 @@ name = "receive_emails"
|
|||||||
required-features = ["internals"]
|
required-features = ["internals"]
|
||||||
harness = false
|
harness = false
|
||||||
|
|
||||||
|
[[bench]]
|
||||||
|
name = "decrypting"
|
||||||
|
required-features = ["internals"]
|
||||||
|
harness = false
|
||||||
|
|
||||||
[[bench]]
|
[[bench]]
|
||||||
name = "get_chat_msgs"
|
name = "get_chat_msgs"
|
||||||
harness = false
|
harness = false
|
||||||
@@ -172,38 +179,38 @@ harness = false
|
|||||||
|
|
||||||
[workspace.dependencies]
|
[workspace.dependencies]
|
||||||
anyhow = "1"
|
anyhow = "1"
|
||||||
async-channel = "2.3.1"
|
async-channel = "2.5.0"
|
||||||
base64 = "0.22"
|
base64 = "0.22"
|
||||||
chrono = { version = "0.4.40", default-features = false }
|
chrono = { version = "0.4.42", default-features = false }
|
||||||
deltachat-contact-tools = { path = "deltachat-contact-tools" }
|
deltachat-contact-tools = { path = "deltachat-contact-tools" }
|
||||||
deltachat-jsonrpc = { path = "deltachat-jsonrpc", default-features = false }
|
deltachat-jsonrpc = { path = "deltachat-jsonrpc", default-features = false }
|
||||||
deltachat = { path = ".", default-features = false }
|
deltachat = { path = ".", default-features = false }
|
||||||
futures = "0.3.31"
|
futures = "0.3.31"
|
||||||
futures-lite = "2.6.0"
|
futures-lite = "2.6.1"
|
||||||
libc = "0.2"
|
libc = "0.2"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
mailparse = "0.16.1"
|
mailparse = "0.16.1"
|
||||||
nu-ansi-term = "0.46"
|
nu-ansi-term = "0.50"
|
||||||
num-traits = "0.2"
|
num-traits = "0.2"
|
||||||
once_cell = "1.20.2"
|
rand = "0.9"
|
||||||
rand = "0.8"
|
|
||||||
regex = "1.10"
|
regex = "1.10"
|
||||||
rusqlite = "0.32"
|
rusqlite = "0.36"
|
||||||
sanitize-filename = "0.5"
|
sanitize-filename = "0.5"
|
||||||
serde = "1.0"
|
serde = "1.0"
|
||||||
serde_json = "1"
|
serde_json = "1"
|
||||||
tempfile = "3.14.0"
|
tempfile = "3.23.0"
|
||||||
thiserror = "2"
|
thiserror = "2"
|
||||||
tokio = "1"
|
tokio = "1"
|
||||||
tokio-util = "0.7.13"
|
tokio-util = "0.7.16"
|
||||||
tracing-subscriber = "0.3"
|
tracing-subscriber = "0.3"
|
||||||
yerpc = "0.6.2"
|
yerpc = "0.6.4"
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["vendored"]
|
default = ["vendored"]
|
||||||
internals = []
|
internals = []
|
||||||
vendored = [
|
vendored = [
|
||||||
"rusqlite/bundled-sqlcipher-vendored-openssl"
|
"rusqlite/bundled-sqlcipher-vendored-openssl",
|
||||||
|
"async-native-tls/vendored"
|
||||||
]
|
]
|
||||||
|
|
||||||
[lints.rust]
|
[lints.rust]
|
||||||
|
|||||||
97
README.md
97
README.md
@@ -1,5 +1,5 @@
|
|||||||
<p align="center">
|
<p align="center">
|
||||||
<img alt="Delta Chat Logo" height="200px" src="https://raw.githubusercontent.com/deltachat/deltachat-pages/master/assets/blog/rust-delta.png">
|
<img alt="Chatmail logo" src="https://github.com/user-attachments/assets/25742da7-a837-48cd-a503-b303af55f10d" width="300" style="float:middle;" />
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
@@ -11,9 +11,31 @@
|
|||||||
</a>
|
</a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p align="center">
|
The chatmail core library implements low-level network and encryption protocols,
|
||||||
The core library for Delta Chat, written in Rust
|
integrated by many chat bots and higher level applications,
|
||||||
</p>
|
allowing to securely participate in the globally scaled e-mail server network.
|
||||||
|
We provide reproducibly-built `deltachat-rpc-server` static binaries
|
||||||
|
that offer a stdio-based high-level JSON-RPC API for instant messaging purposes.
|
||||||
|
|
||||||
|
The following protocols are handled without requiring API users to know much about them:
|
||||||
|
|
||||||
|
- secure TLS setup with DNS caching and shadowsocks/proxy support
|
||||||
|
|
||||||
|
- robust [SMTP](https://github.com/chatmail/async-imap)
|
||||||
|
and [IMAP](https://github.com/chatmail/async-smtp) handling
|
||||||
|
|
||||||
|
- safe and interoperable [MIME parsing](https://github.com/staktrace/mailparse)
|
||||||
|
and [MIME building](https://github.com/stalwartlabs/mail-builder).
|
||||||
|
|
||||||
|
- security-audited end-to-end encryption with [rPGP](https://github.com/rpgp/rpgp)
|
||||||
|
and [Autocrypt and SecureJoin protocols](https://securejoin.rtfd.io)
|
||||||
|
|
||||||
|
- ephemeral [Peer-to-Peer networking using Iroh](https://iroh.computer) for multi-device setup and
|
||||||
|
[webxdc realtime data](https://delta.chat/en/2024-11-20-webxdc-realtime).
|
||||||
|
|
||||||
|
- a simulation- and real-world tested [P2P group membership
|
||||||
|
protocol without requiring server state](https://github.com/chatmail/models/tree/main/group-membership).
|
||||||
|
|
||||||
|
|
||||||
## Installing Rust and Cargo
|
## Installing Rust and Cargo
|
||||||
|
|
||||||
@@ -27,12 +49,12 @@ $ curl https://sh.rustup.rs -sSf | sh
|
|||||||
|
|
||||||
## Using the CLI client
|
## Using the CLI client
|
||||||
|
|
||||||
Compile and run Delta Chat Core command line utility, using `cargo`:
|
Compile and run the command line utility, using `cargo`:
|
||||||
|
|
||||||
```
|
```
|
||||||
$ cargo run --locked -p deltachat-repl -- ~/deltachat-db
|
$ cargo run --locked -p deltachat-repl -- ~/profile-db
|
||||||
```
|
```
|
||||||
where ~/deltachat-db is the database file. Delta Chat will create it if it does not exist.
|
where ~/profile-db is the database file. The utility will create it if it does not exist.
|
||||||
|
|
||||||
Optionally, install `deltachat-repl` binary with
|
Optionally, install `deltachat-repl` binary with
|
||||||
```
|
```
|
||||||
@@ -40,13 +62,13 @@ $ cargo install --locked --path deltachat-repl/
|
|||||||
```
|
```
|
||||||
and run as
|
and run as
|
||||||
```
|
```
|
||||||
$ deltachat-repl ~/deltachat-db
|
$ deltachat-repl ~/profile-db
|
||||||
```
|
```
|
||||||
|
|
||||||
Configure your account (if not already configured):
|
Configure your account (if not already configured):
|
||||||
|
|
||||||
```
|
```
|
||||||
Delta Chat Core is awaiting your commands.
|
Chatmail is awaiting your commands.
|
||||||
> set addr your@email.org
|
> set addr your@email.org
|
||||||
> set mail_pw yourpassword
|
> set mail_pw yourpassword
|
||||||
> configure
|
> configure
|
||||||
@@ -58,37 +80,43 @@ Connect to your mail server (if already configured):
|
|||||||
> connect
|
> connect
|
||||||
```
|
```
|
||||||
|
|
||||||
Create a contact:
|
Export your public key to a vCard file:
|
||||||
|
|
||||||
|
```
|
||||||
|
> make-vcard my.vcard 1
|
||||||
|
```
|
||||||
|
|
||||||
|
Create contacts by address or vCard file:
|
||||||
|
|
||||||
```
|
```
|
||||||
> addcontact yourfriends@email.org
|
> addcontact yourfriends@email.org
|
||||||
Command executed successfully.
|
> import-vcard key-contact.vcard
|
||||||
```
|
```
|
||||||
|
|
||||||
List contacts:
|
List contacts:
|
||||||
|
|
||||||
```
|
```
|
||||||
> listcontacts
|
> listcontacts
|
||||||
Contact#10: <name unset> <yourfriends@email.org>
|
Contact#Contact#11: key-contact@email.org <key-contact@email.org>
|
||||||
Contact#1: Me √√ <your@email.org>
|
Contact#Contact#Self: Me √ <your@email.org>
|
||||||
|
2 key contacts.
|
||||||
|
Contact#Contact#10: yourfriends@email.org <yourfriends@email.org>
|
||||||
|
1 address contacts.
|
||||||
```
|
```
|
||||||
|
|
||||||
Create a chat with your friend and send a message:
|
Create a chat with your friend and send a message:
|
||||||
|
|
||||||
```
|
```
|
||||||
> createchat 10
|
> createchat 10
|
||||||
Single#10 created successfully.
|
Single#Chat#12 created successfully.
|
||||||
> chat 10
|
> chat 12
|
||||||
Single#10: yourfriends@email.org [yourfriends@email.org]
|
Selecting chat Chat#12
|
||||||
|
Single#Chat#12: yourfriends@email.org [yourfriends@email.org] Icon: profile-db-blobs/4138c52e5bc1c576cda7dd44d088c07.png
|
||||||
|
0 messages.
|
||||||
|
81.252µs to create this list, 123.625µs to mark all messages as noticed.
|
||||||
> send hi
|
> send hi
|
||||||
Message sent.
|
|
||||||
```
|
```
|
||||||
|
|
||||||
If `yourfriend@email.org` uses DeltaChat, but does not receive message just
|
|
||||||
sent, it is advisable to check `Spam` folder. It is known that at least
|
|
||||||
`gmx.com` treat such test messages as spam, unless told otherwise with web
|
|
||||||
interface.
|
|
||||||
|
|
||||||
List messages when inside a chat:
|
List messages when inside a chat:
|
||||||
|
|
||||||
```
|
```
|
||||||
@@ -139,13 +167,13 @@ $ cargo test -- --ignored
|
|||||||
|
|
||||||
Install [`cargo-bolero`](https://github.com/camshaft/bolero) with
|
Install [`cargo-bolero`](https://github.com/camshaft/bolero) with
|
||||||
```sh
|
```sh
|
||||||
$ cargo install cargo-bolero@0.8.0
|
$ cargo install cargo-bolero
|
||||||
```
|
```
|
||||||
|
|
||||||
Run fuzzing tests with
|
Run fuzzing tests with
|
||||||
```sh
|
```sh
|
||||||
$ cd fuzz
|
$ cd fuzz
|
||||||
$ cargo bolero test fuzz_mailparse --release=false -s NONE
|
$ cargo bolero test fuzz_mailparse -s NONE
|
||||||
```
|
```
|
||||||
|
|
||||||
Corpus is created at `fuzz/fuzz_targets/corpus`,
|
Corpus is created at `fuzz/fuzz_targets/corpus`,
|
||||||
@@ -153,11 +181,6 @@ you can add initial inputs there.
|
|||||||
For `fuzz_mailparse` target corpus can be populated with
|
For `fuzz_mailparse` target corpus can be populated with
|
||||||
`../test-data/message/*.eml`.
|
`../test-data/message/*.eml`.
|
||||||
|
|
||||||
To run with AFL instead of libFuzzer:
|
|
||||||
```sh
|
|
||||||
$ cargo bolero test fuzz_format_flowed --release=false -e afl -s NONE
|
|
||||||
```
|
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- `vendored`: When using Openssl for TLS, this bundles a vendored version.
|
- `vendored`: When using Openssl for TLS, this bundles a vendored version.
|
||||||
@@ -165,23 +188,19 @@ $ cargo bolero test fuzz_format_flowed --release=false -e afl -s NONE
|
|||||||
## Update Provider Data
|
## Update Provider Data
|
||||||
|
|
||||||
To add the updates from the
|
To add the updates from the
|
||||||
[provider-db](https://github.com/deltachat/provider-db) to the core, run:
|
[provider-db](https://github.com/chatmail/provider-db) to the core,
|
||||||
|
check line `REV=` inside `./scripts/update-provider-database.sh`
|
||||||
```
|
and then run the script.
|
||||||
./src/provider/update.py ../provider-db/_providers/ > src/provider/data.rs
|
|
||||||
```
|
|
||||||
|
|
||||||
## Language bindings and frontend projects
|
## Language bindings and frontend projects
|
||||||
|
|
||||||
Language bindings are available for:
|
Language bindings are available for:
|
||||||
|
|
||||||
- **C** \[[📂 source](./deltachat-ffi) | [📚 docs](https://c.delta.chat)\]
|
- **C** \[[📂 source](./deltachat-ffi) | [📚 docs](https://c.delta.chat)\]
|
||||||
|
- -> libdeltachat is going to be deprecated and only exists because Android, iOS and Ubuntu Touch are still using it. If you build a new project, then please use the jsonrpc api instead.
|
||||||
- **JS**: \[[📂 source](./deltachat-rpc-client) | [📦 npm](https://www.npmjs.com/package/@deltachat/jsonrpc-client) | [📚 docs](https://js.jsonrpc.delta.chat/)\]
|
- **JS**: \[[📂 source](./deltachat-rpc-client) | [📦 npm](https://www.npmjs.com/package/@deltachat/jsonrpc-client) | [📚 docs](https://js.jsonrpc.delta.chat/)\]
|
||||||
- **Python** \[[📂 source](./python) | [📦 pypi](https://pypi.org/project/deltachat) | [📚 docs](https://py.delta.chat)\]
|
- **Python** \[[📂 source](./python) | [📦 pypi](https://pypi.org/project/deltachat) | [📚 docs](https://py.delta.chat)\]
|
||||||
- **Go**
|
- **Go** \[[📂 source](https://github.com/deltachat/deltachat-rpc-client-go/)\]
|
||||||
- over jsonrpc: \[[📂 source](https://github.com/deltachat/deltachat-rpc-client-go/)\]
|
|
||||||
- over cffi[^1]: \[[📂 source](https://github.com/deltachat/go-deltachat/)\]
|
|
||||||
- **Free Pascal**[^1] \[[📂 source](https://github.com/deltachat/deltachat-fp/)\]
|
|
||||||
- **Java** and **Swift** (contained in the Android/iOS repos)
|
- **Java** and **Swift** (contained in the Android/iOS repos)
|
||||||
|
|
||||||
The following "frontend" projects make use of the Rust-library
|
The following "frontend" projects make use of the Rust-library
|
||||||
@@ -194,5 +213,3 @@ or its language bindings:
|
|||||||
- [Telepathy](https://code.ur.gs/lupine/telepathy-padfoot/)
|
- [Telepathy](https://code.ur.gs/lupine/telepathy-padfoot/)
|
||||||
- [Ubuntu Touch](https://codeberg.org/lk108/deltatouch)
|
- [Ubuntu Touch](https://codeberg.org/lk108/deltatouch)
|
||||||
- several **Bots**
|
- several **Bots**
|
||||||
|
|
||||||
[^1]: Out of date / unmaintained, if you like those languages feel free to start maintaining them. If you have questions we'll help you, please ask in the issues.
|
|
||||||
|
|||||||
54
STYLE.md
54
STYLE.md
@@ -78,12 +78,52 @@ All errors should be handled in one of these ways:
|
|||||||
- With `.log_err().ok()`.
|
- With `.log_err().ok()`.
|
||||||
- Bubbled up with `?`.
|
- Bubbled up with `?`.
|
||||||
|
|
||||||
|
When working with [async streams](https://docs.rs/futures/0.3.31/futures/stream/index.html),
|
||||||
|
prefer [`try_next`](https://docs.rs/futures/0.3.31/futures/stream/trait.TryStreamExt.html#method.try_next) over `next()`, e.g. do
|
||||||
|
```
|
||||||
|
while let Some(event) = stream.try_next().await? {
|
||||||
|
todo!();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
instead of
|
||||||
|
```
|
||||||
|
while let Some(event_res) = stream.next().await {
|
||||||
|
todo!();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
as it allows bubbling up the error early with `?`
|
||||||
|
with no way to accidentally skip error processing
|
||||||
|
with early `continue` or `break`.
|
||||||
|
Some streams reading from a connection
|
||||||
|
return infinite number of `Some(Err(_))`
|
||||||
|
items when connection breaks and not processing
|
||||||
|
errors may result in infinite loop.
|
||||||
|
|
||||||
`backtrace` feature is enabled for `anyhow` crate
|
`backtrace` feature is enabled for `anyhow` crate
|
||||||
and `debug = 1` option is set in the test profile.
|
and `debug = 1` option is set in the test profile.
|
||||||
This allows to run `RUST_BACKTRACE=1 cargo test`
|
This allows to run `RUST_BACKTRACE=1 cargo test`
|
||||||
and get a backtrace with line numbers in resultified tests
|
and get a backtrace with line numbers in resultified tests
|
||||||
which return `anyhow::Result`.
|
which return `anyhow::Result`.
|
||||||
|
|
||||||
|
`unwrap` and `expect` are not used in the library
|
||||||
|
because panics are difficult to debug on user devices.
|
||||||
|
However, in the tests `.expect` may be used.
|
||||||
|
Follow
|
||||||
|
<https://doc.rust-lang.org/core/error/index.html#common-message-styles>
|
||||||
|
for `.expect` message style.
|
||||||
|
|
||||||
|
## BTreeMap vs HashMap
|
||||||
|
|
||||||
|
Prefer [BTreeMap](https://doc.rust-lang.org/std/collections/struct.BTreeMap.html)
|
||||||
|
over [HashMap](https://doc.rust-lang.org/std/collections/struct.HashMap.html)
|
||||||
|
and [BTreeSet](https://doc.rust-lang.org/std/collections/struct.BTreeSet.html)
|
||||||
|
over [HashSet](https://doc.rust-lang.org/std/collections/struct.HashSet.html)
|
||||||
|
as iterating over these structures returns items in deterministic order.
|
||||||
|
|
||||||
|
Non-deterministic code may result in difficult to reproduce bugs,
|
||||||
|
flaky tests, regression tests that miss bugs
|
||||||
|
or different behavior on different devices when processing the same messages.
|
||||||
|
|
||||||
## Logging
|
## Logging
|
||||||
|
|
||||||
For logging, use `info!`, `warn!` and `error!` macros.
|
For logging, use `info!`, `warn!` and `error!` macros.
|
||||||
@@ -96,3 +136,17 @@ Format anyhow errors with `{:#}` to print all the contexts like this:
|
|||||||
```
|
```
|
||||||
error!(context, "Failed to set selfavatar timestamp: {err:#}.");
|
error!(context, "Failed to set selfavatar timestamp: {err:#}.");
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Documentation comments
|
||||||
|
|
||||||
|
All public modules, methods and fields should be documented.
|
||||||
|
This is checked by [`missing_docs`](https://doc.rust-lang.org/rustdoc/lints.html#missing_docs) lint.
|
||||||
|
|
||||||
|
Private items do not have to be documented,
|
||||||
|
but CI uses `cargo doc --document-private-items`
|
||||||
|
to build the documentation,
|
||||||
|
so it is preferred that new items
|
||||||
|
are documented.
|
||||||
|
|
||||||
|
Follow Rust guidelines for the documentation comments:
|
||||||
|
<https://rust-lang.github.io/rfcs/1574-more-api-documentation-conventions.html#summary-sentence>
|
||||||
|
|||||||
BIN
assets/icon-unencrypted.png
Normal file
BIN
assets/icon-unencrypted.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 2.2 KiB |
47
assets/icon-unencrypted.svg
Normal file
47
assets/icon-unencrypted.svg
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<svg
|
||||||
|
height="480"
|
||||||
|
viewBox="0 -960 9600 9600"
|
||||||
|
width="480"
|
||||||
|
fill="#ffffff"
|
||||||
|
version="1.1"
|
||||||
|
id="svg1"
|
||||||
|
sodipodi:docname="icon-email.svg"
|
||||||
|
inkscape:version="1.4.2 (ebf0e940d0, 2025-05-08)"
|
||||||
|
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||||
|
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg">
|
||||||
|
<defs
|
||||||
|
id="defs1" />
|
||||||
|
<sodipodi:namedview
|
||||||
|
id="namedview1"
|
||||||
|
pagecolor="#ffffff"
|
||||||
|
bordercolor="#666666"
|
||||||
|
borderopacity="1.0"
|
||||||
|
inkscape:showpageshadow="2"
|
||||||
|
inkscape:pageopacity="0.0"
|
||||||
|
inkscape:pagecheckerboard="0"
|
||||||
|
inkscape:deskcolor="#d1d1d1"
|
||||||
|
inkscape:zoom="0.99091847"
|
||||||
|
inkscape:cx="263.392"
|
||||||
|
inkscape:cy="177.613"
|
||||||
|
inkscape:window-width="1884"
|
||||||
|
inkscape:window-height="1052"
|
||||||
|
inkscape:window-x="36"
|
||||||
|
inkscape:window-y="0"
|
||||||
|
inkscape:window-maximized="1"
|
||||||
|
inkscape:current-layer="svg1" />
|
||||||
|
<rect
|
||||||
|
style="fill:#8c8c8c;fill-opacity:1;stroke:none;stroke-width:680.523;stroke-dasharray:none;paint-order:markers fill stroke"
|
||||||
|
id="rect1"
|
||||||
|
width="9951.9541"
|
||||||
|
height="9767.4756"
|
||||||
|
x="-71.697792"
|
||||||
|
y="-1012.83"
|
||||||
|
ry="0.43547946" />
|
||||||
|
<path
|
||||||
|
d="m 2948.0033,5553.6941 q -130.7292,0 -228.7761,-96.3953 -98.0468,-96.3953 -98.0468,-224.9223 V 2447.6234 q 0,-128.527 98.0468,-224.9223 98.0469,-96.3953 228.7761,-96.3953 h 3703.9934 q 130.7292,0 228.776,96.3953 98.0469,96.3953 98.0469,224.9223 v 2784.7531 q 0,128.527 -98.0469,224.9223 -98.0468,96.3953 -228.776,96.3953 z M 4800,3936.3952 2948.0033,2742.1646 V 5232.3765 H 6651.9967 V 2742.1646 Z m 0,-321.3176 1830.2085,-1167.4541 h -3654.97 z m -1851.9967,-872.913 v -294.5412 2784.7531 z"
|
||||||
|
id="path1"
|
||||||
|
style="stroke-width:5.40098" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.8 KiB |
7
assets/statistics-bot.vcf
Normal file
7
assets/statistics-bot.vcf
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
BEGIN:VCARD
|
||||||
|
VERSION:4.0
|
||||||
|
EMAIL:self_reporting@testrun.org
|
||||||
|
FN:Statistics bot
|
||||||
|
KEY:data:application/pgp-keys;base64,xjMEZbfBlBYJKwYBBAHaRw8BAQdABpLWS2PUIGGo4pslVt4R8sylP5wZihmhf1DTDr3oCMPNHDxzZWxmX3JlcG9ydGluZ0B0ZXN0cnVuLm9yZz7CiwQQFggAMwIZAQUCZbfBlAIbAwQLCQgHBhUICQoLAgMWAgEWIQTS2i16sHeYTckGn284K3M5Z4oohAAKCRA4K3M5Z4oohD8dAQCQV7CoH6UP4PD+NqI4kW5tbbqdh2AnDROg60qotmLExAEAxDfd3QHAK9f8b9qQUbLmHIztCLxhEuVbWPBEYeVW0gvOOARlt8GUEgorBgEEAZdVAQUBAQdAMBUhYoAAcI625vGZqnM5maPX4sGJ7qvJxPAFILPy6AcDAQgHwngEGBYIACAFAmW3wZQCGwwWIQTS2i16sHeYTckGn284K3M5Z4oohAAKCRA4K3M5Z4oohPwCAQCvzk1ObIkj2GqsuIfaULlgdnfdZY8LNary425CEfHZDQD5AblXVrlMO1frdlc/Vo9z3pEeCrfYdD7ITD3/OeVoiQ4=
|
||||||
|
REV:20250412T195751Z
|
||||||
|
END:VCARD
|
||||||
@@ -1,9 +1,11 @@
|
|||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
use criterion::{black_box, criterion_group, criterion_main, Criterion};
|
use std::hint::black_box;
|
||||||
|
|
||||||
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
use deltachat::Events;
|
||||||
use deltachat::contact::Contact;
|
use deltachat::contact::Contact;
|
||||||
use deltachat::context::Context;
|
use deltachat::context::Context;
|
||||||
use deltachat::stock_str::StockStrings;
|
use deltachat::stock_str::StockStrings;
|
||||||
use deltachat::Events;
|
|
||||||
use tempfile::tempdir;
|
use tempfile::tempdir;
|
||||||
|
|
||||||
async fn address_book_benchmark(n: u32, read_count: u32) {
|
async fn address_book_benchmark(n: u32, read_count: u32) {
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
|
use std::hint::black_box;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use criterion::{black_box, criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
use deltachat::accounts::Accounts;
|
use deltachat::accounts::Accounts;
|
||||||
use tempfile::tempdir;
|
use tempfile::tempdir;
|
||||||
|
|
||||||
|
|||||||
200
benches/decrypting.rs
Normal file
200
benches/decrypting.rs
Normal file
@@ -0,0 +1,200 @@
|
|||||||
|
//! Benchmarks for message decryption,
|
||||||
|
//! comparing decryption of symmetrically-encrypted messages
|
||||||
|
//! to decryption of asymmetrically-encrypted messages.
|
||||||
|
//!
|
||||||
|
//! Call with
|
||||||
|
//!
|
||||||
|
//! ```text
|
||||||
|
//! cargo bench --bench decrypting --features="internals"
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! or, if you want to only run e.g. the 'Decrypt a symmetrically encrypted message' benchmark:
|
||||||
|
//!
|
||||||
|
//! ```text
|
||||||
|
//! cargo bench --bench decrypting --features="internals" -- 'Decrypt a symmetrically encrypted message'
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! You can also pass a substring.
|
||||||
|
//! So, you can run all 'Decrypt and parse' benchmarks with:
|
||||||
|
//!
|
||||||
|
//! ```text
|
||||||
|
//! cargo bench --bench decrypting --features="internals" -- 'Decrypt and parse'
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! Symmetric decryption has to try out all known secrets,
|
||||||
|
//! You can benchmark this by adapting the `NUM_SECRETS` variable.
|
||||||
|
|
||||||
|
use std::hint::black_box;
|
||||||
|
|
||||||
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
use deltachat::internals_for_benches::create_broadcast_secret;
|
||||||
|
use deltachat::internals_for_benches::create_dummy_keypair;
|
||||||
|
use deltachat::internals_for_benches::save_broadcast_secret;
|
||||||
|
use deltachat::{
|
||||||
|
Events,
|
||||||
|
chat::ChatId,
|
||||||
|
config::Config,
|
||||||
|
context::Context,
|
||||||
|
internals_for_benches::key_from_asc,
|
||||||
|
internals_for_benches::parse_and_get_text,
|
||||||
|
internals_for_benches::store_self_keypair,
|
||||||
|
pgp::{KeyPair, decrypt, pk_encrypt, symm_encrypt_message},
|
||||||
|
stock_str::StockStrings,
|
||||||
|
};
|
||||||
|
use rand::{Rng, rng};
|
||||||
|
use tempfile::tempdir;
|
||||||
|
|
||||||
|
const NUM_SECRETS: usize = 500;
|
||||||
|
|
||||||
|
async fn create_context() -> Context {
|
||||||
|
let dir = tempdir().unwrap();
|
||||||
|
let dbfile = dir.path().join("db.sqlite");
|
||||||
|
let context = Context::new(dbfile.as_path(), 100, Events::new(), StockStrings::new())
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
context
|
||||||
|
.set_config(Config::ConfiguredAddr, Some("bob@example.net"))
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
let secret = key_from_asc(include_str!("../test-data/key/bob-secret.asc")).unwrap();
|
||||||
|
let public = secret.signed_public_key();
|
||||||
|
let key_pair = KeyPair { public, secret };
|
||||||
|
store_self_keypair(&context, &key_pair)
|
||||||
|
.await
|
||||||
|
.expect("Failed to save key");
|
||||||
|
|
||||||
|
context
|
||||||
|
}
|
||||||
|
|
||||||
|
fn criterion_benchmark(c: &mut Criterion) {
|
||||||
|
let mut group = c.benchmark_group("Decrypt");
|
||||||
|
|
||||||
|
// ===========================================================================================
|
||||||
|
// Benchmarks for decryption only, without any other parsing
|
||||||
|
// ===========================================================================================
|
||||||
|
|
||||||
|
group.sample_size(10);
|
||||||
|
|
||||||
|
group.bench_function("Decrypt a symmetrically encrypted message", |b| {
|
||||||
|
let plain = generate_plaintext();
|
||||||
|
let secrets = generate_secrets();
|
||||||
|
let encrypted = tokio::runtime::Runtime::new().unwrap().block_on(async {
|
||||||
|
let secret = secrets[NUM_SECRETS / 2].clone();
|
||||||
|
symm_encrypt_message(
|
||||||
|
plain.clone(),
|
||||||
|
create_dummy_keypair("alice@example.org").unwrap().secret,
|
||||||
|
black_box(&secret),
|
||||||
|
true,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
.unwrap()
|
||||||
|
});
|
||||||
|
|
||||||
|
b.iter(|| {
|
||||||
|
let mut msg =
|
||||||
|
decrypt(encrypted.clone().into_bytes(), &[], black_box(&secrets)).unwrap();
|
||||||
|
let decrypted = msg.as_data_vec().unwrap();
|
||||||
|
|
||||||
|
assert_eq!(black_box(decrypted), plain);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
group.bench_function("Decrypt a public-key encrypted message", |b| {
|
||||||
|
let plain = generate_plaintext();
|
||||||
|
let key_pair = create_dummy_keypair("alice@example.org").unwrap();
|
||||||
|
let secrets = generate_secrets();
|
||||||
|
let encrypted = tokio::runtime::Runtime::new().unwrap().block_on(async {
|
||||||
|
pk_encrypt(
|
||||||
|
plain.clone(),
|
||||||
|
vec![black_box(key_pair.public.clone())],
|
||||||
|
Some(key_pair.secret.clone()),
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
.unwrap()
|
||||||
|
});
|
||||||
|
|
||||||
|
b.iter(|| {
|
||||||
|
let mut msg = decrypt(
|
||||||
|
encrypted.clone().into_bytes(),
|
||||||
|
std::slice::from_ref(&key_pair.secret),
|
||||||
|
black_box(&secrets),
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
let decrypted = msg.as_data_vec().unwrap();
|
||||||
|
|
||||||
|
assert_eq!(black_box(decrypted), plain);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// ===========================================================================================
|
||||||
|
// Benchmarks for the whole parsing pipeline, incl. decryption (but excl. receive_imf())
|
||||||
|
// ===========================================================================================
|
||||||
|
|
||||||
|
let rt = tokio::runtime::Runtime::new().unwrap();
|
||||||
|
let mut secrets = generate_secrets();
|
||||||
|
|
||||||
|
// "secret" is the shared secret that was used to encrypt text_symmetrically_encrypted.eml.
|
||||||
|
// Put it into the middle of our secrets:
|
||||||
|
secrets[NUM_SECRETS / 2] = "secret".to_string();
|
||||||
|
|
||||||
|
let context = rt.block_on(async {
|
||||||
|
let context = create_context().await;
|
||||||
|
for (i, secret) in secrets.iter().enumerate() {
|
||||||
|
save_broadcast_secret(&context, ChatId::new(10 + i as u32), secret)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
context
|
||||||
|
});
|
||||||
|
|
||||||
|
group.bench_function("Decrypt and parse a symmetrically encrypted message", |b| {
|
||||||
|
b.to_async(&rt).iter(|| {
|
||||||
|
let ctx = context.clone();
|
||||||
|
async move {
|
||||||
|
let text = parse_and_get_text(
|
||||||
|
&ctx,
|
||||||
|
include_bytes!("../test-data/message/text_symmetrically_encrypted.eml"),
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(text, "Symmetrically encrypted message");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
group.bench_function("Decrypt and parse a public-key encrypted message", |b| {
|
||||||
|
b.to_async(&rt).iter(|| {
|
||||||
|
let ctx = context.clone();
|
||||||
|
async move {
|
||||||
|
let text = parse_and_get_text(
|
||||||
|
&ctx,
|
||||||
|
include_bytes!("../test-data/message/text_from_alice_encrypted.eml"),
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(text, "hi");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
group.finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn generate_secrets() -> Vec<String> {
|
||||||
|
let secrets: Vec<String> = (0..NUM_SECRETS)
|
||||||
|
.map(|_| create_broadcast_secret())
|
||||||
|
.collect();
|
||||||
|
secrets
|
||||||
|
}
|
||||||
|
|
||||||
|
fn generate_plaintext() -> Vec<u8> {
|
||||||
|
let mut plain: Vec<u8> = vec![0; 500];
|
||||||
|
rng().fill(&mut plain[..]);
|
||||||
|
plain
|
||||||
|
}
|
||||||
|
|
||||||
|
criterion_group!(benches, criterion_benchmark);
|
||||||
|
criterion_main!(benches);
|
||||||
@@ -1,12 +1,13 @@
|
|||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
|
use std::hint::black_box;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
use criterion::{black_box, criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
use deltachat::Events;
|
||||||
use deltachat::chat::{self, ChatId};
|
use deltachat::chat::{self, ChatId};
|
||||||
use deltachat::chatlist::Chatlist;
|
use deltachat::chatlist::Chatlist;
|
||||||
use deltachat::context::Context;
|
use deltachat::context::Context;
|
||||||
use deltachat::stock_str::StockStrings;
|
use deltachat::stock_str::StockStrings;
|
||||||
use deltachat::Events;
|
|
||||||
|
|
||||||
async fn get_chat_msgs_benchmark(dbfile: &Path, chats: &[ChatId]) {
|
async fn get_chat_msgs_benchmark(dbfile: &Path, chats: &[ChatId]) {
|
||||||
let id = 100;
|
let id = 100;
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
|
use std::hint::black_box;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
use criterion::{black_box, criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
use deltachat::Events;
|
||||||
use deltachat::chatlist::Chatlist;
|
use deltachat::chatlist::Chatlist;
|
||||||
use deltachat::context::Context;
|
use deltachat::context::Context;
|
||||||
use deltachat::stock_str::StockStrings;
|
use deltachat::stock_str::StockStrings;
|
||||||
use deltachat::Events;
|
|
||||||
|
|
||||||
async fn get_chat_list_benchmark(context: &Context) {
|
async fn get_chat_list_benchmark(context: &Context) {
|
||||||
Chatlist::try_load(context, 0, None, None).await.unwrap();
|
Chatlist::try_load(context, 0, None, None).await.unwrap();
|
||||||
|
|||||||
@@ -1,12 +1,13 @@
|
|||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
|
use std::hint::black_box;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
use criterion::{black_box, criterion_group, criterion_main, BatchSize, Criterion};
|
use criterion::{BatchSize, Criterion, criterion_group, criterion_main};
|
||||||
|
use deltachat::Events;
|
||||||
use deltachat::chat::{self, ChatId};
|
use deltachat::chat::{self, ChatId};
|
||||||
use deltachat::chatlist::Chatlist;
|
use deltachat::chatlist::Chatlist;
|
||||||
use deltachat::context::Context;
|
use deltachat::context::Context;
|
||||||
use deltachat::stock_str::StockStrings;
|
use deltachat::stock_str::StockStrings;
|
||||||
use deltachat::Events;
|
|
||||||
use futures_lite::future::block_on;
|
use futures_lite::future::block_on;
|
||||||
use tempfile::tempdir;
|
use tempfile::tempdir;
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,15 @@
|
|||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
|
use std::hint::black_box;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use criterion::{black_box, criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
use deltachat::{
|
use deltachat::{
|
||||||
|
Events,
|
||||||
config::Config,
|
config::Config,
|
||||||
context::Context,
|
context::Context,
|
||||||
imex::{imex, ImexMode},
|
imex::{ImexMode, imex},
|
||||||
receive_imf::receive_imf,
|
receive_imf::receive_imf,
|
||||||
stock_str::StockStrings,
|
stock_str::StockStrings,
|
||||||
Events,
|
|
||||||
};
|
};
|
||||||
use tempfile::tempdir;
|
use tempfile::tempdir;
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
|
use std::hint::black_box;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
use criterion::{black_box, criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
use deltachat::Events;
|
||||||
use deltachat::context::Context;
|
use deltachat::context::Context;
|
||||||
use deltachat::stock_str::StockStrings;
|
use deltachat::stock_str::StockStrings;
|
||||||
use deltachat::Events;
|
|
||||||
|
|
||||||
async fn search_benchmark(dbfile: impl AsRef<Path>) {
|
async fn search_benchmark(dbfile: impl AsRef<Path>) {
|
||||||
let id = 100;
|
let id = 100;
|
||||||
|
|||||||
@@ -1,17 +1,17 @@
|
|||||||
#![recursion_limit = "256"]
|
#![recursion_limit = "256"]
|
||||||
use criterion::{criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
|
||||||
use deltachat::context::Context;
|
use deltachat::context::Context;
|
||||||
use deltachat::stock_str::StockStrings;
|
use deltachat::stock_str::StockStrings;
|
||||||
use deltachat::{info, Event, EventType, Events};
|
use deltachat::{Event, EventType, Events};
|
||||||
use tempfile::tempdir;
|
use tempfile::tempdir;
|
||||||
|
|
||||||
async fn send_events_benchmark(context: &Context) {
|
async fn send_events_benchmark(context: &Context) {
|
||||||
let emitter = context.get_event_emitter();
|
let emitter = context.get_event_emitter();
|
||||||
for _i in 0..1_000_000 {
|
for _i in 0..1_000_000 {
|
||||||
info!(context, "interesting event...");
|
context.emit_event(EventType::Info("interesting event...".to_string()));
|
||||||
}
|
}
|
||||||
info!(context, "DONE");
|
context.emit_event(EventType::Info("DONE".to_string()));
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
match emitter.recv().await.unwrap() {
|
match emitter.recv().await.unwrap() {
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ license = "MPL-2.0"
|
|||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
once_cell = { workspace = true }
|
|
||||||
regex = { workspace = true }
|
regex = { workspace = true }
|
||||||
rusqlite = { workspace = true } # Needed in order to `impl rusqlite::types::ToSql for EmailAddress`. Could easily be put behind a feature.
|
rusqlite = { workspace = true } # Needed in order to `impl rusqlite::types::ToSql for EmailAddress`. Could easily be put behind a feature.
|
||||||
chrono = { workspace = true, features = ["alloc", "clock", "std"] }
|
chrono = { workspace = true, features = ["alloc", "clock", "std"] }
|
||||||
|
|||||||
@@ -29,202 +29,14 @@
|
|||||||
|
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
use std::ops::Deref;
|
use std::ops::Deref;
|
||||||
|
use std::sync::LazyLock;
|
||||||
|
|
||||||
use anyhow::bail;
|
use anyhow::bail;
|
||||||
use anyhow::Context as _;
|
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use chrono::{DateTime, NaiveDateTime};
|
|
||||||
use once_cell::sync::Lazy;
|
|
||||||
use regex::Regex;
|
use regex::Regex;
|
||||||
|
|
||||||
#[derive(Debug)]
|
mod vcard;
|
||||||
/// A Contact, as represented in a VCard.
|
pub use vcard::{make_vcard, parse_vcard, VcardContact};
|
||||||
pub struct VcardContact {
|
|
||||||
/// The email address, vcard property `email`
|
|
||||||
pub addr: String,
|
|
||||||
/// This must be the name authorized by the contact itself, not a locally given name. Vcard
|
|
||||||
/// property `fn`. Can be empty, one should use `display_name()` to obtain the display name.
|
|
||||||
pub authname: String,
|
|
||||||
/// The contact's public PGP key in Base64, vcard property `key`
|
|
||||||
pub key: Option<String>,
|
|
||||||
/// The contact's profile image (=avatar) in Base64, vcard property `photo`
|
|
||||||
pub profile_image: Option<String>,
|
|
||||||
/// The timestamp when the vcard was created / last updated, vcard property `rev`
|
|
||||||
pub timestamp: Result<i64>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl VcardContact {
|
|
||||||
/// Returns the contact's display name.
|
|
||||||
pub fn display_name(&self) -> &str {
|
|
||||||
match self.authname.is_empty() {
|
|
||||||
false => &self.authname,
|
|
||||||
true => &self.addr,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a vCard containing given contacts.
|
|
||||||
///
|
|
||||||
/// Calling [`parse_vcard()`] on the returned result is a reverse operation.
|
|
||||||
pub fn make_vcard(contacts: &[VcardContact]) -> String {
|
|
||||||
fn format_timestamp(c: &VcardContact) -> Option<String> {
|
|
||||||
let timestamp = *c.timestamp.as_ref().ok()?;
|
|
||||||
let datetime = DateTime::from_timestamp(timestamp, 0)?;
|
|
||||||
Some(datetime.format("%Y%m%dT%H%M%SZ").to_string())
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut res = "".to_string();
|
|
||||||
for c in contacts {
|
|
||||||
let addr = &c.addr;
|
|
||||||
let display_name = c.display_name();
|
|
||||||
res += &format!(
|
|
||||||
"BEGIN:VCARD\r\n\
|
|
||||||
VERSION:4.0\r\n\
|
|
||||||
EMAIL:{addr}\r\n\
|
|
||||||
FN:{display_name}\r\n"
|
|
||||||
);
|
|
||||||
if let Some(key) = &c.key {
|
|
||||||
res += &format!("KEY:data:application/pgp-keys;base64,{key}\r\n");
|
|
||||||
}
|
|
||||||
if let Some(profile_image) = &c.profile_image {
|
|
||||||
res += &format!("PHOTO:data:image/jpeg;base64,{profile_image}\r\n");
|
|
||||||
}
|
|
||||||
if let Some(timestamp) = format_timestamp(c) {
|
|
||||||
res += &format!("REV:{timestamp}\r\n");
|
|
||||||
}
|
|
||||||
res += "END:VCARD\r\n";
|
|
||||||
}
|
|
||||||
res
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Parses `VcardContact`s from a given `&str`.
|
|
||||||
pub fn parse_vcard(vcard: &str) -> Vec<VcardContact> {
|
|
||||||
fn remove_prefix<'a>(s: &'a str, prefix: &str) -> Option<&'a str> {
|
|
||||||
let start_of_s = s.get(..prefix.len())?;
|
|
||||||
|
|
||||||
if start_of_s.eq_ignore_ascii_case(prefix) {
|
|
||||||
s.get(prefix.len()..)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fn vcard_property<'a>(s: &'a str, property: &str) -> Option<&'a str> {
|
|
||||||
let remainder = remove_prefix(s, property)?;
|
|
||||||
// If `s` is `EMAIL;TYPE=work:alice@example.com` and `property` is `EMAIL`,
|
|
||||||
// then `remainder` is now `;TYPE=work:alice@example.com`
|
|
||||||
|
|
||||||
// Note: This doesn't handle the case where there are quotes around a colon,
|
|
||||||
// like `NAME;Foo="Some quoted text: that contains a colon":value`.
|
|
||||||
// This could be improved in the future, but for now, the parsing is good enough.
|
|
||||||
let (params, value) = remainder.split_once(':')?;
|
|
||||||
// In the example from above, `params` is now `;TYPE=work`
|
|
||||||
// and `value` is now `alice@example.com`
|
|
||||||
|
|
||||||
if params
|
|
||||||
.chars()
|
|
||||||
.next()
|
|
||||||
.filter(|c| !c.is_ascii_punctuation() || *c == '_')
|
|
||||||
.is_some()
|
|
||||||
{
|
|
||||||
// `s` started with `property`, but the next character after it was not punctuation,
|
|
||||||
// so this line's property is actually something else
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
Some(value)
|
|
||||||
}
|
|
||||||
fn parse_datetime(datetime: &str) -> Result<i64> {
|
|
||||||
// According to https://www.rfc-editor.org/rfc/rfc6350#section-4.3.5, the timestamp
|
|
||||||
// is in ISO.8601.2004 format. DateTime::parse_from_rfc3339() apparently parses
|
|
||||||
// ISO.8601, but fails to parse any of the examples given.
|
|
||||||
// So, instead just parse using a format string.
|
|
||||||
|
|
||||||
// Parses 19961022T140000Z, 19961022T140000-05, or 19961022T140000-0500.
|
|
||||||
let timestamp = match DateTime::parse_from_str(datetime, "%Y%m%dT%H%M%S%#z") {
|
|
||||||
Ok(datetime) => datetime.timestamp(),
|
|
||||||
// Parses 19961022T140000.
|
|
||||||
Err(e) => match NaiveDateTime::parse_from_str(datetime, "%Y%m%dT%H%M%S") {
|
|
||||||
Ok(datetime) => datetime
|
|
||||||
.and_local_timezone(chrono::offset::Local)
|
|
||||||
.single()
|
|
||||||
.context("Could not apply local timezone to parsed date and time")?
|
|
||||||
.timestamp(),
|
|
||||||
Err(_) => return Err(e.into()),
|
|
||||||
},
|
|
||||||
};
|
|
||||||
Ok(timestamp)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove line folding, see https://datatracker.ietf.org/doc/html/rfc6350#section-3.2
|
|
||||||
static NEWLINE_AND_SPACE_OR_TAB: Lazy<Regex> = Lazy::new(|| Regex::new("\r?\n[\t ]").unwrap());
|
|
||||||
let unfolded_lines = NEWLINE_AND_SPACE_OR_TAB.replace_all(vcard, "");
|
|
||||||
|
|
||||||
let mut lines = unfolded_lines.lines().peekable();
|
|
||||||
let mut contacts = Vec::new();
|
|
||||||
|
|
||||||
while lines.peek().is_some() {
|
|
||||||
// Skip to the start of the vcard:
|
|
||||||
for line in lines.by_ref() {
|
|
||||||
if line.eq_ignore_ascii_case("BEGIN:VCARD") {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut display_name = None;
|
|
||||||
let mut addr = None;
|
|
||||||
let mut key = None;
|
|
||||||
let mut photo = None;
|
|
||||||
let mut datetime = None;
|
|
||||||
|
|
||||||
for mut line in lines.by_ref() {
|
|
||||||
if let Some(remainder) = remove_prefix(line, "item1.") {
|
|
||||||
// Remove the group name, if the group is called "item1".
|
|
||||||
// If necessary, we can improve this to also remove groups that are called something different that "item1".
|
|
||||||
//
|
|
||||||
// Search "group name" at https://datatracker.ietf.org/doc/html/rfc6350 for more infos.
|
|
||||||
line = remainder;
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(email) = vcard_property(line, "email") {
|
|
||||||
addr.get_or_insert(email);
|
|
||||||
} else if let Some(name) = vcard_property(line, "fn") {
|
|
||||||
display_name.get_or_insert(name);
|
|
||||||
} else if let Some(k) = remove_prefix(line, "KEY;PGP;ENCODING=BASE64:")
|
|
||||||
.or_else(|| remove_prefix(line, "KEY;TYPE=PGP;ENCODING=b:"))
|
|
||||||
.or_else(|| remove_prefix(line, "KEY:data:application/pgp-keys;base64,"))
|
|
||||||
.or_else(|| remove_prefix(line, "KEY;PREF=1:data:application/pgp-keys;base64,"))
|
|
||||||
{
|
|
||||||
key.get_or_insert(k);
|
|
||||||
} else if let Some(p) = remove_prefix(line, "PHOTO;JPEG;ENCODING=BASE64:")
|
|
||||||
.or_else(|| remove_prefix(line, "PHOTO;ENCODING=BASE64;JPEG:"))
|
|
||||||
.or_else(|| remove_prefix(line, "PHOTO;TYPE=JPEG;ENCODING=b:"))
|
|
||||||
.or_else(|| remove_prefix(line, "PHOTO;ENCODING=b;TYPE=JPEG:"))
|
|
||||||
.or_else(|| remove_prefix(line, "PHOTO;ENCODING=BASE64;TYPE=JPEG:"))
|
|
||||||
.or_else(|| remove_prefix(line, "PHOTO;TYPE=JPEG;ENCODING=BASE64:"))
|
|
||||||
.or_else(|| remove_prefix(line, "PHOTO:data:image/jpeg;base64,"))
|
|
||||||
{
|
|
||||||
photo.get_or_insert(p);
|
|
||||||
} else if let Some(rev) = vcard_property(line, "rev") {
|
|
||||||
datetime.get_or_insert(rev);
|
|
||||||
} else if line.eq_ignore_ascii_case("END:VCARD") {
|
|
||||||
let (authname, addr) =
|
|
||||||
sanitize_name_and_addr(display_name.unwrap_or(""), addr.unwrap_or(""));
|
|
||||||
|
|
||||||
contacts.push(VcardContact {
|
|
||||||
authname,
|
|
||||||
addr,
|
|
||||||
key: key.map(|s| s.to_string()),
|
|
||||||
profile_image: photo.map(|s| s.to_string()),
|
|
||||||
timestamp: datetime
|
|
||||||
.context("No timestamp in vcard")
|
|
||||||
.and_then(parse_datetime),
|
|
||||||
});
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
contacts
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Valid contact address.
|
/// Valid contact address.
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
@@ -256,7 +68,7 @@ impl ContactAddress {
|
|||||||
pub fn new(s: &str) -> Result<Self> {
|
pub fn new(s: &str) -> Result<Self> {
|
||||||
let addr = addr_normalize(s);
|
let addr = addr_normalize(s);
|
||||||
if !may_be_valid_addr(&addr) {
|
if !may_be_valid_addr(&addr) {
|
||||||
bail!("invalid address {:?}", s);
|
bail!("invalid address {s:?}");
|
||||||
}
|
}
|
||||||
Ok(Self(addr.to_string()))
|
Ok(Self(addr.to_string()))
|
||||||
}
|
}
|
||||||
@@ -264,7 +76,7 @@ impl ContactAddress {
|
|||||||
|
|
||||||
/// Allow converting [`ContactAddress`] to an SQLite type.
|
/// Allow converting [`ContactAddress`] to an SQLite type.
|
||||||
impl rusqlite::types::ToSql for ContactAddress {
|
impl rusqlite::types::ToSql for ContactAddress {
|
||||||
fn to_sql(&self) -> rusqlite::Result<rusqlite::types::ToSqlOutput> {
|
fn to_sql(&self) -> rusqlite::Result<rusqlite::types::ToSqlOutput<'_>> {
|
||||||
let val = rusqlite::types::Value::Text(self.0.to_string());
|
let val = rusqlite::types::Value::Text(self.0.to_string());
|
||||||
let out = rusqlite::types::ToSqlOutput::Owned(val);
|
let out = rusqlite::types::ToSqlOutput::Owned(val);
|
||||||
Ok(out)
|
Ok(out)
|
||||||
@@ -276,7 +88,8 @@ impl rusqlite::types::ToSql for ContactAddress {
|
|||||||
/// - Removes special characters from the name, see [`sanitize_name()`]
|
/// - Removes special characters from the name, see [`sanitize_name()`]
|
||||||
/// - Removes the name if it is equal to the address by setting it to ""
|
/// - Removes the name if it is equal to the address by setting it to ""
|
||||||
pub fn sanitize_name_and_addr(name: &str, addr: &str) -> (String, String) {
|
pub fn sanitize_name_and_addr(name: &str, addr: &str) -> (String, String) {
|
||||||
static ADDR_WITH_NAME_REGEX: Lazy<Regex> = Lazy::new(|| Regex::new("(.*)<(.*)>").unwrap());
|
static ADDR_WITH_NAME_REGEX: LazyLock<Regex> =
|
||||||
|
LazyLock::new(|| Regex::new("(.*)<(.*)>").unwrap());
|
||||||
let (name, addr) = if let Some(captures) = ADDR_WITH_NAME_REGEX.captures(addr.as_ref()) {
|
let (name, addr) = if let Some(captures) = ADDR_WITH_NAME_REGEX.captures(addr.as_ref()) {
|
||||||
(
|
(
|
||||||
if name.is_empty() {
|
if name.is_empty() {
|
||||||
@@ -444,16 +257,16 @@ impl EmailAddress {
|
|||||||
.chars()
|
.chars()
|
||||||
.any(|c| c.is_whitespace() || c == '<' || c == '>')
|
.any(|c| c.is_whitespace() || c == '<' || c == '>')
|
||||||
{
|
{
|
||||||
bail!("Email {:?} must not contain whitespaces, '>' or '<'", input);
|
bail!("Email {input:?} must not contain whitespaces, '>' or '<'");
|
||||||
}
|
}
|
||||||
|
|
||||||
match &parts[..] {
|
match &parts[..] {
|
||||||
[domain, local] => {
|
[domain, local] => {
|
||||||
if local.is_empty() {
|
if local.is_empty() {
|
||||||
bail!("empty string is not valid for local part in {:?}", input);
|
bail!("empty string is not valid for local part in {input:?}");
|
||||||
}
|
}
|
||||||
if domain.is_empty() {
|
if domain.is_empty() {
|
||||||
bail!("missing domain after '@' in {:?}", input);
|
bail!("missing domain after '@' in {input:?}");
|
||||||
}
|
}
|
||||||
if domain.ends_with('.') {
|
if domain.ends_with('.') {
|
||||||
bail!("Domain {domain:?} should not contain the dot in the end");
|
bail!("Domain {domain:?} should not contain the dot in the end");
|
||||||
@@ -463,13 +276,13 @@ impl EmailAddress {
|
|||||||
domain: (*domain).to_string(),
|
domain: (*domain).to_string(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
_ => bail!("Email {:?} must contain '@' character", input),
|
_ => bail!("Email {input:?} must contain '@' character"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl rusqlite::types::ToSql for EmailAddress {
|
impl rusqlite::types::ToSql for EmailAddress {
|
||||||
fn to_sql(&self) -> rusqlite::Result<rusqlite::types::ToSqlOutput> {
|
fn to_sql(&self) -> rusqlite::Result<rusqlite::types::ToSqlOutput<'_>> {
|
||||||
let val = rusqlite::types::Value::Text(self.to_string());
|
let val = rusqlite::types::Value::Text(self.to_string());
|
||||||
let out = rusqlite::types::ToSqlOutput::Owned(val);
|
let out = rusqlite::types::ToSqlOutput::Owned(val);
|
||||||
Ok(out)
|
Ok(out)
|
||||||
@@ -478,148 +291,8 @@ impl rusqlite::types::ToSql for EmailAddress {
|
|||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use chrono::TimeZone;
|
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_vcard_thunderbird() {
|
|
||||||
let contacts = parse_vcard(
|
|
||||||
"BEGIN:VCARD
|
|
||||||
VERSION:4.0
|
|
||||||
FN:'Alice Mueller'
|
|
||||||
EMAIL;PREF=1:alice.mueller@posteo.de
|
|
||||||
UID:a8083264-ca47-4be7-98a8-8ec3db1447ca
|
|
||||||
END:VCARD
|
|
||||||
BEGIN:VCARD
|
|
||||||
VERSION:4.0
|
|
||||||
FN:'bobzzz@freenet.de'
|
|
||||||
EMAIL;PREF=1:bobzzz@freenet.de
|
|
||||||
UID:cac4fef4-6351-4854-bbe4-9b6df857eaed
|
|
||||||
END:VCARD
|
|
||||||
",
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_eq!(contacts[0].addr, "alice.mueller@posteo.de".to_string());
|
|
||||||
assert_eq!(contacts[0].authname, "Alice Mueller".to_string());
|
|
||||||
assert_eq!(contacts[0].key, None);
|
|
||||||
assert_eq!(contacts[0].profile_image, None);
|
|
||||||
assert!(contacts[0].timestamp.is_err());
|
|
||||||
|
|
||||||
assert_eq!(contacts[1].addr, "bobzzz@freenet.de".to_string());
|
|
||||||
assert_eq!(contacts[1].authname, "".to_string());
|
|
||||||
assert_eq!(contacts[1].key, None);
|
|
||||||
assert_eq!(contacts[1].profile_image, None);
|
|
||||||
assert!(contacts[1].timestamp.is_err());
|
|
||||||
|
|
||||||
assert_eq!(contacts.len(), 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_vcard_simple_example() {
|
|
||||||
let contacts = parse_vcard(
|
|
||||||
"BEGIN:VCARD
|
|
||||||
VERSION:4.0
|
|
||||||
FN:Alice Wonderland
|
|
||||||
N:Wonderland;Alice;;;Ms.
|
|
||||||
GENDER:W
|
|
||||||
EMAIL;TYPE=work:alice@example.com
|
|
||||||
KEY;TYPE=PGP;ENCODING=b:[base64-data]
|
|
||||||
REV:20240418T184242Z
|
|
||||||
|
|
||||||
END:VCARD",
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_eq!(contacts[0].addr, "alice@example.com".to_string());
|
|
||||||
assert_eq!(contacts[0].authname, "Alice Wonderland".to_string());
|
|
||||||
assert_eq!(contacts[0].key, Some("[base64-data]".to_string()));
|
|
||||||
assert_eq!(contacts[0].profile_image, None);
|
|
||||||
assert_eq!(*contacts[0].timestamp.as_ref().unwrap(), 1713465762);
|
|
||||||
|
|
||||||
assert_eq!(contacts.len(), 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_vcard_with_trailing_newline() {
|
|
||||||
let contacts = parse_vcard(
|
|
||||||
"BEGIN:VCARD\r
|
|
||||||
VERSION:4.0\r
|
|
||||||
FN:Alice Wonderland\r
|
|
||||||
N:Wonderland;Alice;;;Ms.\r
|
|
||||||
GENDER:W\r
|
|
||||||
EMAIL;TYPE=work:alice@example.com\r
|
|
||||||
KEY;TYPE=PGP;ENCODING=b:[base64-data]\r
|
|
||||||
REV:20240418T184242Z\r
|
|
||||||
END:VCARD\r
|
|
||||||
\r",
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_eq!(contacts[0].addr, "alice@example.com".to_string());
|
|
||||||
assert_eq!(contacts[0].authname, "Alice Wonderland".to_string());
|
|
||||||
assert_eq!(contacts[0].key, Some("[base64-data]".to_string()));
|
|
||||||
assert_eq!(contacts[0].profile_image, None);
|
|
||||||
assert_eq!(*contacts[0].timestamp.as_ref().unwrap(), 1713465762);
|
|
||||||
|
|
||||||
assert_eq!(contacts.len(), 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_make_and_parse_vcard() {
|
|
||||||
let contacts = [
|
|
||||||
VcardContact {
|
|
||||||
addr: "alice@example.org".to_string(),
|
|
||||||
authname: "Alice Wonderland".to_string(),
|
|
||||||
key: Some("[base64-data]".to_string()),
|
|
||||||
profile_image: Some("image in Base64".to_string()),
|
|
||||||
timestamp: Ok(1713465762),
|
|
||||||
},
|
|
||||||
VcardContact {
|
|
||||||
addr: "bob@example.com".to_string(),
|
|
||||||
authname: "".to_string(),
|
|
||||||
key: None,
|
|
||||||
profile_image: None,
|
|
||||||
timestamp: Ok(0),
|
|
||||||
},
|
|
||||||
];
|
|
||||||
let items = [
|
|
||||||
"BEGIN:VCARD\r\n\
|
|
||||||
VERSION:4.0\r\n\
|
|
||||||
EMAIL:alice@example.org\r\n\
|
|
||||||
FN:Alice Wonderland\r\n\
|
|
||||||
KEY:data:application/pgp-keys;base64,[base64-data]\r\n\
|
|
||||||
PHOTO:data:image/jpeg;base64,image in Base64\r\n\
|
|
||||||
REV:20240418T184242Z\r\n\
|
|
||||||
END:VCARD\r\n",
|
|
||||||
"BEGIN:VCARD\r\n\
|
|
||||||
VERSION:4.0\r\n\
|
|
||||||
EMAIL:bob@example.com\r\n\
|
|
||||||
FN:bob@example.com\r\n\
|
|
||||||
REV:19700101T000000Z\r\n\
|
|
||||||
END:VCARD\r\n",
|
|
||||||
];
|
|
||||||
let mut expected = "".to_string();
|
|
||||||
for len in 0..=contacts.len() {
|
|
||||||
let contacts = &contacts[0..len];
|
|
||||||
let vcard = make_vcard(contacts);
|
|
||||||
if len > 0 {
|
|
||||||
expected += items[len - 1];
|
|
||||||
}
|
|
||||||
assert_eq!(vcard, expected);
|
|
||||||
let parsed = parse_vcard(&vcard);
|
|
||||||
assert_eq!(parsed.len(), contacts.len());
|
|
||||||
for i in 0..parsed.len() {
|
|
||||||
assert_eq!(parsed[i].addr, contacts[i].addr);
|
|
||||||
assert_eq!(parsed[i].authname, contacts[i].authname);
|
|
||||||
assert_eq!(parsed[i].key, contacts[i].key);
|
|
||||||
assert_eq!(parsed[i].profile_image, contacts[i].profile_image);
|
|
||||||
assert_eq!(
|
|
||||||
parsed[i].timestamp.as_ref().unwrap(),
|
|
||||||
contacts[i].timestamp.as_ref().unwrap()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_contact_address() -> Result<()> {
|
fn test_contact_address() -> Result<()> {
|
||||||
let alice_addr = "alice@example.org";
|
let alice_addr = "alice@example.org";
|
||||||
@@ -666,112 +339,6 @@ END:VCARD\r
|
|||||||
assert_eq!(EmailAddress::new("@d.tt").is_ok(), false);
|
assert_eq!(EmailAddress::new("@d.tt").is_ok(), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_vcard_android() {
|
|
||||||
let contacts = parse_vcard(
|
|
||||||
"BEGIN:VCARD
|
|
||||||
VERSION:2.1
|
|
||||||
N:;Bob;;;
|
|
||||||
FN:Bob
|
|
||||||
TEL;CELL:+1-234-567-890
|
|
||||||
EMAIL;HOME:bob@example.org
|
|
||||||
END:VCARD
|
|
||||||
BEGIN:VCARD
|
|
||||||
VERSION:2.1
|
|
||||||
N:;Alice;;;
|
|
||||||
FN:Alice
|
|
||||||
EMAIL;HOME:alice@example.org
|
|
||||||
END:VCARD
|
|
||||||
",
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_eq!(contacts[0].addr, "bob@example.org".to_string());
|
|
||||||
assert_eq!(contacts[0].authname, "Bob".to_string());
|
|
||||||
assert_eq!(contacts[0].key, None);
|
|
||||||
assert_eq!(contacts[0].profile_image, None);
|
|
||||||
|
|
||||||
assert_eq!(contacts[1].addr, "alice@example.org".to_string());
|
|
||||||
assert_eq!(contacts[1].authname, "Alice".to_string());
|
|
||||||
assert_eq!(contacts[1].key, None);
|
|
||||||
assert_eq!(contacts[1].profile_image, None);
|
|
||||||
|
|
||||||
assert_eq!(contacts.len(), 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_vcard_local_datetime() {
|
|
||||||
let contacts = parse_vcard(
|
|
||||||
"BEGIN:VCARD\n\
|
|
||||||
VERSION:4.0\n\
|
|
||||||
FN:Alice Wonderland\n\
|
|
||||||
EMAIL;TYPE=work:alice@example.org\n\
|
|
||||||
REV:20240418T184242\n\
|
|
||||||
END:VCARD",
|
|
||||||
);
|
|
||||||
assert_eq!(contacts.len(), 1);
|
|
||||||
assert_eq!(contacts[0].addr, "alice@example.org".to_string());
|
|
||||||
assert_eq!(contacts[0].authname, "Alice Wonderland".to_string());
|
|
||||||
assert_eq!(
|
|
||||||
*contacts[0].timestamp.as_ref().unwrap(),
|
|
||||||
chrono::offset::Local
|
|
||||||
.with_ymd_and_hms(2024, 4, 18, 18, 42, 42)
|
|
||||||
.unwrap()
|
|
||||||
.timestamp()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_vcard_with_base64_avatar() {
|
|
||||||
// This is not an actual base64-encoded avatar, it's just to test the parsing.
|
|
||||||
// This one is Android-like.
|
|
||||||
let vcard0 = "BEGIN:VCARD
|
|
||||||
VERSION:2.1
|
|
||||||
N:;Bob;;;
|
|
||||||
FN:Bob
|
|
||||||
EMAIL;HOME:bob@example.org
|
|
||||||
PHOTO;ENCODING=BASE64;JPEG:/9j/4AAQSkZJRgABAQAAAQABAAD/4gIoSUNDX1BST0ZJTEU
|
|
||||||
AAQEAAAIYAAAAAAQwAABtbnRyUkdCIFhZWiAAAAAAAAAAAAAAAABhY3NwAAAAAAAAAAAAAAAA
|
|
||||||
L8bRuAJYoZUYrI4ZY3VWwxw4Ay28AAGBISScmf/2Q==
|
|
||||||
|
|
||||||
END:VCARD
|
|
||||||
";
|
|
||||||
// This one is DOS-like.
|
|
||||||
let vcard1 = vcard0.replace('\n', "\r\n");
|
|
||||||
for vcard in [vcard0, vcard1.as_str()] {
|
|
||||||
let contacts = parse_vcard(vcard);
|
|
||||||
assert_eq!(contacts.len(), 1);
|
|
||||||
assert_eq!(contacts[0].addr, "bob@example.org".to_string());
|
|
||||||
assert_eq!(contacts[0].authname, "Bob".to_string());
|
|
||||||
assert_eq!(contacts[0].key, None);
|
|
||||||
assert_eq!(contacts[0].profile_image.as_deref().unwrap(), "/9j/4AAQSkZJRgABAQAAAQABAAD/4gIoSUNDX1BST0ZJTEUAAQEAAAIYAAAAAAQwAABtbnRyUkdCIFhZWiAAAAAAAAAAAAAAAABhY3NwAAAAAAAAAAAAAAAAL8bRuAJYoZUYrI4ZY3VWwxw4Ay28AAGBISScmf/2Q==");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_protonmail_vcard() {
|
|
||||||
let contacts = parse_vcard(
|
|
||||||
"BEGIN:VCARD
|
|
||||||
VERSION:4.0
|
|
||||||
FN;PREF=1:Alice Wonderland
|
|
||||||
UID:proton-web-03747582-328d-38dc-5ddd-000000000000
|
|
||||||
ITEM1.EMAIL;PREF=1:alice@example.org
|
|
||||||
ITEM1.KEY;PREF=1:data:application/pgp-keys;base64,aaaaaaaaaaaaaaaaaaaaaaaaa
|
|
||||||
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
|
|
||||||
ITEM1.KEY;PREF=2:data:application/pgp-keys;base64,bbbbbbbbbbbbbbbbbbbbbbbbb
|
|
||||||
bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb
|
|
||||||
ITEM1.X-PM-ENCRYPT:true
|
|
||||||
ITEM1.X-PM-SIGN:true
|
|
||||||
END:VCARD",
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_eq!(contacts.len(), 1);
|
|
||||||
assert_eq!(&contacts[0].addr, "alice@example.org");
|
|
||||||
assert_eq!(&contacts[0].authname, "Alice Wonderland");
|
|
||||||
assert_eq!(contacts[0].key.as_ref().unwrap(), "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa");
|
|
||||||
assert!(contacts[0].timestamp.is_err());
|
|
||||||
assert_eq!(contacts[0].profile_image, None);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_sanitize_name() {
|
fn test_sanitize_name() {
|
||||||
assert_eq!(&sanitize_name(" hello world "), "hello world");
|
assert_eq!(&sanitize_name(" hello world "), "hello world");
|
||||||
|
|||||||
247
deltachat-contact-tools/src/vcard.rs
Normal file
247
deltachat-contact-tools/src/vcard.rs
Normal file
@@ -0,0 +1,247 @@
|
|||||||
|
use std::sync::LazyLock;
|
||||||
|
|
||||||
|
use anyhow::Context as _;
|
||||||
|
use anyhow::Result;
|
||||||
|
use chrono::DateTime;
|
||||||
|
use chrono::NaiveDateTime;
|
||||||
|
use regex::Regex;
|
||||||
|
|
||||||
|
use crate::sanitize_name_and_addr;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
/// A Contact, as represented in a VCard.
|
||||||
|
pub struct VcardContact {
|
||||||
|
/// The email address, vcard property `email`
|
||||||
|
pub addr: String,
|
||||||
|
/// This must be the name authorized by the contact itself, not a locally given name. Vcard
|
||||||
|
/// property `fn`. Can be empty, one should use `display_name()` to obtain the display name.
|
||||||
|
pub authname: String,
|
||||||
|
/// The contact's public PGP key in Base64, vcard property `key`
|
||||||
|
pub key: Option<String>,
|
||||||
|
/// The contact's profile image (=avatar) in Base64, vcard property `photo`
|
||||||
|
pub profile_image: Option<String>,
|
||||||
|
/// The biography, stored in the vcard property `note`
|
||||||
|
pub biography: Option<String>,
|
||||||
|
/// The timestamp when the vcard was created / last updated, vcard property `rev`
|
||||||
|
pub timestamp: Result<i64>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl VcardContact {
|
||||||
|
/// Returns the contact's display name.
|
||||||
|
pub fn display_name(&self) -> &str {
|
||||||
|
match self.authname.is_empty() {
|
||||||
|
false => &self.authname,
|
||||||
|
true => &self.addr,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a vCard containing given contacts.
|
||||||
|
///
|
||||||
|
/// Calling [`parse_vcard()`] on the returned result is a reverse operation.
|
||||||
|
pub fn make_vcard(contacts: &[VcardContact]) -> String {
|
||||||
|
fn format_timestamp(c: &VcardContact) -> Option<String> {
|
||||||
|
let timestamp = *c.timestamp.as_ref().ok()?;
|
||||||
|
let datetime = DateTime::from_timestamp(timestamp, 0)?;
|
||||||
|
Some(datetime.format("%Y%m%dT%H%M%SZ").to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn escape(s: &str) -> String {
|
||||||
|
s.replace(',', "\\,")
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut res = "".to_string();
|
||||||
|
for c in contacts {
|
||||||
|
// Mustn't contain ',', but it's easier to escape than to error out.
|
||||||
|
let addr = escape(&c.addr);
|
||||||
|
let display_name = escape(c.display_name());
|
||||||
|
res += &format!(
|
||||||
|
"BEGIN:VCARD\r\n\
|
||||||
|
VERSION:4.0\r\n\
|
||||||
|
EMAIL:{addr}\r\n\
|
||||||
|
FN:{display_name}\r\n"
|
||||||
|
);
|
||||||
|
if let Some(key) = &c.key {
|
||||||
|
res += &format!("KEY:data:application/pgp-keys;base64\\,{key}\r\n");
|
||||||
|
}
|
||||||
|
if let Some(profile_image) = &c.profile_image {
|
||||||
|
res += &format!("PHOTO:data:image/jpeg;base64\\,{profile_image}\r\n");
|
||||||
|
}
|
||||||
|
if let Some(biography) = &c.biography {
|
||||||
|
res += &format!("NOTE:{}\r\n", escape(biography));
|
||||||
|
}
|
||||||
|
if let Some(timestamp) = format_timestamp(c) {
|
||||||
|
res += &format!("REV:{timestamp}\r\n");
|
||||||
|
}
|
||||||
|
res += "END:VCARD\r\n";
|
||||||
|
}
|
||||||
|
res
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Parses `VcardContact`s from a given `&str`.
|
||||||
|
pub fn parse_vcard(vcard: &str) -> Vec<VcardContact> {
|
||||||
|
fn remove_prefix<'a>(s: &'a str, prefix: &str) -> Option<&'a str> {
|
||||||
|
let start_of_s = s.get(..prefix.len())?;
|
||||||
|
|
||||||
|
if start_of_s.eq_ignore_ascii_case(prefix) {
|
||||||
|
s.get(prefix.len()..)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/// Returns (parameters, raw value) tuple.
|
||||||
|
fn vcard_property_raw<'a>(line: &'a str, property: &str) -> Option<(&'a str, &'a str)> {
|
||||||
|
let remainder = remove_prefix(line, property)?;
|
||||||
|
// If `s` is `EMAIL;TYPE=work:alice@example.com` and `property` is `EMAIL`,
|
||||||
|
// then `remainder` is now `;TYPE=work:alice@example.com`
|
||||||
|
|
||||||
|
// Note: This doesn't handle the case where there are quotes around a colon,
|
||||||
|
// like `NAME;Foo="Some quoted text: that contains a colon":value`.
|
||||||
|
// This could be improved in the future, but for now, the parsing is good enough.
|
||||||
|
let (mut params, value) = remainder.split_once(':')?;
|
||||||
|
// In the example from above, `params` is now `;TYPE=work`
|
||||||
|
// and `value` is now `alice@example.com`
|
||||||
|
|
||||||
|
if params
|
||||||
|
.chars()
|
||||||
|
.next()
|
||||||
|
.filter(|c| !c.is_ascii_punctuation() || *c == '_')
|
||||||
|
.is_some()
|
||||||
|
{
|
||||||
|
// `s` started with `property`, but the next character after it was not punctuation,
|
||||||
|
// so this line's property is actually something else
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
if let Some(p) = remove_prefix(params, ";") {
|
||||||
|
params = p;
|
||||||
|
}
|
||||||
|
if let Some(p) = remove_prefix(params, "PREF=1") {
|
||||||
|
params = p;
|
||||||
|
}
|
||||||
|
Some((params, value))
|
||||||
|
}
|
||||||
|
/// Returns (parameters, unescaped value) tuple.
|
||||||
|
fn vcard_property<'a>(line: &'a str, property: &str) -> Option<(&'a str, String)> {
|
||||||
|
let (params, value) = vcard_property_raw(line, property)?;
|
||||||
|
// Some fields can't contain commas, but unescape them everywhere for safety.
|
||||||
|
Some((params, value.replace("\\,", ",")))
|
||||||
|
}
|
||||||
|
fn base64_key(line: &str) -> Option<&str> {
|
||||||
|
let (params, value) = vcard_property_raw(line, "key")?;
|
||||||
|
if params.eq_ignore_ascii_case("PGP;ENCODING=BASE64")
|
||||||
|
|| params.eq_ignore_ascii_case("TYPE=PGP;ENCODING=b")
|
||||||
|
{
|
||||||
|
return Some(value);
|
||||||
|
}
|
||||||
|
remove_prefix(value, "data:application/pgp-keys;base64\\,")
|
||||||
|
// Old Delta Chat format.
|
||||||
|
.or_else(|| remove_prefix(value, "data:application/pgp-keys;base64,"))
|
||||||
|
}
|
||||||
|
fn base64_photo(line: &str) -> Option<&str> {
|
||||||
|
let (params, value) = vcard_property_raw(line, "photo")?;
|
||||||
|
if params.eq_ignore_ascii_case("JPEG;ENCODING=BASE64")
|
||||||
|
|| params.eq_ignore_ascii_case("ENCODING=BASE64;JPEG")
|
||||||
|
|| params.eq_ignore_ascii_case("TYPE=JPEG;ENCODING=b")
|
||||||
|
|| params.eq_ignore_ascii_case("ENCODING=b;TYPE=JPEG")
|
||||||
|
|| params.eq_ignore_ascii_case("ENCODING=BASE64;TYPE=JPEG")
|
||||||
|
|| params.eq_ignore_ascii_case("TYPE=JPEG;ENCODING=BASE64")
|
||||||
|
{
|
||||||
|
return Some(value);
|
||||||
|
}
|
||||||
|
remove_prefix(value, "data:image/jpeg;base64\\,")
|
||||||
|
// Old Delta Chat format.
|
||||||
|
.or_else(|| remove_prefix(value, "data:image/jpeg;base64,"))
|
||||||
|
}
|
||||||
|
fn parse_datetime(datetime: &str) -> Result<i64> {
|
||||||
|
// According to https://www.rfc-editor.org/rfc/rfc6350#section-4.3.5, the timestamp
|
||||||
|
// is in ISO.8601.2004 format. DateTime::parse_from_rfc3339() apparently parses
|
||||||
|
// ISO.8601, but fails to parse any of the examples given.
|
||||||
|
// So, instead just parse using a format string.
|
||||||
|
|
||||||
|
// Parses 19961022T140000Z, 19961022T140000-05, or 19961022T140000-0500.
|
||||||
|
let timestamp = match DateTime::parse_from_str(datetime, "%Y%m%dT%H%M%S%#z") {
|
||||||
|
Ok(datetime) => datetime.timestamp(),
|
||||||
|
// Parses 19961022T140000.
|
||||||
|
Err(e) => match NaiveDateTime::parse_from_str(datetime, "%Y%m%dT%H%M%S") {
|
||||||
|
Ok(datetime) => datetime
|
||||||
|
.and_local_timezone(chrono::offset::Local)
|
||||||
|
.single()
|
||||||
|
.context("Could not apply local timezone to parsed date and time")?
|
||||||
|
.timestamp(),
|
||||||
|
Err(_) => return Err(e.into()),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
Ok(timestamp)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove line folding, see https://datatracker.ietf.org/doc/html/rfc6350#section-3.2
|
||||||
|
static NEWLINE_AND_SPACE_OR_TAB: LazyLock<Regex> =
|
||||||
|
LazyLock::new(|| Regex::new("\r?\n[\t ]").unwrap());
|
||||||
|
let unfolded_lines = NEWLINE_AND_SPACE_OR_TAB.replace_all(vcard, "");
|
||||||
|
|
||||||
|
let mut lines = unfolded_lines.lines().peekable();
|
||||||
|
let mut contacts = Vec::new();
|
||||||
|
|
||||||
|
while lines.peek().is_some() {
|
||||||
|
// Skip to the start of the vcard:
|
||||||
|
for line in lines.by_ref() {
|
||||||
|
if line.eq_ignore_ascii_case("BEGIN:VCARD") {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut display_name = None;
|
||||||
|
let mut addr = None;
|
||||||
|
let mut key = None;
|
||||||
|
let mut photo = None;
|
||||||
|
let mut biography = None;
|
||||||
|
let mut datetime = None;
|
||||||
|
|
||||||
|
for mut line in lines.by_ref() {
|
||||||
|
if let Some(remainder) = remove_prefix(line, "item1.") {
|
||||||
|
// Remove the group name, if the group is called "item1".
|
||||||
|
// If necessary, we can improve this to also remove groups that are called something different that "item1".
|
||||||
|
//
|
||||||
|
// Search "group name" at https://datatracker.ietf.org/doc/html/rfc6350 for more infos.
|
||||||
|
line = remainder;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some((_params, email)) = vcard_property(line, "email") {
|
||||||
|
addr.get_or_insert(email);
|
||||||
|
} else if let Some((_params, name)) = vcard_property(line, "fn") {
|
||||||
|
display_name.get_or_insert(name);
|
||||||
|
} else if let Some(k) = base64_key(line) {
|
||||||
|
key.get_or_insert(k);
|
||||||
|
} else if let Some(p) = base64_photo(line) {
|
||||||
|
photo.get_or_insert(p);
|
||||||
|
} else if let Some((_params, bio)) = vcard_property(line, "note") {
|
||||||
|
biography.get_or_insert(bio);
|
||||||
|
} else if let Some((_params, rev)) = vcard_property(line, "rev") {
|
||||||
|
datetime.get_or_insert(rev);
|
||||||
|
} else if line.eq_ignore_ascii_case("END:VCARD") {
|
||||||
|
let (authname, addr) = sanitize_name_and_addr(
|
||||||
|
&display_name.unwrap_or_default(),
|
||||||
|
&addr.unwrap_or_default(),
|
||||||
|
);
|
||||||
|
|
||||||
|
contacts.push(VcardContact {
|
||||||
|
authname,
|
||||||
|
addr,
|
||||||
|
key: key.map(|s| s.to_string()),
|
||||||
|
profile_image: photo.map(|s| s.to_string()),
|
||||||
|
biography,
|
||||||
|
timestamp: datetime
|
||||||
|
.as_deref()
|
||||||
|
.context("No timestamp in vcard")
|
||||||
|
.and_then(parse_datetime),
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
contacts
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod vcard_tests;
|
||||||
278
deltachat-contact-tools/src/vcard/vcard_tests.rs
Normal file
278
deltachat-contact-tools/src/vcard/vcard_tests.rs
Normal file
@@ -0,0 +1,278 @@
|
|||||||
|
use chrono::TimeZone as _;
|
||||||
|
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_vcard_thunderbird() {
|
||||||
|
let contacts = parse_vcard(
|
||||||
|
"BEGIN:VCARD
|
||||||
|
VERSION:4.0
|
||||||
|
FN:'Alice Mueller'
|
||||||
|
EMAIL;PREF=1:alice.mueller@posteo.de
|
||||||
|
UID:a8083264-ca47-4be7-98a8-8ec3db1447ca
|
||||||
|
END:VCARD
|
||||||
|
BEGIN:VCARD
|
||||||
|
VERSION:4.0
|
||||||
|
FN:'bobzzz@freenet.de'
|
||||||
|
EMAIL;PREF=1:bobzzz@freenet.de
|
||||||
|
UID:cac4fef4-6351-4854-bbe4-9b6df857eaed
|
||||||
|
END:VCARD
|
||||||
|
",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(contacts[0].addr, "alice.mueller@posteo.de".to_string());
|
||||||
|
assert_eq!(contacts[0].authname, "Alice Mueller".to_string());
|
||||||
|
assert_eq!(contacts[0].key, None);
|
||||||
|
assert_eq!(contacts[0].profile_image, None);
|
||||||
|
assert!(contacts[0].timestamp.is_err());
|
||||||
|
|
||||||
|
assert_eq!(contacts[1].addr, "bobzzz@freenet.de".to_string());
|
||||||
|
assert_eq!(contacts[1].authname, "".to_string());
|
||||||
|
assert_eq!(contacts[1].key, None);
|
||||||
|
assert_eq!(contacts[1].profile_image, None);
|
||||||
|
assert!(contacts[1].timestamp.is_err());
|
||||||
|
|
||||||
|
assert_eq!(contacts.len(), 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_vcard_simple_example() {
|
||||||
|
let contacts = parse_vcard(
|
||||||
|
"BEGIN:VCARD
|
||||||
|
VERSION:4.0
|
||||||
|
FN:Alice Wonderland
|
||||||
|
N:Wonderland;Alice;;;Ms.
|
||||||
|
GENDER:W
|
||||||
|
EMAIL;TYPE=work:alice@example.com
|
||||||
|
KEY;TYPE=PGP;ENCODING=b:[base64-data]
|
||||||
|
REV:20240418T184242Z
|
||||||
|
|
||||||
|
END:VCARD",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(contacts[0].addr, "alice@example.com".to_string());
|
||||||
|
assert_eq!(contacts[0].authname, "Alice Wonderland".to_string());
|
||||||
|
assert_eq!(contacts[0].key, Some("[base64-data]".to_string()));
|
||||||
|
assert_eq!(contacts[0].profile_image, None);
|
||||||
|
assert_eq!(*contacts[0].timestamp.as_ref().unwrap(), 1713465762);
|
||||||
|
|
||||||
|
assert_eq!(contacts.len(), 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_vcard_with_trailing_newline() {
|
||||||
|
let contacts = parse_vcard(
|
||||||
|
"BEGIN:VCARD\r
|
||||||
|
VERSION:4.0\r
|
||||||
|
FN:Alice Wonderland\r
|
||||||
|
N:Wonderland;Alice;;;Ms.\r
|
||||||
|
GENDER:W\r
|
||||||
|
EMAIL;TYPE=work:alice@example.com\r
|
||||||
|
KEY;TYPE=PGP;ENCODING=b:[base64-data]\r
|
||||||
|
REV:20240418T184242Z\r
|
||||||
|
END:VCARD\r
|
||||||
|
\r",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(contacts[0].addr, "alice@example.com".to_string());
|
||||||
|
assert_eq!(contacts[0].authname, "Alice Wonderland".to_string());
|
||||||
|
assert_eq!(contacts[0].key, Some("[base64-data]".to_string()));
|
||||||
|
assert_eq!(contacts[0].profile_image, None);
|
||||||
|
assert_eq!(*contacts[0].timestamp.as_ref().unwrap(), 1713465762);
|
||||||
|
|
||||||
|
assert_eq!(contacts.len(), 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_make_and_parse_vcard() {
|
||||||
|
let contacts = [
|
||||||
|
VcardContact {
|
||||||
|
addr: "alice@example.org".to_string(),
|
||||||
|
authname: "Alice Wonderland".to_string(),
|
||||||
|
key: Some("[base64-data]".to_string()),
|
||||||
|
profile_image: Some("image in Base64".to_string()),
|
||||||
|
biography: Some("Hi, I'm Alice".to_string()),
|
||||||
|
timestamp: Ok(1713465762),
|
||||||
|
},
|
||||||
|
VcardContact {
|
||||||
|
addr: "bob@example.com".to_string(),
|
||||||
|
authname: "".to_string(),
|
||||||
|
key: None,
|
||||||
|
profile_image: None,
|
||||||
|
biography: None,
|
||||||
|
timestamp: Ok(0),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
let items = [
|
||||||
|
"BEGIN:VCARD\r\n\
|
||||||
|
VERSION:4.0\r\n\
|
||||||
|
EMAIL:alice@example.org\r\n\
|
||||||
|
FN:Alice Wonderland\r\n\
|
||||||
|
KEY:data:application/pgp-keys;base64\\,[base64-data]\r\n\
|
||||||
|
PHOTO:data:image/jpeg;base64\\,image in Base64\r\n\
|
||||||
|
NOTE:Hi\\, I'm Alice\r\n\
|
||||||
|
REV:20240418T184242Z\r\n\
|
||||||
|
END:VCARD\r\n",
|
||||||
|
"BEGIN:VCARD\r\n\
|
||||||
|
VERSION:4.0\r\n\
|
||||||
|
EMAIL:bob@example.com\r\n\
|
||||||
|
FN:bob@example.com\r\n\
|
||||||
|
REV:19700101T000000Z\r\n\
|
||||||
|
END:VCARD\r\n",
|
||||||
|
];
|
||||||
|
let mut expected = "".to_string();
|
||||||
|
for len in 0..=contacts.len() {
|
||||||
|
let contacts = &contacts[0..len];
|
||||||
|
let vcard = make_vcard(contacts);
|
||||||
|
if len > 0 {
|
||||||
|
expected += items[len - 1];
|
||||||
|
}
|
||||||
|
assert_eq!(vcard, expected);
|
||||||
|
let parsed = parse_vcard(&vcard);
|
||||||
|
assert_eq!(parsed.len(), contacts.len());
|
||||||
|
for i in 0..parsed.len() {
|
||||||
|
assert_eq!(parsed[i].addr, contacts[i].addr);
|
||||||
|
assert_eq!(parsed[i].authname, contacts[i].authname);
|
||||||
|
assert_eq!(parsed[i].key, contacts[i].key);
|
||||||
|
assert_eq!(parsed[i].profile_image, contacts[i].profile_image);
|
||||||
|
assert_eq!(
|
||||||
|
parsed[i].timestamp.as_ref().unwrap(),
|
||||||
|
contacts[i].timestamp.as_ref().unwrap()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_vcard_android() {
|
||||||
|
let contacts = parse_vcard(
|
||||||
|
"BEGIN:VCARD
|
||||||
|
VERSION:2.1
|
||||||
|
N:;Bob;;;
|
||||||
|
FN:Bob
|
||||||
|
TEL;CELL:+1-234-567-890
|
||||||
|
EMAIL;HOME:bob@example.org
|
||||||
|
END:VCARD
|
||||||
|
BEGIN:VCARD
|
||||||
|
VERSION:2.1
|
||||||
|
N:;Alice;;;
|
||||||
|
FN:Alice
|
||||||
|
EMAIL;HOME:alice@example.org
|
||||||
|
END:VCARD
|
||||||
|
",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(contacts[0].addr, "bob@example.org".to_string());
|
||||||
|
assert_eq!(contacts[0].authname, "Bob".to_string());
|
||||||
|
assert_eq!(contacts[0].key, None);
|
||||||
|
assert_eq!(contacts[0].profile_image, None);
|
||||||
|
|
||||||
|
assert_eq!(contacts[1].addr, "alice@example.org".to_string());
|
||||||
|
assert_eq!(contacts[1].authname, "Alice".to_string());
|
||||||
|
assert_eq!(contacts[1].key, None);
|
||||||
|
assert_eq!(contacts[1].profile_image, None);
|
||||||
|
|
||||||
|
assert_eq!(contacts.len(), 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_vcard_local_datetime() {
|
||||||
|
let contacts = parse_vcard(
|
||||||
|
"BEGIN:VCARD\n\
|
||||||
|
VERSION:4.0\n\
|
||||||
|
FN:Alice Wonderland\n\
|
||||||
|
EMAIL;TYPE=work:alice@example.org\n\
|
||||||
|
REV:20240418T184242\n\
|
||||||
|
END:VCARD",
|
||||||
|
);
|
||||||
|
assert_eq!(contacts.len(), 1);
|
||||||
|
assert_eq!(contacts[0].addr, "alice@example.org".to_string());
|
||||||
|
assert_eq!(contacts[0].authname, "Alice Wonderland".to_string());
|
||||||
|
assert_eq!(
|
||||||
|
*contacts[0].timestamp.as_ref().unwrap(),
|
||||||
|
chrono::offset::Local
|
||||||
|
.with_ymd_and_hms(2024, 4, 18, 18, 42, 42)
|
||||||
|
.unwrap()
|
||||||
|
.timestamp()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_vcard_with_base64_avatar() {
|
||||||
|
// This is not an actual base64-encoded avatar, it's just to test the parsing.
|
||||||
|
// This one is Android-like.
|
||||||
|
let vcard0 = "BEGIN:VCARD
|
||||||
|
VERSION:2.1
|
||||||
|
N:;Bob;;;
|
||||||
|
FN:Bob
|
||||||
|
EMAIL;HOME:bob@example.org
|
||||||
|
PHOTO;ENCODING=BASE64;JPEG:/9j/4AAQSkZJRgABAQAAAQABAAD/4gIoSUNDX1BST0ZJTEU
|
||||||
|
AAQEAAAIYAAAAAAQwAABtbnRyUkdCIFhZWiAAAAAAAAAAAAAAAABhY3NwAAAAAAAAAAAAAAAA
|
||||||
|
L8bRuAJYoZUYrI4ZY3VWwxw4Ay28AAGBISScmf/2Q==
|
||||||
|
|
||||||
|
END:VCARD
|
||||||
|
";
|
||||||
|
// This one is DOS-like.
|
||||||
|
let vcard1 = vcard0.replace('\n', "\r\n");
|
||||||
|
for vcard in [vcard0, vcard1.as_str()] {
|
||||||
|
let contacts = parse_vcard(vcard);
|
||||||
|
assert_eq!(contacts.len(), 1);
|
||||||
|
assert_eq!(contacts[0].addr, "bob@example.org".to_string());
|
||||||
|
assert_eq!(contacts[0].authname, "Bob".to_string());
|
||||||
|
assert_eq!(contacts[0].key, None);
|
||||||
|
assert_eq!(contacts[0].profile_image.as_deref().unwrap(), "/9j/4AAQSkZJRgABAQAAAQABAAD/4gIoSUNDX1BST0ZJTEUAAQEAAAIYAAAAAAQwAABtbnRyUkdCIFhZWiAAAAAAAAAAAAAAAABhY3NwAAAAAAAAAAAAAAAAL8bRuAJYoZUYrI4ZY3VWwxw4Ay28AAGBISScmf/2Q==");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_protonmail_vcard() {
|
||||||
|
let contacts = parse_vcard(
|
||||||
|
"BEGIN:VCARD
|
||||||
|
VERSION:4.0
|
||||||
|
FN;PREF=1:Alice Wonderland
|
||||||
|
UID:proton-web-03747582-328d-38dc-5ddd-000000000000
|
||||||
|
ITEM1.EMAIL;PREF=1:alice@example.org
|
||||||
|
ITEM1.KEY;PREF=1:data:application/pgp-keys;base64,aaaaaaaaaaaaaaaaaaaaaaaaa
|
||||||
|
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
|
||||||
|
ITEM1.KEY;PREF=2:data:application/pgp-keys;base64,bbbbbbbbbbbbbbbbbbbbbbbbb
|
||||||
|
bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb
|
||||||
|
ITEM1.X-PM-ENCRYPT:true
|
||||||
|
ITEM1.X-PM-SIGN:true
|
||||||
|
END:VCARD",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(contacts.len(), 1);
|
||||||
|
assert_eq!(&contacts[0].addr, "alice@example.org");
|
||||||
|
assert_eq!(&contacts[0].authname, "Alice Wonderland");
|
||||||
|
assert_eq!(contacts[0].key.as_ref().unwrap(), "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa");
|
||||||
|
assert!(contacts[0].timestamp.is_err());
|
||||||
|
assert_eq!(contacts[0].profile_image, None);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Proton at some point slightly changed the format of their vcards.
|
||||||
|
/// This also tests unescaped commas in PHOTO and KEY (old Delta Chat format).
|
||||||
|
#[test]
|
||||||
|
fn test_protonmail_vcard2() {
|
||||||
|
let contacts = parse_vcard(
|
||||||
|
r"BEGIN:VCARD
|
||||||
|
VERSION:4.0
|
||||||
|
FN;PREF=1:Alice
|
||||||
|
PHOTO;PREF=1:data:image/jpeg;base64,/9aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
|
||||||
|
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
|
||||||
|
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaa/Z
|
||||||
|
REV:Invalid Date
|
||||||
|
ITEM1.EMAIL;PREF=1:alice@example.org
|
||||||
|
KEY;PREF=1:data:application/pgp-keys;base64,xsaaaaaaaaaaaaaaaaaaaaaaaaaaaa
|
||||||
|
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
|
||||||
|
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa==
|
||||||
|
UID:proton-web-aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa
|
||||||
|
END:VCARD",
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(contacts.len(), 1);
|
||||||
|
assert_eq!(&contacts[0].addr, "alice@example.org");
|
||||||
|
assert_eq!(&contacts[0].authname, "Alice");
|
||||||
|
assert_eq!(contacts[0].key.as_ref().unwrap(), "xsaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa==");
|
||||||
|
assert!(contacts[0].timestamp.is_err());
|
||||||
|
assert_eq!(contacts[0].profile_image.as_ref().unwrap(), "/9aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa/Z");
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "deltachat_ffi"
|
name = "deltachat_ffi"
|
||||||
version = "1.157.3"
|
version = "2.25.0"
|
||||||
description = "Deltachat FFI"
|
description = "Deltachat FFI"
|
||||||
edition = "2018"
|
edition = "2018"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
@@ -24,7 +24,6 @@ tokio = { workspace = true, features = ["rt-multi-thread"] }
|
|||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
thiserror = { workspace = true }
|
thiserror = { workspace = true }
|
||||||
rand = { workspace = true }
|
rand = { workspace = true }
|
||||||
once_cell = { workspace = true }
|
|
||||||
yerpc = { workspace = true, features = ["anyhow_expose"] }
|
yerpc = { workspace = true, features = ["anyhow_expose"] }
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,5 +1,6 @@
|
|||||||
<doxygenlayout version="1.0">
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<!-- Generated by doxygen 1.8.20 -->
|
<doxygenlayout version="2.0">
|
||||||
|
<!-- Generated by doxygen 1.13.2 -->
|
||||||
<!-- Navigation index tabs for HTML output -->
|
<!-- Navigation index tabs for HTML output -->
|
||||||
<navindex>
|
<navindex>
|
||||||
<tab type="mainpage" visible="yes" title=""/>
|
<tab type="mainpage" visible="yes" title=""/>
|
||||||
@@ -11,10 +12,16 @@
|
|||||||
</tab>
|
</tab>
|
||||||
<tab type="topics" visible="yes" title="Constants" intro="Here is a list of constants:"/>
|
<tab type="topics" visible="yes" title="Constants" intro="Here is a list of constants:"/>
|
||||||
<tab type="pages" visible="yes" title="" intro=""/>
|
<tab type="pages" visible="yes" title="" intro=""/>
|
||||||
|
<tab type="modules" visible="yes" title="" intro="">
|
||||||
|
<tab type="modulelist" visible="yes" title="" intro=""/>
|
||||||
|
<tab type="modulemembers" visible="yes" title="" intro=""/>
|
||||||
|
</tab>
|
||||||
<tab type="namespaces" visible="yes" title="">
|
<tab type="namespaces" visible="yes" title="">
|
||||||
<tab type="namespacelist" visible="yes" title="" intro=""/>
|
<tab type="namespacelist" visible="yes" title="" intro=""/>
|
||||||
<tab type="namespacemembers" visible="yes" title="" intro=""/>
|
<tab type="namespacemembers" visible="yes" title="" intro=""/>
|
||||||
</tab>
|
</tab>
|
||||||
|
<tab type="concepts" visible="yes" title="">
|
||||||
|
</tab>
|
||||||
<tab type="interfaces" visible="yes" title="">
|
<tab type="interfaces" visible="yes" title="">
|
||||||
<tab type="interfacelist" visible="yes" title="" intro=""/>
|
<tab type="interfacelist" visible="yes" title="" intro=""/>
|
||||||
<tab type="interfaceindex" visible="$ALPHABETICAL_INDEX" title=""/>
|
<tab type="interfaceindex" visible="$ALPHABETICAL_INDEX" title=""/>
|
||||||
@@ -35,4 +42,228 @@
|
|||||||
</tab>
|
</tab>
|
||||||
<tab type="examples" visible="yes" title="" intro=""/>
|
<tab type="examples" visible="yes" title="" intro=""/>
|
||||||
</navindex>
|
</navindex>
|
||||||
|
|
||||||
|
<!-- Layout definition for a class page -->
|
||||||
|
<class>
|
||||||
|
<briefdescription visible="yes"/>
|
||||||
|
<includes visible="$SHOW_HEADERFILE"/>
|
||||||
|
<inheritancegraph visible="yes"/>
|
||||||
|
<collaborationgraph visible="yes"/>
|
||||||
|
<memberdecl>
|
||||||
|
<nestedclasses visible="yes" title=""/>
|
||||||
|
<publictypes visible="yes" title=""/>
|
||||||
|
<services visible="yes" title=""/>
|
||||||
|
<interfaces visible="yes" title=""/>
|
||||||
|
<publicslots visible="yes" title=""/>
|
||||||
|
<signals visible="yes" title=""/>
|
||||||
|
<publicmethods visible="yes" title=""/>
|
||||||
|
<publicstaticmethods visible="yes" title=""/>
|
||||||
|
<publicattributes visible="yes" title=""/>
|
||||||
|
<publicstaticattributes visible="yes" title=""/>
|
||||||
|
<protectedtypes visible="yes" title=""/>
|
||||||
|
<protectedslots visible="yes" title=""/>
|
||||||
|
<protectedmethods visible="yes" title=""/>
|
||||||
|
<protectedstaticmethods visible="yes" title=""/>
|
||||||
|
<protectedattributes visible="yes" title=""/>
|
||||||
|
<protectedstaticattributes visible="yes" title=""/>
|
||||||
|
<packagetypes visible="yes" title=""/>
|
||||||
|
<packagemethods visible="yes" title=""/>
|
||||||
|
<packagestaticmethods visible="yes" title=""/>
|
||||||
|
<packageattributes visible="yes" title=""/>
|
||||||
|
<packagestaticattributes visible="yes" title=""/>
|
||||||
|
<properties visible="yes" title=""/>
|
||||||
|
<events visible="yes" title=""/>
|
||||||
|
<privatetypes visible="yes" title=""/>
|
||||||
|
<privateslots visible="yes" title=""/>
|
||||||
|
<privatemethods visible="yes" title=""/>
|
||||||
|
<privatestaticmethods visible="yes" title=""/>
|
||||||
|
<privateattributes visible="yes" title=""/>
|
||||||
|
<privatestaticattributes visible="yes" title=""/>
|
||||||
|
<friends visible="yes" title=""/>
|
||||||
|
<related visible="yes" title="" subtitle=""/>
|
||||||
|
<membergroups visible="yes"/>
|
||||||
|
</memberdecl>
|
||||||
|
<detaileddescription visible="yes" title=""/>
|
||||||
|
<memberdef>
|
||||||
|
<inlineclasses visible="yes" title=""/>
|
||||||
|
<typedefs visible="yes" title=""/>
|
||||||
|
<enums visible="yes" title=""/>
|
||||||
|
<services visible="yes" title=""/>
|
||||||
|
<interfaces visible="yes" title=""/>
|
||||||
|
<constructors visible="yes" title=""/>
|
||||||
|
<functions visible="yes" title=""/>
|
||||||
|
<related visible="yes" title=""/>
|
||||||
|
<variables visible="yes" title=""/>
|
||||||
|
<properties visible="yes" title=""/>
|
||||||
|
<events visible="yes" title=""/>
|
||||||
|
</memberdef>
|
||||||
|
<allmemberslink visible="yes"/>
|
||||||
|
<usedfiles visible="$SHOW_USED_FILES"/>
|
||||||
|
<authorsection visible="yes"/>
|
||||||
|
</class>
|
||||||
|
|
||||||
|
<!-- Layout definition for a namespace page -->
|
||||||
|
<namespace>
|
||||||
|
<briefdescription visible="yes"/>
|
||||||
|
<memberdecl>
|
||||||
|
<nestednamespaces visible="yes" title=""/>
|
||||||
|
<constantgroups visible="yes" title=""/>
|
||||||
|
<interfaces visible="yes" title=""/>
|
||||||
|
<classes visible="yes" title=""/>
|
||||||
|
<concepts visible="yes" title=""/>
|
||||||
|
<structs visible="yes" title=""/>
|
||||||
|
<exceptions visible="yes" title=""/>
|
||||||
|
<typedefs visible="yes" title=""/>
|
||||||
|
<sequences visible="yes" title=""/>
|
||||||
|
<dictionaries visible="yes" title=""/>
|
||||||
|
<enums visible="yes" title=""/>
|
||||||
|
<functions visible="yes" title=""/>
|
||||||
|
<variables visible="yes" title=""/>
|
||||||
|
<properties visible="yes" title=""/>
|
||||||
|
<membergroups visible="yes" visible="yes"/>
|
||||||
|
</memberdecl>
|
||||||
|
<detaileddescription visible="yes" title=""/>
|
||||||
|
<memberdef>
|
||||||
|
<inlineclasses visible="yes" title=""/>
|
||||||
|
<typedefs visible="yes" title=""/>
|
||||||
|
<sequences visible="yes" title=""/>
|
||||||
|
<dictionaries visible="yes" title=""/>
|
||||||
|
<enums visible="yes" title=""/>
|
||||||
|
<functions visible="yes" title=""/>
|
||||||
|
<variables visible="yes" title=""/>
|
||||||
|
<properties visible="yes" title=""/>
|
||||||
|
</memberdef>
|
||||||
|
<authorsection visible="yes"/>
|
||||||
|
</namespace>
|
||||||
|
|
||||||
|
<!-- Layout definition for a concept page -->
|
||||||
|
<concept>
|
||||||
|
<briefdescription visible="yes"/>
|
||||||
|
<includes visible="$SHOW_HEADERFILE"/>
|
||||||
|
<definition visible="yes" title=""/>
|
||||||
|
<detaileddescription visible="yes" title=""/>
|
||||||
|
<authorsection visible="yes"/>
|
||||||
|
</concept>
|
||||||
|
|
||||||
|
<!-- Layout definition for a file page -->
|
||||||
|
<file>
|
||||||
|
<briefdescription visible="yes"/>
|
||||||
|
<includes visible="$SHOW_INCLUDE_FILES"/>
|
||||||
|
<includegraph visible="yes"/>
|
||||||
|
<includedbygraph visible="yes"/>
|
||||||
|
<sourcelink visible="yes"/>
|
||||||
|
<memberdecl>
|
||||||
|
<interfaces visible="yes" title=""/>
|
||||||
|
<classes visible="yes" title=""/>
|
||||||
|
<structs visible="yes" title=""/>
|
||||||
|
<exceptions visible="yes" title=""/>
|
||||||
|
<namespaces visible="yes" title=""/>
|
||||||
|
<concepts visible="yes" title=""/>
|
||||||
|
<constantgroups visible="yes" title=""/>
|
||||||
|
<defines visible="yes" title=""/>
|
||||||
|
<typedefs visible="yes" title=""/>
|
||||||
|
<sequences visible="yes" title=""/>
|
||||||
|
<dictionaries visible="yes" title=""/>
|
||||||
|
<enums visible="yes" title=""/>
|
||||||
|
<functions visible="yes" title=""/>
|
||||||
|
<variables visible="yes" title=""/>
|
||||||
|
<properties visible="yes" title=""/>
|
||||||
|
<membergroups visible="yes" visible="yes"/>
|
||||||
|
</memberdecl>
|
||||||
|
<detaileddescription visible="yes" title=""/>
|
||||||
|
<memberdef>
|
||||||
|
<inlineclasses visible="yes" title=""/>
|
||||||
|
<defines visible="yes" title=""/>
|
||||||
|
<typedefs visible="yes" title=""/>
|
||||||
|
<sequences visible="yes" title=""/>
|
||||||
|
<dictionaries visible="yes" title=""/>
|
||||||
|
<enums visible="yes" title=""/>
|
||||||
|
<functions visible="yes" title=""/>
|
||||||
|
<variables visible="yes" title=""/>
|
||||||
|
<properties visible="yes" title=""/>
|
||||||
|
</memberdef>
|
||||||
|
<authorsection/>
|
||||||
|
</file>
|
||||||
|
|
||||||
|
<!-- Layout definition for a group page -->
|
||||||
|
<group>
|
||||||
|
<briefdescription visible="yes"/>
|
||||||
|
<groupgraph visible="yes"/>
|
||||||
|
<memberdecl>
|
||||||
|
<nestedgroups visible="yes" title=""/>
|
||||||
|
<modules visible="yes" title=""/>
|
||||||
|
<dirs visible="yes" title=""/>
|
||||||
|
<files visible="yes" title=""/>
|
||||||
|
<namespaces visible="yes" title=""/>
|
||||||
|
<concepts visible="yes" title=""/>
|
||||||
|
<classes visible="yes" title=""/>
|
||||||
|
<defines visible="yes" title=""/>
|
||||||
|
<typedefs visible="yes" title=""/>
|
||||||
|
<sequences visible="yes" title=""/>
|
||||||
|
<dictionaries visible="yes" title=""/>
|
||||||
|
<enums visible="yes" title=""/>
|
||||||
|
<enumvalues visible="yes" title=""/>
|
||||||
|
<functions visible="yes" title=""/>
|
||||||
|
<variables visible="yes" title=""/>
|
||||||
|
<signals visible="yes" title=""/>
|
||||||
|
<publicslots visible="yes" title=""/>
|
||||||
|
<protectedslots visible="yes" title=""/>
|
||||||
|
<privateslots visible="yes" title=""/>
|
||||||
|
<events visible="yes" title=""/>
|
||||||
|
<properties visible="yes" title=""/>
|
||||||
|
<friends visible="yes" title=""/>
|
||||||
|
<membergroups visible="yes"/>
|
||||||
|
</memberdecl>
|
||||||
|
<detaileddescription visible="yes" title=""/>
|
||||||
|
<memberdef>
|
||||||
|
<pagedocs/>
|
||||||
|
<inlineclasses visible="yes" title=""/>
|
||||||
|
<defines visible="yes" title=""/>
|
||||||
|
<typedefs visible="yes" title=""/>
|
||||||
|
<sequences visible="yes" title=""/>
|
||||||
|
<dictionaries visible="yes" title=""/>
|
||||||
|
<enums visible="yes" title=""/>
|
||||||
|
<enumvalues visible="yes" title=""/>
|
||||||
|
<functions visible="yes" title=""/>
|
||||||
|
<variables visible="yes" title=""/>
|
||||||
|
<signals visible="yes" title=""/>
|
||||||
|
<publicslots visible="yes" title=""/>
|
||||||
|
<protectedslots visible="yes" title=""/>
|
||||||
|
<privateslots visible="yes" title=""/>
|
||||||
|
<events visible="yes" title=""/>
|
||||||
|
<properties visible="yes" title=""/>
|
||||||
|
<friends visible="yes" title=""/>
|
||||||
|
</memberdef>
|
||||||
|
<authorsection visible="yes"/>
|
||||||
|
</group>
|
||||||
|
|
||||||
|
<!-- Layout definition for a C++20 module page -->
|
||||||
|
<module>
|
||||||
|
<briefdescription visible="yes"/>
|
||||||
|
<exportedmodules visible="yes"/>
|
||||||
|
<memberdecl>
|
||||||
|
<concepts visible="yes" title=""/>
|
||||||
|
<classes visible="yes" title=""/>
|
||||||
|
<enums visible="yes" title=""/>
|
||||||
|
<typedefs visible="yes" title=""/>
|
||||||
|
<functions visible="yes" title=""/>
|
||||||
|
<variables visible="yes" title=""/>
|
||||||
|
<membergroups visible="yes" title=""/>
|
||||||
|
</memberdecl>
|
||||||
|
<detaileddescription visible="yes" title=""/>
|
||||||
|
<memberdecl>
|
||||||
|
<files visible="yes"/>
|
||||||
|
</memberdecl>
|
||||||
|
</module>
|
||||||
|
|
||||||
|
<!-- Layout definition for a directory page -->
|
||||||
|
<directory>
|
||||||
|
<briefdescription visible="yes"/>
|
||||||
|
<directorygraph visible="yes"/>
|
||||||
|
<memberdecl>
|
||||||
|
<dirs visible="yes"/>
|
||||||
|
<files visible="yes"/>
|
||||||
|
</memberdecl>
|
||||||
|
<detaileddescription visible="yes" title=""/>
|
||||||
|
</directory>
|
||||||
</doxygenlayout>
|
</doxygenlayout>
|
||||||
|
|||||||
@@ -415,7 +415,6 @@ char* dc_get_blobdir (const dc_context_t* context);
|
|||||||
* As for `displayname` and `selfstatus`, also the avatar is sent to the recipients.
|
* As for `displayname` and `selfstatus`, also the avatar is sent to the recipients.
|
||||||
* To save traffic, however, the avatar is attached only as needed
|
* To save traffic, however, the avatar is attached only as needed
|
||||||
* and also recoded to a reasonable size.
|
* and also recoded to a reasonable size.
|
||||||
* - `e2ee_enabled` = 0=no end-to-end-encryption, 1=prefer end-to-end-encryption (default)
|
|
||||||
* - `mdns_enabled` = 0=do not send or request read receipts,
|
* - `mdns_enabled` = 0=do not send or request read receipts,
|
||||||
* 1=send and request read receipts
|
* 1=send and request read receipts
|
||||||
* default=send and request read receipts, only send but not request if `bot` is set
|
* default=send and request read receipts, only send but not request if `bot` is set
|
||||||
@@ -459,12 +458,6 @@ char* dc_get_blobdir (const dc_context_t* context);
|
|||||||
* The library uses the `media_quality` setting to use different defaults
|
* The library uses the `media_quality` setting to use different defaults
|
||||||
* for recoding images sent with type #DC_MSG_IMAGE.
|
* for recoding images sent with type #DC_MSG_IMAGE.
|
||||||
* If needed, recoding other file types is up to the UI.
|
* If needed, recoding other file types is up to the UI.
|
||||||
* - `webrtc_instance` = webrtc instance to use for videochats in the form
|
|
||||||
* `[basicwebrtc:|jitsi:]https://example.com/subdir#roomname=$ROOM`
|
|
||||||
* if the URL is prefixed by `basicwebrtc`, the server is assumed to be of the type
|
|
||||||
* https://github.com/cracker0dks/basicwebrtc which some UIs have native support for.
|
|
||||||
* The type `jitsi:` may be handled by external apps.
|
|
||||||
* If no type is prefixed, the videochat is handled completely in a browser.
|
|
||||||
* - `bot` = Set to "1" if this is a bot.
|
* - `bot` = Set to "1" if this is a bot.
|
||||||
* Prevents adding the "Device messages" and "Saved messages" chats,
|
* Prevents adding the "Device messages" and "Saved messages" chats,
|
||||||
* adds Auto-Submitted header to outgoing messages,
|
* adds Auto-Submitted header to outgoing messages,
|
||||||
@@ -503,13 +496,6 @@ char* dc_get_blobdir (const dc_context_t* context);
|
|||||||
* - `gossip_period` = How often to gossip Autocrypt keys in chats with multiple recipients, in
|
* - `gossip_period` = How often to gossip Autocrypt keys in chats with multiple recipients, in
|
||||||
* seconds. 2 days by default.
|
* seconds. 2 days by default.
|
||||||
* This is not supposed to be changed by UIs and only used for testing.
|
* This is not supposed to be changed by UIs and only used for testing.
|
||||||
* - `verified_one_on_one_chats` = Feature flag for verified 1:1 chats; the UI should set it
|
|
||||||
* to 1 if it supports verified 1:1 chats.
|
|
||||||
* Regardless of this setting, `dc_chat_is_protected()` returns true while the key is verified,
|
|
||||||
* and when the key changes, an info message is posted into the chat.
|
|
||||||
* 0=Nothing else happens when the key changes.
|
|
||||||
* 1=After the key changed, `dc_chat_can_send()` returns false and `dc_chat_is_protection_broken()` returns true
|
|
||||||
* until `dc_accept_chat()` is called.
|
|
||||||
* - `is_chatmail` = 1 if the the server is a chatmail server, 0 otherwise.
|
* - `is_chatmail` = 1 if the the server is a chatmail server, 0 otherwise.
|
||||||
* - `is_muted` = Whether a context is muted by the user.
|
* - `is_muted` = Whether a context is muted by the user.
|
||||||
* Muted contexts should not sound, vibrate or show notifications.
|
* Muted contexts should not sound, vibrate or show notifications.
|
||||||
@@ -583,11 +569,10 @@ int dc_set_stock_translation(dc_context_t* context, uint32_t stock_i
|
|||||||
/**
|
/**
|
||||||
* Set configuration values from a QR code.
|
* Set configuration values from a QR code.
|
||||||
* Before this function is called, dc_check_qr() should confirm the type of the
|
* Before this function is called, dc_check_qr() should confirm the type of the
|
||||||
* QR code is DC_QR_ACCOUNT, DC_QR_LOGIN or DC_QR_WEBRTC_INSTANCE.
|
* QR code is DC_QR_ACCOUNT or DC_QR_LOGIN.
|
||||||
*
|
*
|
||||||
* Internally, the function will call dc_set_config() with the appropriate keys,
|
* Internally, the function will call dc_set_config() with the appropriate keys,
|
||||||
* e.g. `addr` and `mail_pw` for DC_QR_ACCOUNT and DC_QR_LOGIN
|
* e.g. `addr` and `mail_pw` for DC_QR_ACCOUNT and DC_QR_LOGIN.
|
||||||
* or `webrtc_instance` for DC_QR_WEBRTC_INSTANCE.
|
|
||||||
*
|
*
|
||||||
* @memberof dc_context_t
|
* @memberof dc_context_t
|
||||||
* @param context The context object.
|
* @param context The context object.
|
||||||
@@ -1060,42 +1045,6 @@ void dc_send_edit_request (dc_context_t* context, uint32_t ms
|
|||||||
void dc_send_delete_request (dc_context_t* context, const uint32_t* msg_ids, int msg_cnt);
|
void dc_send_delete_request (dc_context_t* context, const uint32_t* msg_ids, int msg_cnt);
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send invitation to a videochat.
|
|
||||||
*
|
|
||||||
* This function reads the `webrtc_instance` config value,
|
|
||||||
* may check that the server is working in some way
|
|
||||||
* and creates a unique room for this chat, if needed doing a TOKEN roundtrip for that.
|
|
||||||
*
|
|
||||||
* After that, the function sends out a message that contains information to join the room:
|
|
||||||
*
|
|
||||||
* - To allow non-delta-clients to join the chat,
|
|
||||||
* the message contains a text-area with some descriptive text
|
|
||||||
* and a URL that can be opened in a supported browser to join the videochat.
|
|
||||||
*
|
|
||||||
* - delta-clients can get all information needed from
|
|
||||||
* the message object, using e.g.
|
|
||||||
* dc_msg_get_videochat_url() and check dc_msg_get_viewtype() for #DC_MSG_VIDEOCHAT_INVITATION.
|
|
||||||
*
|
|
||||||
* dc_send_videochat_invitation() is blocking and may take a while,
|
|
||||||
* so the UIs will typically call the function from within a thread.
|
|
||||||
* Moreover, UIs will typically enter the room directly without an additional click on the message,
|
|
||||||
* for this purpose, the function returns the message id directly.
|
|
||||||
*
|
|
||||||
* As for other messages sent, this function
|
|
||||||
* sends the event #DC_EVENT_MSGS_CHANGED on success, the message has a delivery state, and so on.
|
|
||||||
* The recipient will get noticed by the call as usual by #DC_EVENT_INCOMING_MSG or #DC_EVENT_MSGS_CHANGED,
|
|
||||||
* However, UIs might some things differently, e.g. play a different sound.
|
|
||||||
*
|
|
||||||
* @memberof dc_context_t
|
|
||||||
* @param context The context object.
|
|
||||||
* @param chat_id The chat to start a videochat for.
|
|
||||||
* @return The ID of the message sent out
|
|
||||||
* or 0 for errors.
|
|
||||||
*/
|
|
||||||
uint32_t dc_send_videochat_invitation (dc_context_t* context, uint32_t chat_id);
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A webxdc instance sends a status update to its other members.
|
* A webxdc instance sends a status update to its other members.
|
||||||
*
|
*
|
||||||
@@ -1222,6 +1171,117 @@ void dc_set_webxdc_integration (dc_context_t* context, const char* f
|
|||||||
uint32_t dc_init_webxdc_integration (dc_context_t* context, uint32_t chat_id);
|
uint32_t dc_init_webxdc_integration (dc_context_t* context, uint32_t chat_id);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start an outgoing call.
|
||||||
|
* This sends a message of type #DC_MSG_CALL with all relevant information to the callee,
|
||||||
|
* who will get informed by an #DC_EVENT_INCOMING_CALL event and rings.
|
||||||
|
*
|
||||||
|
* Possible actions during ringing:
|
||||||
|
*
|
||||||
|
* - caller cancels the call using dc_end_call():
|
||||||
|
* callee receives #DC_EVENT_CALL_ENDED and has a "Missed call"
|
||||||
|
*
|
||||||
|
* - callee accepts using dc_accept_incoming_call():
|
||||||
|
* caller receives #DC_EVENT_OUTGOING_CALL_ACCEPTED.
|
||||||
|
* callee's devices receive #DC_EVENT_INCOMING_CALL_ACCEPTED, call starts
|
||||||
|
*
|
||||||
|
* - callee declines using dc_end_call():
|
||||||
|
* caller receives #DC_EVENT_CALL_ENDED and has a "Declinced Call".
|
||||||
|
* callee's other devices receive #DC_EVENT_CALL_ENDED and have a "Canceled Call",
|
||||||
|
*
|
||||||
|
* - callee is already in a call:
|
||||||
|
* what to do depends on the capabilities of UI to handle calls.
|
||||||
|
* if UI cannot handle multiple calls, an easy approach would be to decline the new call automatically
|
||||||
|
* and make that visble to the user in the call, e.g. by a notification
|
||||||
|
*
|
||||||
|
* - timeout:
|
||||||
|
* after 1 minute without action,
|
||||||
|
* caller and callee receive #DC_EVENT_CALL_ENDED
|
||||||
|
* to prevent endless ringing of callee
|
||||||
|
* in case caller got offline without being able to send cancellation message.
|
||||||
|
* for caller, this is a "Canceled call";
|
||||||
|
* for callee, this is a "Missed call"
|
||||||
|
*
|
||||||
|
* Actions during the call:
|
||||||
|
*
|
||||||
|
* - caller ends the call using dc_end_call():
|
||||||
|
* callee receives #DC_EVENT_CALL_ENDED
|
||||||
|
*
|
||||||
|
* - callee ends the call using dc_end_call():
|
||||||
|
* caller receives #DC_EVENT_CALL_ENDED
|
||||||
|
*
|
||||||
|
* Contact request handling:
|
||||||
|
*
|
||||||
|
* - placing or accepting calls implies accepting contact requests
|
||||||
|
*
|
||||||
|
* - ending a call does not accept a contact request;
|
||||||
|
* instead, the call will timeout on all affected devices.
|
||||||
|
*
|
||||||
|
* Note, that the events are for updating the call screen,
|
||||||
|
* possible status messages are added and updated as usual, including the known events.
|
||||||
|
* In the UI, the sorted chatlist is used as an overview about calls as well as messages.
|
||||||
|
* To place a call with a contact that has no chat yet, use dc_create_chat_by_contact_id() first.
|
||||||
|
*
|
||||||
|
* UI will usually allow only one call at the same time,
|
||||||
|
* this has to be tracked by UI across profile, the core does not track this.
|
||||||
|
*
|
||||||
|
* @memberof dc_context_t
|
||||||
|
* @param context The context object.
|
||||||
|
* @param chat_id The chat to place a call for.
|
||||||
|
* This needs to be a one-to-one chat.
|
||||||
|
* @param place_call_info any data that other devices receive
|
||||||
|
* in #DC_EVENT_INCOMING_CALL.
|
||||||
|
* @return ID of the system message announcing the call.
|
||||||
|
*/
|
||||||
|
uint32_t dc_place_outgoing_call (dc_context_t* context, uint32_t chat_id, const char* place_call_info);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Accept incoming call.
|
||||||
|
*
|
||||||
|
* This implicitly accepts the contact request, if not yet done.
|
||||||
|
* All affected devices will receive
|
||||||
|
* either #DC_EVENT_OUTGOING_CALL_ACCEPTED or #DC_EVENT_INCOMING_CALL_ACCEPTED.
|
||||||
|
*
|
||||||
|
* If the call is already accepted or ended, nothing happens.
|
||||||
|
* If the chat is a contact request, it is accepted implicitly.
|
||||||
|
*
|
||||||
|
* @memberof dc_context_t
|
||||||
|
* @param context The context object.
|
||||||
|
* @param msg_id The ID of the call to accept.
|
||||||
|
* This is the ID reported by #DC_EVENT_INCOMING_CALL
|
||||||
|
* and equals to the ID of the corresponding info message.
|
||||||
|
* @param accept_call_info any data that other devices receive
|
||||||
|
* in #DC_EVENT_OUTGOING_CALL_ACCEPTED.
|
||||||
|
* @return 1=success, 0=error
|
||||||
|
*/
|
||||||
|
int dc_accept_incoming_call (dc_context_t* context, uint32_t msg_id, const char* accept_call_info);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* End incoming or outgoing call.
|
||||||
|
*
|
||||||
|
* For unaccepted calls ended by the caller, this is a "cancellation".
|
||||||
|
* Unaccepted calls ended by the callee are a "decline".
|
||||||
|
* If the call was accepted, this is a "hangup".
|
||||||
|
*
|
||||||
|
* All participant devices get informed about the ended call via #DC_EVENT_CALL_ENDED unless they are contact requests.
|
||||||
|
* For contact requests, the call times out on all other affected devices.
|
||||||
|
*
|
||||||
|
* If the message ID is wrong or does not exist for whatever reasons, nothing happens.
|
||||||
|
* Therefore, and for resilience, UI should remove the call UI directly when calling
|
||||||
|
* this function and not only on the event.
|
||||||
|
*
|
||||||
|
* If the call is already ended, nothing happens.
|
||||||
|
*
|
||||||
|
* @memberof dc_context_t
|
||||||
|
* @param context The context object.
|
||||||
|
* @param msg_id the ID of the call.
|
||||||
|
* @return 1=success, 0=error
|
||||||
|
*/
|
||||||
|
int dc_end_call (dc_context_t* context, uint32_t msg_id);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save a draft for a chat in the database.
|
* Save a draft for a chat in the database.
|
||||||
*
|
*
|
||||||
@@ -1339,12 +1399,14 @@ dc_msg_t* dc_get_draft (dc_context_t* context, uint32_t ch
|
|||||||
* Optionally, some special markers added to the ID array may help to
|
* Optionally, some special markers added to the ID array may help to
|
||||||
* implement virtual lists.
|
* implement virtual lists.
|
||||||
*
|
*
|
||||||
|
* To get the concrete time of the message, use dc_array_get_timestamp().
|
||||||
|
*
|
||||||
* @memberof dc_context_t
|
* @memberof dc_context_t
|
||||||
* @param context The context object as returned from dc_context_new().
|
* @param context The context object as returned from dc_context_new().
|
||||||
* @param chat_id The chat ID of which the messages IDs should be queried.
|
* @param chat_id The chat ID of which the messages IDs should be queried.
|
||||||
* @param flags If set to DC_GCM_ADDDAYMARKER, the marker DC_MSG_ID_DAYMARKER will
|
* @param flags If set to DC_GCM_ADDDAYMARKER, the marker DC_MSG_ID_DAYMARKER will
|
||||||
* be added before each day (regarding the local timezone). Set this to 0 if you do not want this behaviour.
|
* be added before each day (regarding the local timezone). Set this to 0 if you do not want this behaviour.
|
||||||
* To get the concrete time of the marker, use dc_array_get_timestamp().
|
* The day marker timestamp is the midnight one for the corresponding (following) day in the local timezone.
|
||||||
* If set to DC_GCM_INFO_ONLY, only system messages will be returned, can be combined with DC_GCM_ADDDAYMARKER.
|
* If set to DC_GCM_INFO_ONLY, only system messages will be returned, can be combined with DC_GCM_ADDDAYMARKER.
|
||||||
* @param marker1before Deprecated, set this to 0.
|
* @param marker1before Deprecated, set this to 0.
|
||||||
* @return Array of message IDs, must be dc_array_unref()'d when no longer used.
|
* @return Array of message IDs, must be dc_array_unref()'d when no longer used.
|
||||||
@@ -1701,9 +1763,7 @@ dc_chat_t* dc_get_chat (dc_context_t* context, uint32_t ch
|
|||||||
*
|
*
|
||||||
* @memberof dc_context_t
|
* @memberof dc_context_t
|
||||||
* @param context The context object.
|
* @param context The context object.
|
||||||
* @param protect If set to 1 the function creates group with protection initially enabled.
|
* @param protect Deprecated 2025-08-31, ignored.
|
||||||
* Only verified members are allowed in these groups
|
|
||||||
* and end-to-end-encryption is always enabled.
|
|
||||||
* @param name The name of the group chat to create.
|
* @param name The name of the group chat to create.
|
||||||
* The name may be changed later using dc_set_chat_name().
|
* The name may be changed later using dc_set_chat_name().
|
||||||
* To find out the name of a group later, see dc_chat_get_name()
|
* To find out the name of a group later, see dc_chat_get_name()
|
||||||
@@ -2094,9 +2154,19 @@ int dc_may_be_valid_addr (const char* addr);
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if an e-mail address belongs to a known and unblocked contact.
|
* Looks up a known and unblocked contact with a given e-mail address.
|
||||||
* To get a list of all known and unblocked contacts, use dc_get_contacts().
|
* To get a list of all known and unblocked contacts, use dc_get_contacts().
|
||||||
*
|
*
|
||||||
|
* **POTENTIAL SECURITY ISSUE**: If there are multiple contacts with this address
|
||||||
|
* (e.g. an address-contact and a key-contact),
|
||||||
|
* this looks up the most recently seen contact,
|
||||||
|
* i.e. which contact is returned depends on which contact last sent a message.
|
||||||
|
* If the user just clicked on a mailto: link, then this is the best thing you can do.
|
||||||
|
* But **DO NOT** internally represent contacts by their email address
|
||||||
|
* and do not use this function to look them up;
|
||||||
|
* otherwise this function will sometimes look up the wrong contact.
|
||||||
|
* Instead, you should internally represent contacts by their ids.
|
||||||
|
*
|
||||||
* To validate an e-mail address independently of the contact database
|
* To validate an e-mail address independently of the contact database
|
||||||
* use dc_may_be_valid_addr().
|
* use dc_may_be_valid_addr().
|
||||||
*
|
*
|
||||||
@@ -2118,6 +2188,13 @@ uint32_t dc_lookup_contact_id_by_addr (dc_context_t* context, const char*
|
|||||||
* To add a number of contacts, see dc_add_address_book() which is much faster for adding
|
* To add a number of contacts, see dc_add_address_book() which is much faster for adding
|
||||||
* a bunch of addresses.
|
* a bunch of addresses.
|
||||||
*
|
*
|
||||||
|
* This will always create or look up an address-contact,
|
||||||
|
* i.e. a contact identified by an email address,
|
||||||
|
* with all messages sent to and from this contact being unencrypted.
|
||||||
|
* If the user just clicked on an email address,
|
||||||
|
* you should first check `lookup_contact_id_by_addr`,
|
||||||
|
* and only if there is no contact yet, call this function here.
|
||||||
|
*
|
||||||
* May result in a #DC_EVENT_CONTACTS_CHANGED event.
|
* May result in a #DC_EVENT_CONTACTS_CHANGED event.
|
||||||
*
|
*
|
||||||
* @memberof dc_context_t
|
* @memberof dc_context_t
|
||||||
@@ -2132,8 +2209,12 @@ uint32_t dc_lookup_contact_id_by_addr (dc_context_t* context, const char*
|
|||||||
uint32_t dc_create_contact (dc_context_t* context, const char* name, const char* addr);
|
uint32_t dc_create_contact (dc_context_t* context, const char* name, const char* addr);
|
||||||
|
|
||||||
|
|
||||||
#define DC_GCL_VERIFIED_ONLY 0x01
|
|
||||||
|
// Deprecated 2025-05-20, setting this flag is a no-op.
|
||||||
|
#define DC_GCL_DEPRECATED_VERIFIED_ONLY 0x01
|
||||||
|
|
||||||
#define DC_GCL_ADD_SELF 0x02
|
#define DC_GCL_ADD_SELF 0x02
|
||||||
|
#define DC_GCL_ADDRESS 0x04
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -2162,17 +2243,40 @@ uint32_t dc_create_contact (dc_context_t* context, const char*
|
|||||||
int dc_add_address_book (dc_context_t* context, const char* addr_book);
|
int dc_add_address_book (dc_context_t* context, const char* addr_book);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make a vCard.
|
||||||
|
*
|
||||||
|
* @memberof dc_context_t
|
||||||
|
* @param context The context object.
|
||||||
|
* @param contact_id The ID of the contact to make the vCard of.
|
||||||
|
* @return vCard, must be released using dc_str_unref() after usage.
|
||||||
|
*/
|
||||||
|
char* dc_make_vcard (dc_context_t* context, uint32_t contact_id);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Import a vCard.
|
||||||
|
*
|
||||||
|
* @memberof dc_context_t
|
||||||
|
* @param context The context object.
|
||||||
|
* @param vcard vCard contents.
|
||||||
|
* @return Returns the IDs of the contacts in the order they appear in the vCard.
|
||||||
|
* Must be dc_array_unref()'d after usage.
|
||||||
|
*/
|
||||||
|
dc_array_t* dc_import_vcard (dc_context_t* context, const char* vcard);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns known and unblocked contacts.
|
* Returns known and unblocked contacts.
|
||||||
*
|
*
|
||||||
* To get information about a single contact, see dc_get_contact().
|
* To get information about a single contact, see dc_get_contact().
|
||||||
|
* By default, key-contacts are listed.
|
||||||
*
|
*
|
||||||
* @memberof dc_context_t
|
* @memberof dc_context_t
|
||||||
* @param context The context object.
|
* @param context The context object.
|
||||||
* @param flags A combination of flags:
|
* @param flags A combination of flags:
|
||||||
* - if the flag DC_GCL_ADD_SELF is set, SELF is added to the list unless filtered by other parameters
|
* - DC_GCL_ADD_SELF: SELF is added to the list unless filtered by other parameters
|
||||||
* - if the flag DC_GCL_VERIFIED_ONLY is set, only verified contacts are returned.
|
* - DC_GCL_ADDRESS: List address-contacts instead of key-contacts.
|
||||||
* if DC_GCL_VERIFIED_ONLY is not set, verified and unverified contacts are returned.
|
|
||||||
* @param query A string to filter the list. Typically used to implement an
|
* @param query A string to filter the list. Typically used to implement an
|
||||||
* incremental search. NULL for no filtering.
|
* incremental search. NULL for no filtering.
|
||||||
* @return An array containing all contact IDs. Must be dc_array_unref()'d
|
* @return An array containing all contact IDs. Must be dc_array_unref()'d
|
||||||
@@ -2459,6 +2563,7 @@ void dc_stop_ongoing_process (dc_context_t* context);
|
|||||||
|
|
||||||
#define DC_QR_ASK_VERIFYCONTACT 200 // id=contact
|
#define DC_QR_ASK_VERIFYCONTACT 200 // id=contact
|
||||||
#define DC_QR_ASK_VERIFYGROUP 202 // text1=groupname
|
#define DC_QR_ASK_VERIFYGROUP 202 // text1=groupname
|
||||||
|
#define DC_QR_ASK_VERIFYBROADCAST 204 // text1=broadcast name
|
||||||
#define DC_QR_FPR_OK 210 // id=contact
|
#define DC_QR_FPR_OK 210 // id=contact
|
||||||
#define DC_QR_FPR_MISMATCH 220 // id=contact
|
#define DC_QR_FPR_MISMATCH 220 // id=contact
|
||||||
#define DC_QR_FPR_WITHOUT_ADDR 230 // test1=formatted fingerprint
|
#define DC_QR_FPR_WITHOUT_ADDR 230 // test1=formatted fingerprint
|
||||||
@@ -2466,7 +2571,6 @@ void dc_stop_ongoing_process (dc_context_t* context);
|
|||||||
#define DC_QR_BACKUP 251 // deprecated
|
#define DC_QR_BACKUP 251 // deprecated
|
||||||
#define DC_QR_BACKUP2 252
|
#define DC_QR_BACKUP2 252
|
||||||
#define DC_QR_BACKUP_TOO_NEW 255
|
#define DC_QR_BACKUP_TOO_NEW 255
|
||||||
#define DC_QR_WEBRTC_INSTANCE 260 // text1=domain, text2=instance pattern
|
|
||||||
#define DC_QR_PROXY 271 // text1=address (e.g. "127.0.0.1:9050")
|
#define DC_QR_PROXY 271 // text1=address (e.g. "127.0.0.1:9050")
|
||||||
#define DC_QR_ADDR 320 // id=contact
|
#define DC_QR_ADDR 320 // id=contact
|
||||||
#define DC_QR_TEXT 330 // text1=text
|
#define DC_QR_TEXT 330 // text1=text
|
||||||
@@ -2492,8 +2596,9 @@ void dc_stop_ongoing_process (dc_context_t* context);
|
|||||||
* ask whether to verify the contact;
|
* ask whether to verify the contact;
|
||||||
* if so, start the protocol with dc_join_securejoin().
|
* if so, start the protocol with dc_join_securejoin().
|
||||||
*
|
*
|
||||||
* - DC_QR_ASK_VERIFYGROUP with dc_lot_t::text1=Group name:
|
* - DC_QR_ASK_VERIFYGROUP or DC_QR_ASK_VERIFYBROADCAST
|
||||||
* ask whether to join the group;
|
* with dc_lot_t::text1=Group name:
|
||||||
|
* ask whether to join the chat;
|
||||||
* if so, start the protocol with dc_join_securejoin().
|
* if so, start the protocol with dc_join_securejoin().
|
||||||
*
|
*
|
||||||
* - DC_QR_FPR_OK with dc_lot_t::id=Contact ID:
|
* - DC_QR_FPR_OK with dc_lot_t::id=Contact ID:
|
||||||
@@ -2520,10 +2625,6 @@ void dc_stop_ongoing_process (dc_context_t* context);
|
|||||||
* show a hint to the user that this backup comes from a newer Delta Chat version
|
* show a hint to the user that this backup comes from a newer Delta Chat version
|
||||||
* and this device needs an update
|
* and this device needs an update
|
||||||
*
|
*
|
||||||
* - DC_QR_WEBRTC_INSTANCE with dc_lot_t::text1=domain:
|
|
||||||
* ask the user if they want to use the given service for video chats;
|
|
||||||
* if so, call dc_set_config_from_qr().
|
|
||||||
*
|
|
||||||
* - DC_QR_PROXY with dc_lot_t::text1=address:
|
* - DC_QR_PROXY with dc_lot_t::text1=address:
|
||||||
* ask the user if they want to use the given proxy.
|
* ask the user if they want to use the given proxy.
|
||||||
* if so, call dc_set_config_from_qr() and restart I/O.
|
* if so, call dc_set_config_from_qr() and restart I/O.
|
||||||
@@ -2580,7 +2681,8 @@ dc_lot_t* dc_check_qr (dc_context_t* context, const char*
|
|||||||
* Get QR code text that will offer an Setup-Contact or Verified-Group invitation.
|
* Get QR code text that will offer an Setup-Contact or Verified-Group invitation.
|
||||||
*
|
*
|
||||||
* The scanning device will pass the scanned content to dc_check_qr() then;
|
* The scanning device will pass the scanned content to dc_check_qr() then;
|
||||||
* if dc_check_qr() returns DC_QR_ASK_VERIFYCONTACT or DC_QR_ASK_VERIFYGROUP
|
* if dc_check_qr() returns
|
||||||
|
* DC_QR_ASK_VERIFYCONTACT, DC_QR_ASK_VERIFYGROUP or DC_QR_ASK_VERIFYBROADCAST
|
||||||
* an out-of-band-verification can be joined using dc_join_securejoin()
|
* an out-of-band-verification can be joined using dc_join_securejoin()
|
||||||
*
|
*
|
||||||
* The returned text will also work as a normal https:-link,
|
* The returned text will also work as a normal https:-link,
|
||||||
@@ -2621,7 +2723,7 @@ char* dc_get_securejoin_qr_svg (dc_context_t* context, uint32_
|
|||||||
* Continue a Setup-Contact or Verified-Group-Invite protocol
|
* Continue a Setup-Contact or Verified-Group-Invite protocol
|
||||||
* started on another device with dc_get_securejoin_qr().
|
* started on another device with dc_get_securejoin_qr().
|
||||||
* This function is typically called when dc_check_qr() returns
|
* This function is typically called when dc_check_qr() returns
|
||||||
* lot.state=DC_QR_ASK_VERIFYCONTACT or lot.state=DC_QR_ASK_VERIFYGROUP.
|
* lot.state=DC_QR_ASK_VERIFYCONTACT, lot.state=DC_QR_ASK_VERIFYGROUP or lot.state=DC_QR_ASK_VERIFYBROADCAST
|
||||||
*
|
*
|
||||||
* The function returns immediately and the handshake runs in background,
|
* The function returns immediately and the handshake runs in background,
|
||||||
* sending and receiving several messages.
|
* sending and receiving several messages.
|
||||||
@@ -3789,49 +3891,29 @@ int dc_chat_can_send (const dc_chat_t* chat);
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if a chat is protected.
|
* Deprecated, always returns 0.
|
||||||
*
|
|
||||||
* End-to-end encryption is guaranteed in protected chats
|
|
||||||
* and only verified contacts
|
|
||||||
* as determined by dc_contact_is_verified()
|
|
||||||
* can be added to protected chats.
|
|
||||||
*
|
|
||||||
* Protected chats are created using dc_create_group_chat()
|
|
||||||
* by setting the 'protect' parameter to 1.
|
|
||||||
* 1:1 chats become protected or unprotected automatically
|
|
||||||
* if `verified_one_on_one_chats` setting is enabled.
|
|
||||||
*
|
|
||||||
* UI should display a green checkmark
|
|
||||||
* in the chat title,
|
|
||||||
* in the chatlist item
|
|
||||||
* and in the chat profile
|
|
||||||
* if chat protection is enabled.
|
|
||||||
*
|
*
|
||||||
* @memberof dc_chat_t
|
* @memberof dc_chat_t
|
||||||
* @param chat The chat object.
|
* @param chat The chat object.
|
||||||
* @return 1=chat protected, 0=chat is not protected.
|
* @return Always 0.
|
||||||
|
* @deprecated 2025-09-09
|
||||||
*/
|
*/
|
||||||
int dc_chat_is_protected (const dc_chat_t* chat);
|
int dc_chat_is_protected (const dc_chat_t* chat);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if the chat was protected, and then an incoming message broke this protection.
|
* Check if the chat is encrypted.
|
||||||
*
|
*
|
||||||
* This function is only useful if the UI enabled the `verified_one_on_one_chats` feature flag,
|
* 1:1 chats with key-contacts and group chats with key-contacts
|
||||||
* otherwise it will return false for all chats.
|
* are encrypted.
|
||||||
|
* 1:1 chats with emails contacts and ad-hoc groups
|
||||||
|
* created for email threads are not encrypted.
|
||||||
*
|
*
|
||||||
* 1:1 chats are automatically set as protected when a contact is verified.
|
|
||||||
* When a message comes in that is not encrypted / signed correctly,
|
|
||||||
* the chat is automatically set as unprotected again.
|
|
||||||
* dc_chat_is_protection_broken() will return true until dc_accept_chat() is called.
|
|
||||||
*
|
|
||||||
* The UI should let the user confirm that this is OK with a message like
|
|
||||||
* `Bob sent a message from another device. Tap to learn more` and then call dc_accept_chat().
|
|
||||||
* @memberof dc_chat_t
|
* @memberof dc_chat_t
|
||||||
* @param chat The chat object.
|
* @param chat The chat object.
|
||||||
* @return 1=chat protection broken, 0=otherwise.
|
* @return 1=chat is encrypted, 0=chat is not encrypted.
|
||||||
*/
|
*/
|
||||||
int dc_chat_is_protection_broken (const dc_chat_t* chat);
|
int dc_chat_is_encrypted (const dc_chat_t *chat);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -4261,11 +4343,16 @@ int dc_msg_get_duration (const dc_msg_t* msg);
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if a padlock should be shown beside the message.
|
* Check if message was correctly encrypted and signed.
|
||||||
|
*
|
||||||
|
* Historically, UIs showed a small padlock on the message then.
|
||||||
|
* Today, the UIs should instead
|
||||||
|
* show a small email-icon on the message if the message is not encrypted or signed,
|
||||||
|
* and nothing otherwise.
|
||||||
*
|
*
|
||||||
* @memberof dc_msg_t
|
* @memberof dc_msg_t
|
||||||
* @param msg The message object.
|
* @param msg The message object.
|
||||||
* @return 1=padlock should be shown beside message, 0=do not show a padlock beside the message.
|
* @return 1=message correctly encrypted and signed, no need to show anything; 0=show email-icon beside the message.
|
||||||
*/
|
*/
|
||||||
int dc_msg_get_showpadlock (const dc_msg_t* msg);
|
int dc_msg_get_showpadlock (const dc_msg_t* msg);
|
||||||
|
|
||||||
@@ -4483,12 +4570,21 @@ int dc_msg_is_info (const dc_msg_t* msg);
|
|||||||
* UIs can display e.g. an icon based upon the type.
|
* UIs can display e.g. an icon based upon the type.
|
||||||
*
|
*
|
||||||
* Currently, the following types are defined:
|
* Currently, the following types are defined:
|
||||||
* - DC_INFO_PROTECTION_ENABLED (11) - Info-message for "Chat is now protected"
|
* - DC_INFO_GROUP_NAME_CHANGED (2) - "Group name changd from OLD to BY by CONTACT"
|
||||||
* - DC_INFO_PROTECTION_DISABLED (12) - Info-message for "Chat is no longer protected"
|
* - DC_INFO_GROUP_IMAGE_CHANGED (3) - "Group image changd by CONTACT"
|
||||||
|
* - DC_INFO_MEMBER_ADDED_TO_GROUP (4) - "Member CONTACT added by OTHER_CONTACT"
|
||||||
|
* - DC_INFO_MEMBER_REMOVED_FROM_GROUP (5) - "Member CONTACT removed by OTHER_CONTACT"
|
||||||
|
* - DC_INFO_EPHEMERAL_TIMER_CHANGED (10) - "Disappearing messages CHANGED_TO by CONTACT"
|
||||||
|
* - DC_INFO_PROTECTION_ENABLED (11) - Info-message for "Chat is protected"
|
||||||
* - DC_INFO_INVALID_UNENCRYPTED_MAIL (13) - Info-message for "Provider requires end-to-end encryption which is not setup yet",
|
* - DC_INFO_INVALID_UNENCRYPTED_MAIL (13) - Info-message for "Provider requires end-to-end encryption which is not setup yet",
|
||||||
* the UI should change the corresponding string using #DC_STR_INVALID_UNENCRYPTED_MAIL
|
* the UI should change the corresponding string using #DC_STR_INVALID_UNENCRYPTED_MAIL
|
||||||
* and also offer a way to fix the encryption, eg. by a button offering a QR scan
|
* and also offer a way to fix the encryption, eg. by a button offering a QR scan
|
||||||
* - DC_INFO_WEBXDC_INFO_MESSAGE (32) - Info-message created by webxdc app sending `update.info`
|
* - DC_INFO_WEBXDC_INFO_MESSAGE (32) - Info-message created by webxdc app sending `update.info`
|
||||||
|
* - DC_INFO_CHAT_E2EE (50) - Info-message for "Chat is end-to-end-encrypted"
|
||||||
|
*
|
||||||
|
* For the messages that refer to a CONTACT,
|
||||||
|
* dc_msg_get_info_contact_id() returns the contact ID.
|
||||||
|
* The UI should open the contact's profile when tapping the info message.
|
||||||
*
|
*
|
||||||
* Even when you display an icon,
|
* Even when you display an icon,
|
||||||
* you should still display the text of the informational message using dc_msg_get_text()
|
* you should still display the text of the informational message using dc_msg_get_text()
|
||||||
@@ -4502,6 +4598,29 @@ int dc_msg_is_info (const dc_msg_t* msg);
|
|||||||
int dc_msg_get_info_type (const dc_msg_t* msg);
|
int dc_msg_get_info_type (const dc_msg_t* msg);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the contact ID of the profile to open when tapping the info message.
|
||||||
|
*
|
||||||
|
* - For DC_INFO_MEMBER_ADDED_TO_GROUP and DC_INFO_MEMBER_REMOVED_FROM_GROUP,
|
||||||
|
* this is the contact being added/removed.
|
||||||
|
* The contact that did the adding/removal is usually only a tap away
|
||||||
|
* (as introducer and/or atop of the memberlist),
|
||||||
|
* and usually more known anyways.
|
||||||
|
* - For DC_INFO_GROUP_NAME_CHANGED, DC_INFO_GROUP_IMAGE_CHANGED and DC_INFO_EPHEMERAL_TIMER_CHANGED
|
||||||
|
* this is the contact who did the change.
|
||||||
|
*
|
||||||
|
* No need to check additionally for dc_msg_get_info_type(),
|
||||||
|
* unless you e.g. want to show the info message in another style.
|
||||||
|
*
|
||||||
|
* @memberof dc_msg_t
|
||||||
|
* @param msg The message object.
|
||||||
|
* @return If the info message refers to a contact,
|
||||||
|
* this contact ID or DC_CONTACT_ID_SELF is returned.
|
||||||
|
* Otherwise 0.
|
||||||
|
*/
|
||||||
|
uint32_t dc_msg_get_info_contact_id (const dc_msg_t* msg);
|
||||||
|
|
||||||
|
|
||||||
// DC_INFO* uses the same values as SystemMessage in rust-land
|
// DC_INFO* uses the same values as SystemMessage in rust-land
|
||||||
#define DC_INFO_UNKNOWN 0
|
#define DC_INFO_UNKNOWN 0
|
||||||
#define DC_INFO_GROUP_NAME_CHANGED 2
|
#define DC_INFO_GROUP_NAME_CHANGED 2
|
||||||
@@ -4514,9 +4633,10 @@ int dc_msg_get_info_type (const dc_msg_t* msg);
|
|||||||
#define DC_INFO_LOCATION_ONLY 9
|
#define DC_INFO_LOCATION_ONLY 9
|
||||||
#define DC_INFO_EPHEMERAL_TIMER_CHANGED 10
|
#define DC_INFO_EPHEMERAL_TIMER_CHANGED 10
|
||||||
#define DC_INFO_PROTECTION_ENABLED 11
|
#define DC_INFO_PROTECTION_ENABLED 11
|
||||||
#define DC_INFO_PROTECTION_DISABLED 12
|
#define DC_INFO_PROTECTION_DISABLED 12 // deprecated 2025-07
|
||||||
#define DC_INFO_INVALID_UNENCRYPTED_MAIL 13
|
#define DC_INFO_INVALID_UNENCRYPTED_MAIL 13
|
||||||
#define DC_INFO_WEBXDC_INFO_MESSAGE 32
|
#define DC_INFO_WEBXDC_INFO_MESSAGE 32
|
||||||
|
#define DC_INFO_CHAT_E2EE 50
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -4571,22 +4691,6 @@ int dc_msg_is_setupmessage (const dc_msg_t* msg);
|
|||||||
char* dc_msg_get_setupcodebegin (const dc_msg_t* msg);
|
char* dc_msg_get_setupcodebegin (const dc_msg_t* msg);
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get URL of a videochat invitation.
|
|
||||||
*
|
|
||||||
* Videochat invitations are sent out using dc_send_videochat_invitation()
|
|
||||||
* and dc_msg_get_viewtype() returns #DC_MSG_VIDEOCHAT_INVITATION for such invitations.
|
|
||||||
*
|
|
||||||
* @memberof dc_msg_t
|
|
||||||
* @param msg The message object.
|
|
||||||
* @return If the message contains a videochat invitation,
|
|
||||||
* the URL of the invitation is returned.
|
|
||||||
* If the message is no videochat invitation, NULL is returned.
|
|
||||||
* Must be released using dc_str_unref() when done.
|
|
||||||
*/
|
|
||||||
char* dc_msg_get_videochat_url (const dc_msg_t* msg);
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the error status of the message.
|
* Gets the error status of the message.
|
||||||
* If there is no error associated with the message, NULL is returned.
|
* If there is no error associated with the message, NULL is returned.
|
||||||
@@ -4609,41 +4713,6 @@ char* dc_msg_get_videochat_url (const dc_msg_t* msg);
|
|||||||
char* dc_msg_get_error (const dc_msg_t* msg);
|
char* dc_msg_get_error (const dc_msg_t* msg);
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get type of videochat.
|
|
||||||
*
|
|
||||||
* Calling this functions only makes sense for messages of type #DC_MSG_VIDEOCHAT_INVITATION,
|
|
||||||
* in this case, if `basicwebrtc:` as of https://github.com/cracker0dks/basicwebrtc or `jitsi`
|
|
||||||
* were used to initiate the videochat,
|
|
||||||
* dc_msg_get_videochat_type() returns the corresponding type.
|
|
||||||
*
|
|
||||||
* The videochat URL can be retrieved using dc_msg_get_videochat_url().
|
|
||||||
* To check if a message is a videochat invitation at all, check the message type for #DC_MSG_VIDEOCHAT_INVITATION.
|
|
||||||
*
|
|
||||||
* @memberof dc_msg_t
|
|
||||||
* @param msg The message object.
|
|
||||||
* @return Type of the videochat as of DC_VIDEOCHATTYPE_BASICWEBRTC, DC_VIDEOCHATTYPE_JITSI or DC_VIDEOCHATTYPE_UNKNOWN.
|
|
||||||
*
|
|
||||||
* Example:
|
|
||||||
* ~~~
|
|
||||||
* if (dc_msg_get_viewtype(msg) == DC_MSG_VIDEOCHAT_INVITATION) {
|
|
||||||
* if (dc_msg_get_videochat_type(msg) == DC_VIDEOCHATTYPE_BASICWEBRTC) {
|
|
||||||
* // videochat invitation that we ship a client for
|
|
||||||
* } else {
|
|
||||||
* // use browser for videochat - or add an additional check for DC_VIDEOCHATTYPE_JITSI
|
|
||||||
* }
|
|
||||||
* } else {
|
|
||||||
* // not a videochat invitation
|
|
||||||
* }
|
|
||||||
* ~~~
|
|
||||||
*/
|
|
||||||
int dc_msg_get_videochat_type (const dc_msg_t* msg);
|
|
||||||
|
|
||||||
#define DC_VIDEOCHATTYPE_UNKNOWN 0
|
|
||||||
#define DC_VIDEOCHATTYPE_BASICWEBRTC 1
|
|
||||||
#define DC_VIDEOCHATTYPE_JITSI 2
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if the message has a full HTML version.
|
* Checks if the message has a full HTML version.
|
||||||
*
|
*
|
||||||
@@ -5187,20 +5256,14 @@ int dc_contact_is_blocked (const dc_contact_t* contact);
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if the contact
|
* Check if the contact
|
||||||
* can be added to verified chats,
|
* can be added to protected chats.
|
||||||
* i.e. has a verified key
|
|
||||||
* and Autocrypt key matches the verified key.
|
|
||||||
*
|
*
|
||||||
* If contact is verified
|
* See dc_contact_get_verifier_id() for a guidance how to display these information.
|
||||||
* UI should display green checkmark after the contact name
|
|
||||||
* in contact list items,
|
|
||||||
* in chat member list items
|
|
||||||
* and in profiles if no chat with the contact exist (otherwise, use dc_chat_is_protected()).
|
|
||||||
*
|
*
|
||||||
* @memberof dc_contact_t
|
* @memberof dc_contact_t
|
||||||
* @param contact The contact object.
|
* @param contact The contact object.
|
||||||
* @return 0: contact is not verified.
|
* @return 0: contact is not verified.
|
||||||
* 2: SELF and contact have verified their fingerprints in both directions; in the UI typically checkmarks are shown.
|
* 2: SELF and contact have verified their fingerprints in both directions.
|
||||||
*/
|
*/
|
||||||
int dc_contact_is_verified (dc_contact_t* contact);
|
int dc_contact_is_verified (dc_contact_t* contact);
|
||||||
|
|
||||||
@@ -5214,19 +5277,39 @@ int dc_contact_is_verified (dc_contact_t* contact);
|
|||||||
int dc_contact_is_bot (dc_contact_t* contact);
|
int dc_contact_is_bot (dc_contact_t* contact);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns whether contact is a key-contact,
|
||||||
|
* i.e. it is identified by the public key
|
||||||
|
* rather than the email address.
|
||||||
|
*
|
||||||
|
* If so, all messages to and from this contact are encrypted.
|
||||||
|
*
|
||||||
|
* @memberof dc_contact_t
|
||||||
|
* @param contact The contact object.
|
||||||
|
* @return 1 if the contact is a key-contact, 0 if it is an address-contact.
|
||||||
|
*/
|
||||||
|
int dc_contact_is_key_contact (dc_contact_t* contact);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the contact ID that verified a contact.
|
* Return the contact ID that verified a contact.
|
||||||
*
|
*
|
||||||
* If the function returns non-zero result,
|
* As verifier may be unknown,
|
||||||
* display green checkmark in the profile and "Introduced by ..." line
|
* use dc_contact_is_verified() to check if a contact can be added to a protected chat.
|
||||||
* with the name and address of the contact
|
|
||||||
* formatted by dc_contact_get_name_n_addr.
|
|
||||||
*
|
*
|
||||||
* If this function returns a verifier,
|
* UI should display the information in the contact's profile as follows:
|
||||||
* this does not necessarily mean
|
*
|
||||||
* you can add the contact to verified chats.
|
* - If dc_contact_get_verifier_id() != 0,
|
||||||
* Use dc_contact_is_verified() to check
|
* display text "Introduced by ..."
|
||||||
* if a contact can be added to a verified chat instead.
|
* with the name and address of the contact
|
||||||
|
* formatted by dc_contact_get_name_n_addr().
|
||||||
|
* Prefix the text by a green checkmark.
|
||||||
|
*
|
||||||
|
* - If dc_contact_get_verifier_id() == 0 and dc_contact_is_verified() != 0,
|
||||||
|
* display "Introduced" prefixed by a green checkmark.
|
||||||
|
*
|
||||||
|
* - if dc_contact_get_verifier_id() == 0 and dc_contact_is_verified() == 0,
|
||||||
|
* display nothing
|
||||||
*
|
*
|
||||||
* @memberof dc_contact_t
|
* @memberof dc_contact_t
|
||||||
* @param contact The contact object.
|
* @param contact The contact object.
|
||||||
@@ -5262,11 +5345,9 @@ dc_provider_t* dc_provider_new_from_email (const dc_context_t* conte
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a provider struct for the given e-mail address by local and DNS lookup.
|
* Create a provider struct for the given e-mail address by local lookup.
|
||||||
*
|
*
|
||||||
* First lookup is done from the local database as of dc_provider_new_from_email().
|
* DNS lookup is not used anymore and this function is deprecated.
|
||||||
* If the first lookup fails, an additional DNS lookup is done,
|
|
||||||
* trying to figure out the provider belonging to custom domains.
|
|
||||||
*
|
*
|
||||||
* @memberof dc_provider_t
|
* @memberof dc_provider_t
|
||||||
* @param context The context object.
|
* @param context The context object.
|
||||||
@@ -5274,6 +5355,7 @@ dc_provider_t* dc_provider_new_from_email (const dc_context_t* conte
|
|||||||
* @return A dc_provider_t struct which can be used with the dc_provider_get_*
|
* @return A dc_provider_t struct which can be used with the dc_provider_get_*
|
||||||
* accessor functions. If no provider info is found, NULL will be
|
* accessor functions. If no provider info is found, NULL will be
|
||||||
* returned.
|
* returned.
|
||||||
|
* @deprecated 2025-10-17 use dc_provider_new_from_email() instead.
|
||||||
*/
|
*/
|
||||||
dc_provider_t* dc_provider_new_from_email_with_dns (const dc_context_t* context, const char* email);
|
dc_provider_t* dc_provider_new_from_email_with_dns (const dc_context_t* context, const char* email);
|
||||||
|
|
||||||
@@ -5529,14 +5611,21 @@ int64_t dc_lot_get_timestamp (const dc_lot_t* lot);
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Message indicating an incoming or outgoing videochat.
|
* Message indicating an incoming or outgoing call.
|
||||||
* The message was created via dc_send_videochat_invitation() on this or a remote device.
|
|
||||||
*
|
*
|
||||||
* Typically, such messages are rendered differently by the UIs,
|
* These messages are created by dc_place_outgoing_call()
|
||||||
* e.g. contain a button to join the videochat.
|
* and should be rendered by UI similar to text messages,
|
||||||
* The URL for joining can be retrieved using dc_msg_get_videochat_url().
|
* maybe with some "phone icon" at the side.
|
||||||
|
*
|
||||||
|
* The message text is updated as needed
|
||||||
|
* and UI will be informed via #DC_EVENT_MSGS_CHANGED as usual.
|
||||||
|
*
|
||||||
|
* Do not start ringing when seeing this message;
|
||||||
|
* the mesage may belong e.g. to an old missed call.
|
||||||
|
*
|
||||||
|
* Instead, ringing should start on the event #DC_EVENT_INCOMING_CALL
|
||||||
*/
|
*/
|
||||||
#define DC_MSG_VIDEOCHAT_INVITATION 70
|
#define DC_MSG_CALL 71
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -5659,9 +5748,33 @@ int64_t dc_lot_get_timestamp (const dc_lot_t* lot);
|
|||||||
#define DC_CHAT_TYPE_MAILINGLIST 140
|
#define DC_CHAT_TYPE_MAILINGLIST 140
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A broadcast list. See dc_chat_get_type() for details.
|
* Outgoing broadcast channel, called "Channel" in the UI.
|
||||||
|
*
|
||||||
|
* The user can send into this chat,
|
||||||
|
* and all recipients will receive messages
|
||||||
|
* in a `DC_CHAT_TYPE_IN_BROADCAST`.
|
||||||
|
*
|
||||||
|
* Called `broadcast` here rather than `channel`,
|
||||||
|
* because the word "channel" already appears a lot in the code,
|
||||||
|
* which would make it hard to grep for it.
|
||||||
*/
|
*/
|
||||||
#define DC_CHAT_TYPE_BROADCAST 160
|
#define DC_CHAT_TYPE_OUT_BROADCAST 160
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Incoming broadcast channel, called "Channel" in the UI.
|
||||||
|
*
|
||||||
|
* This chat is read-only,
|
||||||
|
* and we do not know who the other recipients are.
|
||||||
|
*
|
||||||
|
* This is similar to `DC_CHAT_TYPE_MAILINGLIST`,
|
||||||
|
* with the main difference being that
|
||||||
|
* broadcasts are encrypted.
|
||||||
|
*
|
||||||
|
* Called `broadcast` here rather than `channel`,
|
||||||
|
* because the word "channel" already appears a lot in the code,
|
||||||
|
* which would make it hard to grep for it.
|
||||||
|
*/
|
||||||
|
#define DC_CHAT_TYPE_IN_BROADCAST 165
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
@@ -6268,7 +6381,6 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Chat changed. The name or the image of a chat group was changed or members were added or removed.
|
* Chat changed. The name or the image of a chat group was changed or members were added or removed.
|
||||||
* Or the verify state of a chat has changed.
|
|
||||||
* See dc_set_chat_name(), dc_set_chat_profile_image(), dc_add_contact_to_chat()
|
* See dc_set_chat_name(), dc_set_chat_profile_image(), dc_add_contact_to_chat()
|
||||||
* and dc_remove_contact_from_chat().
|
* and dc_remove_contact_from_chat().
|
||||||
*
|
*
|
||||||
@@ -6358,11 +6470,7 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
* generated by dc_get_securejoin_qr().
|
* generated by dc_get_securejoin_qr().
|
||||||
*
|
*
|
||||||
* @param data1 (int) The ID of the contact that wants to join.
|
* @param data1 (int) The ID of the contact that wants to join.
|
||||||
* @param data2 (int) The progress as:
|
* @param data2 (int) The progress, always 1000.
|
||||||
* 300=vg-/vc-request received, typically shown as "bob@addr joins".
|
|
||||||
* 600=vg-/vc-request-with-auth received, vg-member-added/vc-contact-confirm sent, typically shown as "bob@addr verified".
|
|
||||||
* 800=contact added to chat, shown as "bob@addr securely joined GROUP". Only for the verified-group-protocol.
|
|
||||||
* 1000=Protocol finished for this contact.
|
|
||||||
*/
|
*/
|
||||||
#define DC_EVENT_SECUREJOIN_INVITER_PROGRESS 2060
|
#define DC_EVENT_SECUREJOIN_INVITER_PROGRESS 2060
|
||||||
|
|
||||||
@@ -6514,6 +6622,60 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
*/
|
*/
|
||||||
#define DC_EVENT_CHANNEL_OVERFLOW 2400
|
#define DC_EVENT_CHANNEL_OVERFLOW 2400
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Incoming call.
|
||||||
|
* UI will usually start ringing,
|
||||||
|
* or show a notification if there is already a call in some profile.
|
||||||
|
*
|
||||||
|
* Together with this event,
|
||||||
|
* a message of type #DC_MSG_CALL is added to the corresponding chat;
|
||||||
|
* this message is announced and updated by the usual event as #DC_EVENT_MSGS_CHANGED,
|
||||||
|
* there is usually no need to take care of this message from any of the CALL events.
|
||||||
|
*
|
||||||
|
* If user takes action, dc_accept_incoming_call() or dc_end_call() should be called.
|
||||||
|
*
|
||||||
|
* Otherwise, ringing should end on #DC_EVENT_CALL_ENDED
|
||||||
|
* or #DC_EVENT_INCOMING_CALL_ACCEPTED
|
||||||
|
*
|
||||||
|
* @param data1 (int) msg_id ID of the message referring to the call.
|
||||||
|
* @param data2 (char*) place_call_info, text passed to dc_place_outgoing_call()
|
||||||
|
* @param data2 (int) 1 if incoming call is a video call, 0 otherwise
|
||||||
|
*/
|
||||||
|
#define DC_EVENT_INCOMING_CALL 2550
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The callee accepted an incoming call on this or another device using dc_accept_incoming_call().
|
||||||
|
* The caller gets the event #DC_EVENT_OUTGOING_CALL_ACCEPTED at the same time.
|
||||||
|
*
|
||||||
|
* UI usually only takes action in case call UI was opened before, otherwise the event should be ignored.
|
||||||
|
*
|
||||||
|
* @param data1 (int) msg_id ID of the message referring to the call
|
||||||
|
*/
|
||||||
|
#define DC_EVENT_INCOMING_CALL_ACCEPTED 2560
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A call placed using dc_place_outgoing_call() was accepted by the callee using dc_accept_incoming_call().
|
||||||
|
*
|
||||||
|
* UI usually only takes action in case call UI was opened before, otherwise the event should be ignored.
|
||||||
|
*
|
||||||
|
* @param data1 (int) msg_id ID of the message referring to the call
|
||||||
|
* @param data2 (char*) accept_call_info, text passed to dc_accept_incoming_call()
|
||||||
|
*/
|
||||||
|
#define DC_EVENT_OUTGOING_CALL_ACCEPTED 2570
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An incoming or outgoing call was ended using dc_end_call() on this or another device, by caller or callee.
|
||||||
|
* Moreover, the event is sent when the call was not accepted within 1 minute timeout.
|
||||||
|
*
|
||||||
|
* UI usually only takes action in case call UI was opened before, otherwise the event should be ignored.
|
||||||
|
*
|
||||||
|
* @param data1 (int) msg_id ID of the message referring to the call
|
||||||
|
*/
|
||||||
|
#define DC_EVENT_CALL_ENDED 2580
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
*/
|
*/
|
||||||
@@ -6781,9 +6943,7 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// Used in summaries.
|
/// Used in summaries.
|
||||||
#define DC_STR_GIF 23
|
#define DC_STR_GIF 23
|
||||||
|
|
||||||
/// "Encrypted message"
|
/// @deprecated 2025-07, this string is no longer needed.
|
||||||
///
|
|
||||||
/// Used in subjects of outgoing messages.
|
|
||||||
#define DC_STR_ENCRYPTEDMSG 24
|
#define DC_STR_ENCRYPTEDMSG 24
|
||||||
|
|
||||||
/// "End-to-end encryption available."
|
/// "End-to-end encryption available."
|
||||||
@@ -6799,11 +6959,6 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// Used to build the string returned by dc_get_contact_encrinfo().
|
/// Used to build the string returned by dc_get_contact_encrinfo().
|
||||||
#define DC_STR_ENCR_NONE 28
|
#define DC_STR_ENCR_NONE 28
|
||||||
|
|
||||||
/// "This message was encrypted for another setup."
|
|
||||||
///
|
|
||||||
/// Used as message text if decryption fails.
|
|
||||||
#define DC_STR_CANTDECRYPT_MSG_BODY 29
|
|
||||||
|
|
||||||
/// "Fingerprints"
|
/// "Fingerprints"
|
||||||
///
|
///
|
||||||
/// Used to build the string returned by dc_get_contact_encrinfo().
|
/// Used to build the string returned by dc_get_contact_encrinfo().
|
||||||
@@ -6830,6 +6985,7 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// "End-to-end encryption preferred."
|
/// "End-to-end encryption preferred."
|
||||||
///
|
///
|
||||||
/// Used to build the string returned by dc_get_contact_encrinfo().
|
/// Used to build the string returned by dc_get_contact_encrinfo().
|
||||||
|
/// @deprecated 2025-06-05
|
||||||
#define DC_STR_E2E_PREFERRED 34
|
#define DC_STR_E2E_PREFERRED 34
|
||||||
|
|
||||||
/// "%1$s verified"
|
/// "%1$s verified"
|
||||||
@@ -6842,12 +6998,14 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
///
|
///
|
||||||
/// Used in status messages.
|
/// Used in status messages.
|
||||||
/// - %1$s will be replaced by the name of the contact that cannot be verified
|
/// - %1$s will be replaced by the name of the contact that cannot be verified
|
||||||
|
/// @deprecated 2025-06-05
|
||||||
#define DC_STR_CONTACT_NOT_VERIFIED 36
|
#define DC_STR_CONTACT_NOT_VERIFIED 36
|
||||||
|
|
||||||
/// "Changed setup for %1$s."
|
/// "Changed setup for %1$s."
|
||||||
///
|
///
|
||||||
/// Used in status messages.
|
/// Used in status messages.
|
||||||
/// - %1$s will be replaced by the name of the contact with the changed setup
|
/// - %1$s will be replaced by the name of the contact with the changed setup
|
||||||
|
/// @deprecated 2025-06-05
|
||||||
#define DC_STR_CONTACT_SETUP_CHANGED 37
|
#define DC_STR_CONTACT_SETUP_CHANGED 37
|
||||||
|
|
||||||
/// "Archived chats"
|
/// "Archived chats"
|
||||||
@@ -6857,12 +7015,12 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
|
|
||||||
/// "Autocrypt Setup Message"
|
/// "Autocrypt Setup Message"
|
||||||
///
|
///
|
||||||
/// Used in subjects of outgoing Autocrypt Setup Messages.
|
/// @deprecated 2025-04
|
||||||
#define DC_STR_AC_SETUP_MSG_SUBJECT 42
|
#define DC_STR_AC_SETUP_MSG_SUBJECT 42
|
||||||
|
|
||||||
/// "This is the Autocrypt Setup Message, open it in a compatible client to use your setup"
|
/// "This is the Autocrypt Setup Message, open it in a compatible client to use your setup"
|
||||||
///
|
///
|
||||||
/// Used as message text of outgoing Autocrypt Setup Messages.
|
/// @deprecated 2025-04
|
||||||
#define DC_STR_AC_SETUP_MSG_BODY 43
|
#define DC_STR_AC_SETUP_MSG_BODY 43
|
||||||
|
|
||||||
/// "Cannot login as %1$s."
|
/// "Cannot login as %1$s."
|
||||||
@@ -6933,6 +7091,8 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// "Unknown sender for this chat. See 'info' for more details."
|
/// "Unknown sender for this chat. See 'info' for more details."
|
||||||
///
|
///
|
||||||
/// Use as message text if assigning the message to a chat is not totally correct.
|
/// Use as message text if assigning the message to a chat is not totally correct.
|
||||||
|
///
|
||||||
|
/// @deprecated 2025-08-18
|
||||||
#define DC_STR_UNKNOWN_SENDER_FOR_CHAT 72
|
#define DC_STR_UNKNOWN_SENDER_FOR_CHAT 72
|
||||||
|
|
||||||
/// "Message from %1$s"
|
/// "Message from %1$s"
|
||||||
@@ -6995,17 +7155,6 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// @deprecated Deprecated 2021-01-30, DC_STR_EPHEMERAL_WEEKS is used instead.
|
/// @deprecated Deprecated 2021-01-30, DC_STR_EPHEMERAL_WEEKS is used instead.
|
||||||
#define DC_STR_EPHEMERAL_FOUR_WEEKS 81
|
#define DC_STR_EPHEMERAL_FOUR_WEEKS 81
|
||||||
|
|
||||||
/// "Video chat invitation"
|
|
||||||
///
|
|
||||||
/// Used in summaries.
|
|
||||||
#define DC_STR_VIDEOCHAT_INVITATION 82
|
|
||||||
|
|
||||||
/// "You are invited to a video chat, click %1$s to join."
|
|
||||||
///
|
|
||||||
/// Used as message text of outgoing video chat invitations.
|
|
||||||
/// - %1$s will be replaced by the URL of the video chat
|
|
||||||
#define DC_STR_VIDEOCHAT_INVITE_MSG_BODY 83
|
|
||||||
|
|
||||||
/// "Error: %1$s"
|
/// "Error: %1$s"
|
||||||
///
|
///
|
||||||
/// Used in error strings.
|
/// Used in error strings.
|
||||||
@@ -7237,6 +7386,7 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// "%1$s changed their address from %2$s to %3$s"
|
/// "%1$s changed their address from %2$s to %3$s"
|
||||||
///
|
///
|
||||||
/// Used as an info message to chats with contacts that changed their address.
|
/// Used as an info message to chats with contacts that changed their address.
|
||||||
|
/// @deprecated 2025-06-05
|
||||||
#define DC_STR_AEAP_ADDR_CHANGED 122
|
#define DC_STR_AEAP_ADDR_CHANGED 122
|
||||||
|
|
||||||
/// "You changed your email address from %1$s to %2$s.
|
/// "You changed your email address from %1$s to %2$s.
|
||||||
@@ -7474,6 +7624,18 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// `%2$s` will be replaced by name and address of the contact.
|
/// `%2$s` will be replaced by name and address of the contact.
|
||||||
#define DC_STR_EPHEMERAL_TIMER_WEEKS_BY_OTHER 157
|
#define DC_STR_EPHEMERAL_TIMER_WEEKS_BY_OTHER 157
|
||||||
|
|
||||||
|
/// "You set message deletion timer to 1 year."
|
||||||
|
///
|
||||||
|
/// Used in status messages.
|
||||||
|
#define DC_STR_EPHEMERAL_TIMER_1_YEAR_BY_YOU 158
|
||||||
|
|
||||||
|
/// "Message deletion timer is set to 1 year by %1$s."
|
||||||
|
///
|
||||||
|
/// `%1$s` will be replaced by name and address of the contact.
|
||||||
|
///
|
||||||
|
/// Used in status messages.
|
||||||
|
#define DC_STR_EPHEMERAL_TIMER_1_YEAR_BY_OTHER 159
|
||||||
|
|
||||||
/// "Scan to set up second device for %1$s"
|
/// "Scan to set up second device for %1$s"
|
||||||
///
|
///
|
||||||
/// `%1$s` will be replaced by name and address of the account.
|
/// `%1$s` will be replaced by name and address of the account.
|
||||||
@@ -7484,7 +7646,7 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// Used as a device message after a successful backup transfer.
|
/// Used as a device message after a successful backup transfer.
|
||||||
#define DC_STR_BACKUP_TRANSFER_MSG_BODY 163
|
#define DC_STR_BACKUP_TRANSFER_MSG_BODY 163
|
||||||
|
|
||||||
/// "Messages are guaranteed to be end-to-end encrypted from now on."
|
/// "Messages are end-to-end encrypted."
|
||||||
///
|
///
|
||||||
/// Used in info messages.
|
/// Used in info messages.
|
||||||
#define DC_STR_CHAT_PROTECTION_ENABLED 170
|
#define DC_STR_CHAT_PROTECTION_ENABLED 170
|
||||||
@@ -7492,6 +7654,7 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
/// "%1$s sent a message from another device."
|
/// "%1$s sent a message from another device."
|
||||||
///
|
///
|
||||||
/// Used in info messages.
|
/// Used in info messages.
|
||||||
|
/// @deprecated 2025-07
|
||||||
#define DC_STR_CHAT_PROTECTION_DISABLED 171
|
#define DC_STR_CHAT_PROTECTION_DISABLED 171
|
||||||
|
|
||||||
/// "Others will only see this group after you sent a first message."
|
/// "Others will only see this group after you sent a first message."
|
||||||
@@ -7537,11 +7700,54 @@ void dc_event_unref(dc_event_t* event);
|
|||||||
|
|
||||||
/// "Could not yet establish guaranteed end-to-end encryption, but you may already send a message."
|
/// "Could not yet establish guaranteed end-to-end encryption, but you may already send a message."
|
||||||
///
|
///
|
||||||
/// Used as info message.
|
/// @deprecated 2025-03
|
||||||
#define DC_STR_SECUREJOIN_WAIT_TIMEOUT 191
|
#define DC_STR_SECUREJOIN_WAIT_TIMEOUT 191
|
||||||
|
|
||||||
/// "Contact". Deprecated, currently unused.
|
/// "The contact must be online to proceed. This process will continue automatically in background."
|
||||||
#define DC_STR_CONTACT 200
|
///
|
||||||
|
/// Used as info message.
|
||||||
|
/// @deprecated 2025-06-05
|
||||||
|
#define DC_STR_SECUREJOIN_TAKES_LONGER 192
|
||||||
|
|
||||||
|
/// "❤️ Seems you're enjoying Delta Chat!"… (donation request device message)
|
||||||
|
#define DC_STR_DONATION_REQUEST 193
|
||||||
|
|
||||||
|
/// "Outgoing call"
|
||||||
|
#define DC_STR_OUTGOING_CALL 194
|
||||||
|
|
||||||
|
/// "Incoming call"
|
||||||
|
#define DC_STR_INCOMING_CALL 195
|
||||||
|
|
||||||
|
/// "Declined call"
|
||||||
|
#define DC_STR_DECLINED_CALL 196
|
||||||
|
|
||||||
|
/// "Canceled call"
|
||||||
|
#define DC_STR_CANCELED_CALL 197
|
||||||
|
|
||||||
|
/// "Missed call"
|
||||||
|
#define DC_STR_MISSED_CALL 198
|
||||||
|
|
||||||
|
/// "You left the channel."
|
||||||
|
///
|
||||||
|
/// Used in status messages.
|
||||||
|
#define DC_STR_CHANNEL_LEFT_BY_YOU 200
|
||||||
|
|
||||||
|
/// "Scan to join channel %1$s"
|
||||||
|
///
|
||||||
|
/// Subtitle for channel join qrcode svg image generated by the core.
|
||||||
|
///
|
||||||
|
/// `%1$s` will be replaced with the channel name.
|
||||||
|
#define DC_STR_SECURE_JOIN_CHANNEL_QR_DESC 201
|
||||||
|
|
||||||
|
/// "Proxy Enabled"
|
||||||
|
///
|
||||||
|
/// Title for proxy section in connectivity view.
|
||||||
|
#define DC_STR_PROXY_ENABLED 220
|
||||||
|
|
||||||
|
/// "You are using a proxy. If you're having trouble connecting, try a different proxy."
|
||||||
|
///
|
||||||
|
/// Description in connectivity view when proxy is enabled.
|
||||||
|
#define DC_STR_PROXY_ENABLED_DESCRIPTION 221
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -34,7 +34,7 @@ pub enum Meaning {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Lot {
|
impl Lot {
|
||||||
pub fn get_text1(&self) -> Option<Cow<str>> {
|
pub fn get_text1(&self) -> Option<Cow<'_, str>> {
|
||||||
match self {
|
match self {
|
||||||
Self::Summary(summary) => match &summary.prefix {
|
Self::Summary(summary) => match &summary.prefix {
|
||||||
None => None,
|
None => None,
|
||||||
@@ -45,13 +45,13 @@ impl Lot {
|
|||||||
Self::Qr(qr) => match qr {
|
Self::Qr(qr) => match qr {
|
||||||
Qr::AskVerifyContact { .. } => None,
|
Qr::AskVerifyContact { .. } => None,
|
||||||
Qr::AskVerifyGroup { grpname, .. } => Some(Cow::Borrowed(grpname)),
|
Qr::AskVerifyGroup { grpname, .. } => Some(Cow::Borrowed(grpname)),
|
||||||
|
Qr::AskJoinBroadcast { name, .. } => Some(Cow::Borrowed(name)),
|
||||||
Qr::FprOk { .. } => None,
|
Qr::FprOk { .. } => None,
|
||||||
Qr::FprMismatch { .. } => None,
|
Qr::FprMismatch { .. } => None,
|
||||||
Qr::FprWithoutAddr { fingerprint, .. } => Some(Cow::Borrowed(fingerprint)),
|
Qr::FprWithoutAddr { fingerprint, .. } => Some(Cow::Borrowed(fingerprint)),
|
||||||
Qr::Account { domain } => Some(Cow::Borrowed(domain)),
|
Qr::Account { domain } => Some(Cow::Borrowed(domain)),
|
||||||
Qr::Backup2 { .. } => None,
|
Qr::Backup2 { .. } => None,
|
||||||
Qr::BackupTooNew { .. } => None,
|
Qr::BackupTooNew { .. } => None,
|
||||||
Qr::WebrtcInstance { domain, .. } => Some(Cow::Borrowed(domain)),
|
|
||||||
Qr::Proxy { host, port, .. } => Some(Cow::Owned(format!("{host}:{port}"))),
|
Qr::Proxy { host, port, .. } => Some(Cow::Owned(format!("{host}:{port}"))),
|
||||||
Qr::Addr { draft, .. } => draft.as_deref().map(Cow::Borrowed),
|
Qr::Addr { draft, .. } => draft.as_deref().map(Cow::Borrowed),
|
||||||
Qr::Url { url } => Some(Cow::Borrowed(url)),
|
Qr::Url { url } => Some(Cow::Borrowed(url)),
|
||||||
@@ -66,7 +66,7 @@ impl Lot {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_text2(&self) -> Option<Cow<str>> {
|
pub fn get_text2(&self) -> Option<Cow<'_, str>> {
|
||||||
match self {
|
match self {
|
||||||
Self::Summary(summary) => Some(summary.truncated_text(160)),
|
Self::Summary(summary) => Some(summary.truncated_text(160)),
|
||||||
Self::Qr(_) => None,
|
Self::Qr(_) => None,
|
||||||
@@ -99,13 +99,13 @@ impl Lot {
|
|||||||
Self::Qr(qr) => match qr {
|
Self::Qr(qr) => match qr {
|
||||||
Qr::AskVerifyContact { .. } => LotState::QrAskVerifyContact,
|
Qr::AskVerifyContact { .. } => LotState::QrAskVerifyContact,
|
||||||
Qr::AskVerifyGroup { .. } => LotState::QrAskVerifyGroup,
|
Qr::AskVerifyGroup { .. } => LotState::QrAskVerifyGroup,
|
||||||
|
Qr::AskJoinBroadcast { .. } => LotState::QrAskJoinBroadcast,
|
||||||
Qr::FprOk { .. } => LotState::QrFprOk,
|
Qr::FprOk { .. } => LotState::QrFprOk,
|
||||||
Qr::FprMismatch { .. } => LotState::QrFprMismatch,
|
Qr::FprMismatch { .. } => LotState::QrFprMismatch,
|
||||||
Qr::FprWithoutAddr { .. } => LotState::QrFprWithoutAddr,
|
Qr::FprWithoutAddr { .. } => LotState::QrFprWithoutAddr,
|
||||||
Qr::Account { .. } => LotState::QrAccount,
|
Qr::Account { .. } => LotState::QrAccount,
|
||||||
Qr::Backup2 { .. } => LotState::QrBackup2,
|
Qr::Backup2 { .. } => LotState::QrBackup2,
|
||||||
Qr::BackupTooNew { .. } => LotState::QrBackupTooNew,
|
Qr::BackupTooNew { .. } => LotState::QrBackupTooNew,
|
||||||
Qr::WebrtcInstance { .. } => LotState::QrWebrtcInstance,
|
|
||||||
Qr::Proxy { .. } => LotState::QrProxy,
|
Qr::Proxy { .. } => LotState::QrProxy,
|
||||||
Qr::Addr { .. } => LotState::QrAddr,
|
Qr::Addr { .. } => LotState::QrAddr,
|
||||||
Qr::Url { .. } => LotState::QrUrl,
|
Qr::Url { .. } => LotState::QrUrl,
|
||||||
@@ -126,13 +126,13 @@ impl Lot {
|
|||||||
Self::Qr(qr) => match qr {
|
Self::Qr(qr) => match qr {
|
||||||
Qr::AskVerifyContact { contact_id, .. } => contact_id.to_u32(),
|
Qr::AskVerifyContact { contact_id, .. } => contact_id.to_u32(),
|
||||||
Qr::AskVerifyGroup { .. } => Default::default(),
|
Qr::AskVerifyGroup { .. } => Default::default(),
|
||||||
|
Qr::AskJoinBroadcast { .. } => Default::default(),
|
||||||
Qr::FprOk { contact_id } => contact_id.to_u32(),
|
Qr::FprOk { contact_id } => contact_id.to_u32(),
|
||||||
Qr::FprMismatch { contact_id } => contact_id.unwrap_or_default().to_u32(),
|
Qr::FprMismatch { contact_id } => contact_id.unwrap_or_default().to_u32(),
|
||||||
Qr::FprWithoutAddr { .. } => Default::default(),
|
Qr::FprWithoutAddr { .. } => Default::default(),
|
||||||
Qr::Account { .. } => Default::default(),
|
Qr::Account { .. } => Default::default(),
|
||||||
Qr::Backup2 { .. } => Default::default(),
|
Qr::Backup2 { .. } => Default::default(),
|
||||||
Qr::BackupTooNew { .. } => Default::default(),
|
Qr::BackupTooNew { .. } => Default::default(),
|
||||||
Qr::WebrtcInstance { .. } => Default::default(),
|
|
||||||
Qr::Proxy { .. } => Default::default(),
|
Qr::Proxy { .. } => Default::default(),
|
||||||
Qr::Addr { contact_id, .. } => contact_id.to_u32(),
|
Qr::Addr { contact_id, .. } => contact_id.to_u32(),
|
||||||
Qr::Url { .. } => Default::default(),
|
Qr::Url { .. } => Default::default(),
|
||||||
@@ -169,6 +169,9 @@ pub enum LotState {
|
|||||||
/// text1=groupname
|
/// text1=groupname
|
||||||
QrAskVerifyGroup = 202,
|
QrAskVerifyGroup = 202,
|
||||||
|
|
||||||
|
/// text1=broadcast_name
|
||||||
|
QrAskJoinBroadcast = 204,
|
||||||
|
|
||||||
/// id=contact
|
/// id=contact
|
||||||
QrFprOk = 210,
|
QrFprOk = 210,
|
||||||
|
|
||||||
@@ -185,9 +188,6 @@ pub enum LotState {
|
|||||||
|
|
||||||
QrBackupTooNew = 255,
|
QrBackupTooNew = 255,
|
||||||
|
|
||||||
/// text1=domain, text2=instance pattern
|
|
||||||
QrWebrtcInstance = 260,
|
|
||||||
|
|
||||||
/// text1=address, text2=protocol
|
/// text1=address, text2=protocol
|
||||||
QrProxy = 271,
|
QrProxy = 271,
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "deltachat-jsonrpc"
|
name = "deltachat-jsonrpc"
|
||||||
version = "1.157.3"
|
version = "2.25.0"
|
||||||
description = "DeltaChat JSON-RPC API"
|
description = "DeltaChat JSON-RPC API"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
license = "MPL-2.0"
|
license = "MPL-2.0"
|
||||||
@@ -13,10 +13,7 @@ deltachat-contact-tools = { workspace = true }
|
|||||||
num-traits = { workspace = true }
|
num-traits = { workspace = true }
|
||||||
schemars = "0.8.22"
|
schemars = "0.8.22"
|
||||||
serde = { workspace = true, features = ["derive"] }
|
serde = { workspace = true, features = ["derive"] }
|
||||||
tempfile = { workspace = true }
|
|
||||||
log = { workspace = true }
|
|
||||||
async-channel = { workspace = true }
|
async-channel = { workspace = true }
|
||||||
futures = { workspace = true }
|
|
||||||
serde_json = { workspace = true }
|
serde_json = { workspace = true }
|
||||||
yerpc = { workspace = true, features = ["anyhow_expose", "openrpc"] }
|
yerpc = { workspace = true, features = ["anyhow_expose", "openrpc"] }
|
||||||
typescript-type-def = { version = "0.5.13", features = ["json_value"] }
|
typescript-type-def = { version = "0.5.13", features = ["json_value"] }
|
||||||
@@ -27,6 +24,8 @@ base64 = { workspace = true }
|
|||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
tokio = { workspace = true, features = ["full", "rt-multi-thread"] }
|
tokio = { workspace = true, features = ["full", "rt-multi-thread"] }
|
||||||
|
tempfile = { workspace = true }
|
||||||
|
futures = { workspace = true }
|
||||||
|
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
|
|||||||
@@ -8,10 +8,10 @@ use std::{collections::HashMap, str::FromStr};
|
|||||||
use anyhow::{anyhow, bail, ensure, Context, Result};
|
use anyhow::{anyhow, bail, ensure, Context, Result};
|
||||||
pub use deltachat::accounts::Accounts;
|
pub use deltachat::accounts::Accounts;
|
||||||
use deltachat::blob::BlobObject;
|
use deltachat::blob::BlobObject;
|
||||||
|
use deltachat::calls::ice_servers;
|
||||||
use deltachat::chat::{
|
use deltachat::chat::{
|
||||||
self, add_contact_to_chat, forward_msgs, get_chat_media, get_chat_msgs, get_chat_msgs_ex,
|
self, add_contact_to_chat, forward_msgs, get_chat_media, get_chat_msgs, get_chat_msgs_ex,
|
||||||
marknoticed_chat, remove_contact_from_chat, Chat, ChatId, ChatItem, MessageListOptions,
|
marknoticed_chat, remove_contact_from_chat, Chat, ChatId, ChatItem, MessageListOptions,
|
||||||
ProtectionStatus,
|
|
||||||
};
|
};
|
||||||
use deltachat::chatlist::Chatlist;
|
use deltachat::chatlist::Chatlist;
|
||||||
use deltachat::config::Config;
|
use deltachat::config::Config;
|
||||||
@@ -19,6 +19,7 @@ use deltachat::constants::DC_MSG_ID_DAYMARKER;
|
|||||||
use deltachat::contact::{may_be_valid_addr, Contact, ContactId, Origin};
|
use deltachat::contact::{may_be_valid_addr, Contact, ContactId, Origin};
|
||||||
use deltachat::context::get_info;
|
use deltachat::context::get_info;
|
||||||
use deltachat::ephemeral::Timer;
|
use deltachat::ephemeral::Timer;
|
||||||
|
use deltachat::imex;
|
||||||
use deltachat::location;
|
use deltachat::location;
|
||||||
use deltachat::message::get_msg_read_receipts;
|
use deltachat::message::get_msg_read_receipts;
|
||||||
use deltachat::message::{
|
use deltachat::message::{
|
||||||
@@ -35,7 +36,6 @@ use deltachat::securejoin;
|
|||||||
use deltachat::stock_str::StockMessage;
|
use deltachat::stock_str::StockMessage;
|
||||||
use deltachat::webxdc::StatusUpdateSerial;
|
use deltachat::webxdc::StatusUpdateSerial;
|
||||||
use deltachat::EventEmitter;
|
use deltachat::EventEmitter;
|
||||||
use deltachat::{imex, info};
|
|
||||||
use sanitize_filename::is_sanitized;
|
use sanitize_filename::is_sanitized;
|
||||||
use tokio::fs;
|
use tokio::fs;
|
||||||
use tokio::sync::{watch, Mutex, RwLock};
|
use tokio::sync::{watch, Mutex, RwLock};
|
||||||
@@ -47,25 +47,27 @@ pub mod types;
|
|||||||
|
|
||||||
use num_traits::FromPrimitive;
|
use num_traits::FromPrimitive;
|
||||||
use types::account::Account;
|
use types::account::Account;
|
||||||
|
use types::calls::JsonrpcCallInfo;
|
||||||
use types::chat::FullChat;
|
use types::chat::FullChat;
|
||||||
use types::contact::{ContactObject, VcardContact};
|
use types::contact::{ContactObject, VcardContact};
|
||||||
use types::events::Event;
|
use types::events::Event;
|
||||||
use types::http::HttpResponse;
|
use types::http::HttpResponse;
|
||||||
use types::message::{MessageData, MessageObject, MessageReadReceipt};
|
use types::message::{MessageData, MessageObject, MessageReadReceipt};
|
||||||
|
use types::notify_state::JsonrpcNotifyState;
|
||||||
use types::provider_info::ProviderInfo;
|
use types::provider_info::ProviderInfo;
|
||||||
use types::reactions::JSONRPCReactions;
|
use types::reactions::JsonrpcReactions;
|
||||||
use types::webxdc::WebxdcMessageInfo;
|
use types::webxdc::WebxdcMessageInfo;
|
||||||
|
|
||||||
use self::types::message::{MessageInfo, MessageLoadResult};
|
use self::types::message::{MessageInfo, MessageLoadResult};
|
||||||
use self::types::{
|
use self::types::{
|
||||||
chat::{BasicChat, JSONRPCChatVisibility, MuteDuration},
|
chat::{BasicChat, JsonrpcChatVisibility, MuteDuration},
|
||||||
location::JsonrpcLocation,
|
location::JsonrpcLocation,
|
||||||
message::{
|
message::{
|
||||||
JSONRPCMessageListItem, MessageNotificationInfo, MessageSearchResult, MessageViewtype,
|
JsonrpcMessageListItem, MessageNotificationInfo, MessageSearchResult, MessageViewtype,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
use crate::api::types::chat_list::{get_chat_list_item_by_id, ChatListItemFetchResult};
|
use crate::api::types::chat_list::{get_chat_list_item_by_id, ChatListItemFetchResult};
|
||||||
use crate::api::types::qr::QrObject;
|
use crate::api::types::qr::{QrObject, SecurejoinSource, SecurejoinUiPath};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct AccountState {
|
struct AccountState {
|
||||||
@@ -91,7 +93,8 @@ pub struct CommandApi {
|
|||||||
|
|
||||||
/// Receiver side of the event channel.
|
/// Receiver side of the event channel.
|
||||||
///
|
///
|
||||||
/// Events from it can be received by calling `get_next_event` method.
|
/// Events from it can be received by calling
|
||||||
|
/// [`CommandApi::get_next_event`] method.
|
||||||
event_emitter: Arc<EventEmitter>,
|
event_emitter: Arc<EventEmitter>,
|
||||||
|
|
||||||
states: Arc<Mutex<BTreeMap<u32, AccountState>>>,
|
states: Arc<Mutex<BTreeMap<u32, AccountState>>>,
|
||||||
@@ -123,7 +126,7 @@ impl CommandApi {
|
|||||||
.read()
|
.read()
|
||||||
.await
|
.await
|
||||||
.get_account(id)
|
.get_account(id)
|
||||||
.ok_or_else(|| anyhow!("account with id {} not found", id))?;
|
.ok_or_else(|| anyhow!("account with id {id} not found"))?;
|
||||||
Ok(sc)
|
Ok(sc)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -173,7 +176,15 @@ impl CommandApi {
|
|||||||
get_info()
|
get_info()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the next event.
|
/// Get the next event, and remove it from the event queue.
|
||||||
|
///
|
||||||
|
/// If no events have happened since the last `get_next_event`
|
||||||
|
/// (i.e. if the event queue is empty), the response will be returned
|
||||||
|
/// only when a new event fires.
|
||||||
|
///
|
||||||
|
/// Note that if you are using the `BaseDeltaChat` JavaScript class
|
||||||
|
/// or the `Rpc` Python class, this function will be invoked
|
||||||
|
/// by those classes internally and should not be used manually.
|
||||||
async fn get_next_event(&self) -> Result<Event> {
|
async fn get_next_event(&self) -> Result<Event> {
|
||||||
self.event_emitter
|
self.event_emitter
|
||||||
.recv()
|
.recv()
|
||||||
@@ -224,11 +235,20 @@ impl CommandApi {
|
|||||||
self.accounts.read().await.get_selected_account_id()
|
self.accounts.read().await.get_selected_account_id()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Set the order of accounts.
|
||||||
|
/// The provided list should contain all account IDs in the desired order.
|
||||||
|
/// If an account ID is missing from the list, it will be appended at the end.
|
||||||
|
/// If the list contains non-existent account IDs, they will be ignored.
|
||||||
|
async fn set_accounts_order(&self, order: Vec<u32>) -> Result<()> {
|
||||||
|
self.accounts.write().await.set_accounts_order(order).await
|
||||||
|
}
|
||||||
|
|
||||||
/// Get a list of all configured accounts.
|
/// Get a list of all configured accounts.
|
||||||
async fn get_all_accounts(&self) -> Result<Vec<Account>> {
|
async fn get_all_accounts(&self) -> Result<Vec<Account>> {
|
||||||
let mut accounts = Vec::new();
|
let mut accounts = Vec::new();
|
||||||
for id in self.accounts.read().await.get_all() {
|
let accounts_lock = self.accounts.read().await;
|
||||||
let context_option = self.accounts.read().await.get_account(id);
|
for id in accounts_lock.get_all() {
|
||||||
|
let context_option = accounts_lock.get_account(id);
|
||||||
if let Some(ctx) = context_option {
|
if let Some(ctx) = context_option {
|
||||||
accounts.push(Account::from_context(&ctx, id).await?)
|
accounts.push(Account::from_context(&ctx, id).await?)
|
||||||
}
|
}
|
||||||
@@ -288,12 +308,17 @@ impl CommandApi {
|
|||||||
Ok(Account::from_context(&ctx, account_id).await?)
|
Ok(Account::from_context(&ctx, account_id).await?)
|
||||||
} else {
|
} else {
|
||||||
Err(anyhow!(
|
Err(anyhow!(
|
||||||
"account with id {} doesn't exist anymore",
|
"account with id {account_id} doesn't exist anymore"
|
||||||
account_id
|
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get the current push notification state.
|
||||||
|
async fn get_push_state(&self, account_id: u32) -> Result<JsonrpcNotifyState> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
Ok(ctx.push_state().await.into())
|
||||||
|
}
|
||||||
|
|
||||||
/// Get the combined filesize of an account in bytes
|
/// Get the combined filesize of an account in bytes
|
||||||
async fn get_account_file_size(&self, account_id: u32) -> Result<u64> {
|
async fn get_account_file_size(&self, account_id: u32) -> Result<u64> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
@@ -317,17 +342,10 @@ impl CommandApi {
|
|||||||
/// instead of the domain.
|
/// instead of the domain.
|
||||||
async fn get_provider_info(
|
async fn get_provider_info(
|
||||||
&self,
|
&self,
|
||||||
account_id: u32,
|
_account_id: u32,
|
||||||
email: String,
|
email: String,
|
||||||
) -> Result<Option<ProviderInfo>> {
|
) -> Result<Option<ProviderInfo>> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let provider_info = get_provider_info(email.split('@').next_back().unwrap_or(""));
|
||||||
|
|
||||||
let proxy_enabled = ctx
|
|
||||||
.get_config_bool(deltachat::config::Config::ProxyEnabled)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
let provider_info =
|
|
||||||
get_provider_info(&ctx, email.split('@').last().unwrap_or(""), proxy_enabled).await;
|
|
||||||
Ok(ProviderInfo::from_dc_type(provider_info))
|
Ok(ProviderInfo::from_dc_type(provider_info))
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -349,6 +367,20 @@ impl CommandApi {
|
|||||||
Ok(ctx.get_blobdir().to_str().map(|s| s.to_owned()))
|
Ok(ctx.get_blobdir().to_str().map(|s| s.to_owned()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// If there was an error while the account was opened
|
||||||
|
/// and migrated to the current version,
|
||||||
|
/// then this function returns it.
|
||||||
|
///
|
||||||
|
/// This function is useful because the key-contacts migration could fail due to bugs
|
||||||
|
/// and then the account will not work properly.
|
||||||
|
///
|
||||||
|
/// After opening an account, the UI should call this function
|
||||||
|
/// and show the error string if one is returned.
|
||||||
|
async fn get_migration_error(&self, account_id: u32) -> Result<Option<String>> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
Ok(ctx.get_migration_error())
|
||||||
|
}
|
||||||
|
|
||||||
/// Copy file to blob dir.
|
/// Copy file to blob dir.
|
||||||
async fn copy_to_blob_dir(&self, account_id: u32, path: String) -> Result<PathBuf> {
|
async fn copy_to_blob_dir(&self, account_id: u32, path: String) -> Result<PathBuf> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
@@ -356,11 +388,6 @@ impl CommandApi {
|
|||||||
Ok(BlobObject::create_and_deduplicate(&ctx, file, file)?.to_abs_path())
|
Ok(BlobObject::create_and_deduplicate(&ctx, file, file)?.to_abs_path())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn draft_self_report(&self, account_id: u32) -> Result<u32> {
|
|
||||||
let ctx = self.get_context(account_id).await?;
|
|
||||||
Ok(ctx.draft_self_report().await?.to_u32())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Sets the given configuration key.
|
/// Sets the given configuration key.
|
||||||
async fn set_config(&self, account_id: u32, key: String, value: Option<String>) -> Result<()> {
|
async fn set_config(&self, account_id: u32, key: String, value: Option<String>) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
@@ -434,7 +461,7 @@ impl CommandApi {
|
|||||||
/// Setup the credential config before calling this.
|
/// Setup the credential config before calling this.
|
||||||
///
|
///
|
||||||
/// Deprecated as of 2025-02; use `add_transport_from_qr()`
|
/// Deprecated as of 2025-02; use `add_transport_from_qr()`
|
||||||
/// or `add_transport()` instead.
|
/// or `add_or_update_transport()` instead.
|
||||||
async fn configure(&self, account_id: u32) -> Result<()> {
|
async fn configure(&self, account_id: u32) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
ctx.stop_io().await;
|
ctx.stop_io().await;
|
||||||
@@ -457,7 +484,7 @@ impl CommandApi {
|
|||||||
///
|
///
|
||||||
/// This function stops and starts IO as needed.
|
/// This function stops and starts IO as needed.
|
||||||
///
|
///
|
||||||
/// Usually it will be enough to only set `addr` and `imap.password`,
|
/// Usually it will be enough to only set `addr` and `password`,
|
||||||
/// and all the other settings will be autoconfigured.
|
/// and all the other settings will be autoconfigured.
|
||||||
///
|
///
|
||||||
/// During configuration, ConfigureProgress events are emitted;
|
/// During configuration, ConfigureProgress events are emitted;
|
||||||
@@ -478,21 +505,30 @@ impl CommandApi {
|
|||||||
/// from a server encoded in a QR code.
|
/// from a server encoded in a QR code.
|
||||||
/// - [Self::list_transports()] to get a list of all configured transports.
|
/// - [Self::list_transports()] to get a list of all configured transports.
|
||||||
/// - [Self::delete_transport()] to remove a transport.
|
/// - [Self::delete_transport()] to remove a transport.
|
||||||
async fn add_transport(&self, account_id: u32, param: EnteredLoginParam) -> Result<()> {
|
async fn add_or_update_transport(
|
||||||
|
&self,
|
||||||
|
account_id: u32,
|
||||||
|
param: EnteredLoginParam,
|
||||||
|
) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
ctx.add_transport(¶m.try_into()?).await
|
ctx.add_or_update_transport(&mut param.try_into()?).await
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Deprecated 2025-04. Alias for [Self::add_or_update_transport()].
|
||||||
|
async fn add_transport(&self, account_id: u32, param: EnteredLoginParam) -> Result<()> {
|
||||||
|
self.add_or_update_transport(account_id, param).await
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Adds a new email account as a transport
|
/// Adds a new email account as a transport
|
||||||
/// using the server encoded in the QR code.
|
/// using the server encoded in the QR code.
|
||||||
/// See [Self::add_transport].
|
/// See [Self::add_or_update_transport].
|
||||||
async fn add_transport_from_qr(&self, account_id: u32, qr: String) -> Result<()> {
|
async fn add_transport_from_qr(&self, account_id: u32, qr: String) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
ctx.add_transport_from_qr(&qr).await
|
ctx.add_transport_from_qr(&qr).await
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the list of all email accounts that are used as a transport in the current profile.
|
/// Returns the list of all email accounts that are used as a transport in the current profile.
|
||||||
/// Use [Self::add_transport()] to add or change a transport
|
/// Use [Self::add_or_update_transport()] to add or change a transport
|
||||||
/// and [Self::delete_transport()] to delete a transport.
|
/// and [Self::delete_transport()] to delete a transport.
|
||||||
async fn list_transports(&self, account_id: u32) -> Result<Vec<EnteredLoginParam>> {
|
async fn list_transports(&self, account_id: u32) -> Result<Vec<EnteredLoginParam>> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
@@ -850,6 +886,38 @@ impl CommandApi {
|
|||||||
Ok(chat_id.to_u32())
|
Ok(chat_id.to_u32())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Like `secure_join()`, but allows to pass a source and a UI-path.
|
||||||
|
/// You only need this if your UI has an option to send statistics
|
||||||
|
/// to Delta Chat's developers.
|
||||||
|
///
|
||||||
|
/// **source**: The source where the QR code came from.
|
||||||
|
/// E.g. a link that was clicked inside or outside Delta Chat,
|
||||||
|
/// the "Paste from Clipboard" action,
|
||||||
|
/// the "Load QR code as image" action,
|
||||||
|
/// or a QR code scan.
|
||||||
|
///
|
||||||
|
/// **uipath**: Which UI path did the user use to arrive at the QR code screen.
|
||||||
|
/// If the SecurejoinSource was ExternalLink or InternalLink,
|
||||||
|
/// pass `None` here, because the QR code screen wasn't even opened.
|
||||||
|
/// ```
|
||||||
|
async fn secure_join_with_ux_info(
|
||||||
|
&self,
|
||||||
|
account_id: u32,
|
||||||
|
qr: String,
|
||||||
|
source: Option<SecurejoinSource>,
|
||||||
|
uipath: Option<SecurejoinUiPath>,
|
||||||
|
) -> Result<u32> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
let chat_id = securejoin::join_securejoin_with_ux_info(
|
||||||
|
&ctx,
|
||||||
|
&qr,
|
||||||
|
source.map(Into::into),
|
||||||
|
uipath.map(Into::into),
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
Ok(chat_id.to_u32())
|
||||||
|
}
|
||||||
|
|
||||||
async fn leave_group(&self, account_id: u32, chat_id: u32) -> Result<()> {
|
async fn leave_group(&self, account_id: u32, chat_id: u32) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
remove_contact_from_chat(&ctx, ChatId::new(chat_id), ContactId::SELF).await
|
remove_contact_from_chat(&ctx, ChatId::new(chat_id), ContactId::SELF).await
|
||||||
@@ -898,7 +966,7 @@ impl CommandApi {
|
|||||||
/// explicitly as it may happen that oneself gets removed from a still existing
|
/// explicitly as it may happen that oneself gets removed from a still existing
|
||||||
/// group
|
/// group
|
||||||
///
|
///
|
||||||
/// - for broadcasts, all recipients are returned, DC_CONTACT_ID_SELF is not included
|
/// - for broadcast channels, all recipients are returned, DC_CONTACT_ID_SELF is not included
|
||||||
///
|
///
|
||||||
/// - for mailing lists, the behavior is not documented currently, we will decide on that later.
|
/// - for mailing lists, the behavior is not documented currently, we will decide on that later.
|
||||||
/// for now, the UI should not show the list for mailing lists.
|
/// for now, the UI should not show the list for mailing lists.
|
||||||
@@ -917,7 +985,7 @@ impl CommandApi {
|
|||||||
Ok(contacts.iter().map(|id| id.to_u32()).collect::<Vec<u32>>())
|
Ok(contacts.iter().map(|id| id.to_u32()).collect::<Vec<u32>>())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create a new group chat.
|
/// Create a new encrypted group chat (with key-contacts).
|
||||||
///
|
///
|
||||||
/// After creation,
|
/// After creation,
|
||||||
/// the group has one member with the ID DC_CONTACT_ID_SELF
|
/// the group has one member with the ID DC_CONTACT_ID_SELF
|
||||||
@@ -933,32 +1001,53 @@ impl CommandApi {
|
|||||||
/// To check, if a chat is still unpromoted, you can look at the `is_unpromoted` property of `BasicChat` or `FullChat`.
|
/// To check, if a chat is still unpromoted, you can look at the `is_unpromoted` property of `BasicChat` or `FullChat`.
|
||||||
/// This may be useful if you want to show some help for just created groups.
|
/// This may be useful if you want to show some help for just created groups.
|
||||||
///
|
///
|
||||||
/// @param protect If set to 1 the function creates group with protection initially enabled.
|
/// `protect` argument is deprecated as of 2025-10-22 and is left for compatibility.
|
||||||
/// Only verified members are allowed in these groups
|
/// Pass `false` here.
|
||||||
/// and end-to-end-encryption is always enabled.
|
async fn create_group_chat(
|
||||||
async fn create_group_chat(&self, account_id: u32, name: String, protect: bool) -> Result<u32> {
|
&self,
|
||||||
|
account_id: u32,
|
||||||
|
name: String,
|
||||||
|
_protect: bool,
|
||||||
|
) -> Result<u32> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
let protect = match protect {
|
chat::create_group(&ctx, &name).await.map(|id| id.to_u32())
|
||||||
true => ProtectionStatus::Protected,
|
}
|
||||||
false => ProtectionStatus::Unprotected,
|
|
||||||
};
|
/// Create a new unencrypted group chat.
|
||||||
chat::create_group_chat(&ctx, protect, &name)
|
///
|
||||||
|
/// Same as [`Self::create_group_chat`], but the chat is unencrypted and can only have
|
||||||
|
/// address-contacts.
|
||||||
|
async fn create_group_chat_unencrypted(&self, account_id: u32, name: String) -> Result<u32> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
chat::create_group_unencrypted(&ctx, &name)
|
||||||
.await
|
.await
|
||||||
.map(|id| id.to_u32())
|
.map(|id| id.to_u32())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create a new broadcast list.
|
/// Deprecated 2025-07 in favor of create_broadcast().
|
||||||
///
|
|
||||||
/// Broadcast lists are similar to groups on the sending device,
|
|
||||||
/// however, recipients get the messages in a read-only chat
|
|
||||||
/// and will see who the other members are.
|
|
||||||
///
|
|
||||||
/// For historical reasons, this function does not take a name directly,
|
|
||||||
/// instead you have to set the name using dc_set_chat_name()
|
|
||||||
/// after creating the broadcast list.
|
|
||||||
async fn create_broadcast_list(&self, account_id: u32) -> Result<u32> {
|
async fn create_broadcast_list(&self, account_id: u32) -> Result<u32> {
|
||||||
|
self.create_broadcast(account_id, "Channel".to_string())
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a new, outgoing **broadcast channel**
|
||||||
|
/// (called "Channel" in the UI).
|
||||||
|
///
|
||||||
|
/// Broadcast channels are similar to groups on the sending device,
|
||||||
|
/// however, recipients get the messages in a read-only chat
|
||||||
|
/// and will not see who the other members are.
|
||||||
|
///
|
||||||
|
/// Called `broadcast` here rather than `channel`,
|
||||||
|
/// because the word "channel" already appears a lot in the code,
|
||||||
|
/// which would make it hard to grep for it.
|
||||||
|
///
|
||||||
|
/// After creation, the chat contains no recipients and is in _unpromoted_ state;
|
||||||
|
/// see [`CommandApi::create_group_chat`] for more information on the unpromoted state.
|
||||||
|
///
|
||||||
|
/// Returns the created chat's id.
|
||||||
|
async fn create_broadcast(&self, account_id: u32, chat_name: String) -> Result<u32> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
chat::create_broadcast_list(&ctx)
|
chat::create_broadcast(&ctx, chat_name)
|
||||||
.await
|
.await
|
||||||
.map(|id| id.to_u32())
|
.map(|id| id.to_u32())
|
||||||
}
|
}
|
||||||
@@ -1002,7 +1091,7 @@ impl CommandApi {
|
|||||||
&self,
|
&self,
|
||||||
account_id: u32,
|
account_id: u32,
|
||||||
chat_id: u32,
|
chat_id: u32,
|
||||||
visibility: JSONRPCChatVisibility,
|
visibility: JsonrpcChatVisibility,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
|
||||||
@@ -1169,8 +1258,10 @@ impl CommandApi {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Returns all messages of a particular chat.
|
/// Returns all messages of a particular chat.
|
||||||
/// If `add_daymarker` is `true`, it will return them as
|
///
|
||||||
/// `DC_MSG_ID_DAYMARKER`, e.g. [1234, 1237, 9, 1239].
|
/// * `add_daymarker` - If `true`, add day markers as `DC_MSG_ID_DAYMARKER` to the result,
|
||||||
|
/// e.g. [1234, 1237, 9, 1239]. The day marker timestamp is the midnight one for the
|
||||||
|
/// corresponding (following) day in the local timezone.
|
||||||
async fn get_message_ids(
|
async fn get_message_ids(
|
||||||
&self,
|
&self,
|
||||||
account_id: u32,
|
account_id: u32,
|
||||||
@@ -1205,7 +1296,7 @@ impl CommandApi {
|
|||||||
chat_id: u32,
|
chat_id: u32,
|
||||||
info_only: bool,
|
info_only: bool,
|
||||||
add_daymarker: bool,
|
add_daymarker: bool,
|
||||||
) -> Result<Vec<JSONRPCMessageListItem>> {
|
) -> Result<Vec<JsonrpcMessageListItem>> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
let msg = get_chat_msgs_ex(
|
let msg = get_chat_msgs_ex(
|
||||||
&ctx,
|
&ctx,
|
||||||
@@ -1219,7 +1310,7 @@ impl CommandApi {
|
|||||||
Ok(msg
|
Ok(msg
|
||||||
.iter()
|
.iter()
|
||||||
.map(|chat_item| (*chat_item).into())
|
.map(|chat_item| (*chat_item).into())
|
||||||
.collect::<Vec<JSONRPCMessageListItem>>())
|
.collect::<Vec<JsonrpcMessageListItem>>())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn get_message(&self, account_id: u32, msg_id: u32) -> Result<MessageObject> {
|
async fn get_message(&self, account_id: u32, msg_id: u32) -> Result<MessageObject> {
|
||||||
@@ -1413,7 +1504,14 @@ impl CommandApi {
|
|||||||
|
|
||||||
/// Add a single contact as a result of an explicit user action.
|
/// Add a single contact as a result of an explicit user action.
|
||||||
///
|
///
|
||||||
/// Returns contact id of the created or existing contact
|
/// This will always create or look up an address-contact,
|
||||||
|
/// i.e. a contact identified by an email address,
|
||||||
|
/// with all messages sent to and from this contact being unencrypted.
|
||||||
|
/// If the user just clicked on an email address,
|
||||||
|
/// you should first check [`Self::lookup_contact_id_by_addr`]/`lookupContactIdByAddr.`,
|
||||||
|
/// and only if there is no contact yet, call this function here.
|
||||||
|
///
|
||||||
|
/// Returns contact id of the created or existing contact.
|
||||||
async fn create_contact(
|
async fn create_contact(
|
||||||
&self,
|
&self,
|
||||||
account_id: u32,
|
account_id: u32,
|
||||||
@@ -1465,6 +1563,14 @@ impl CommandApi {
|
|||||||
Ok(contacts)
|
Ok(contacts)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns ids of known and unblocked contacts.
|
||||||
|
///
|
||||||
|
/// By default, key-contacts are listed.
|
||||||
|
///
|
||||||
|
/// * `list_flags` - A combination of flags:
|
||||||
|
/// - `DC_GCL_ADD_SELF` - Add SELF unless filtered by other parameters.
|
||||||
|
/// - `DC_GCL_ADDRESS` - List address-contacts instead of key-contacts.
|
||||||
|
/// * `query` - A string to filter the list.
|
||||||
async fn get_contact_ids(
|
async fn get_contact_ids(
|
||||||
&self,
|
&self,
|
||||||
account_id: u32,
|
account_id: u32,
|
||||||
@@ -1476,8 +1582,10 @@ impl CommandApi {
|
|||||||
Ok(contacts.into_iter().map(|c| c.to_u32()).collect())
|
Ok(contacts.into_iter().map(|c| c.to_u32()).collect())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get a list of contacts.
|
/// Returns known and unblocked contacts.
|
||||||
/// (formerly called getContacts2 in desktop)
|
///
|
||||||
|
/// Formerly called `getContacts2` in Desktop.
|
||||||
|
/// See [`Self::get_contact_ids`] for parameters and more info.
|
||||||
async fn get_contacts(
|
async fn get_contacts(
|
||||||
&self,
|
&self,
|
||||||
account_id: u32,
|
account_id: u32,
|
||||||
@@ -1528,15 +1636,7 @@ impl CommandApi {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Resets contact encryption.
|
/// Sets display name for existing contact.
|
||||||
async fn reset_contact_encryption(&self, account_id: u32, contact_id: u32) -> Result<()> {
|
|
||||||
let ctx = self.get_context(account_id).await?;
|
|
||||||
let contact_id = ContactId::new(contact_id);
|
|
||||||
|
|
||||||
contact_id.reset_encryption(&ctx).await?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn change_contact_name(
|
async fn change_contact_name(
|
||||||
&self,
|
&self,
|
||||||
account_id: u32,
|
account_id: u32,
|
||||||
@@ -1545,9 +1645,7 @@ impl CommandApi {
|
|||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
let contact_id = ContactId::new(contact_id);
|
let contact_id = ContactId::new(contact_id);
|
||||||
let contact = Contact::get_by_id(&ctx, contact_id).await?;
|
contact_id.set_name(&ctx, &name).await?;
|
||||||
let addr = contact.get_addr();
|
|
||||||
Contact::create(&ctx, &name, addr).await?;
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1563,9 +1661,19 @@ impl CommandApi {
|
|||||||
Contact::get_encrinfo(&ctx, ContactId::new(contact_id)).await
|
Contact::get_encrinfo(&ctx, ContactId::new(contact_id)).await
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Check if an e-mail address belongs to a known and unblocked contact.
|
/// Looks up a known and unblocked contact with a given e-mail address.
|
||||||
/// To get a list of all known and unblocked contacts, use contacts_get_contacts().
|
/// To get a list of all known and unblocked contacts, use contacts_get_contacts().
|
||||||
///
|
///
|
||||||
|
/// **POTENTIAL SECURITY ISSUE**: If there are multiple contacts with this address
|
||||||
|
/// (e.g. an address-contact and a key-contact),
|
||||||
|
/// this looks up the most recently seen contact,
|
||||||
|
/// i.e. which contact is returned depends on which contact last sent a message.
|
||||||
|
/// If the user just clicked on a mailto: link, then this is the best thing you can do.
|
||||||
|
/// But **DO NOT** internally represent contacts by their email address
|
||||||
|
/// and do not use this function to look them up;
|
||||||
|
/// otherwise this function will sometimes look up the wrong contact.
|
||||||
|
/// Instead, you should internally represent contacts by their ids.
|
||||||
|
///
|
||||||
/// To validate an e-mail address independently of the contact database
|
/// To validate an e-mail address independently of the contact database
|
||||||
/// use check_email_validity().
|
/// use check_email_validity().
|
||||||
async fn lookup_contact_id_by_addr(
|
async fn lookup_contact_id_by_addr(
|
||||||
@@ -1721,13 +1829,13 @@ impl CommandApi {
|
|||||||
|
|
||||||
/// Offers a backup for remote devices to retrieve.
|
/// Offers a backup for remote devices to retrieve.
|
||||||
///
|
///
|
||||||
/// Can be cancelled by stopping the ongoing process. Success or failure can be tracked
|
/// Can be canceled by stopping the ongoing process. Success or failure can be tracked
|
||||||
/// via the `ImexProgress` event which should either reach `1000` for success or `0` for
|
/// via the `ImexProgress` event which should either reach `1000` for success or `0` for
|
||||||
/// failure.
|
/// failure.
|
||||||
///
|
///
|
||||||
/// This **stops IO** while it is running.
|
/// This **stops IO** while it is running.
|
||||||
///
|
///
|
||||||
/// Returns once a remote device has retrieved the backup, or is cancelled.
|
/// Returns once a remote device has retrieved the backup, or is canceled.
|
||||||
async fn provide_backup(&self, account_id: u32) -> Result<()> {
|
async fn provide_backup(&self, account_id: u32) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
|
||||||
@@ -1793,7 +1901,7 @@ impl CommandApi {
|
|||||||
/// This retrieves the backup from a remote device over the network and imports it into
|
/// This retrieves the backup from a remote device over the network and imports it into
|
||||||
/// the current device.
|
/// the current device.
|
||||||
///
|
///
|
||||||
/// Can be cancelled by stopping the ongoing process.
|
/// Can be canceled by stopping the ongoing process.
|
||||||
///
|
///
|
||||||
/// Do not forget to call start_io on the account after a successful import,
|
/// Do not forget to call start_io on the account after a successful import,
|
||||||
/// otherwise it will not connect to the email server.
|
/// otherwise it will not connect to the email server.
|
||||||
@@ -1831,7 +1939,7 @@ impl CommandApi {
|
|||||||
/// If the connectivity changes, a #DC_EVENT_CONNECTIVITY_CHANGED will be emitted.
|
/// If the connectivity changes, a #DC_EVENT_CONNECTIVITY_CHANGED will be emitted.
|
||||||
async fn get_connectivity(&self, account_id: u32) -> Result<u32> {
|
async fn get_connectivity(&self, account_id: u32) -> Result<u32> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
Ok(ctx.get_connectivity().await as u32)
|
Ok(ctx.get_connectivity() as u32)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get an overview of the current connectivity, and possibly more statistics.
|
/// Get an overview of the current connectivity, and possibly more statistics.
|
||||||
@@ -1906,16 +2014,19 @@ impl CommandApi {
|
|||||||
instance_msg_id: u32,
|
instance_msg_id: u32,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
let fut = send_webxdc_realtime_advertisement(&ctx, MsgId::new(instance_msg_id)).await?;
|
if let Some(fut) =
|
||||||
if let Some(fut) = fut {
|
send_webxdc_realtime_advertisement(&ctx, MsgId::new(instance_msg_id)).await?
|
||||||
tokio::spawn(async move {
|
{
|
||||||
fut.await.ok();
|
tokio::spawn(fut);
|
||||||
info!(ctx, "send_webxdc_realtime_advertisement done")
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Leaves the gossip of the webxdc with the given message id.
|
||||||
|
///
|
||||||
|
/// NB: When this is called before closing a webxdc app in UIs, it must be guaranteed that
|
||||||
|
/// `send_webxdc_realtime_*()` functions aren't called for the given `instance_message_id`
|
||||||
|
/// anymore until the app is open again.
|
||||||
async fn leave_webxdc_realtime(&self, account_id: u32, instance_message_id: u32) -> Result<()> {
|
async fn leave_webxdc_realtime(&self, account_id: u32, instance_message_id: u32) -> Result<()> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
leave_webxdc_realtime(&ctx, MsgId::new(instance_message_id)).await
|
leave_webxdc_realtime(&ctx, MsgId::new(instance_message_id)).await
|
||||||
@@ -1947,13 +2058,9 @@ impl CommandApi {
|
|||||||
|
|
||||||
/// Get href from a WebxdcInfoMessage which might include a hash holding
|
/// Get href from a WebxdcInfoMessage which might include a hash holding
|
||||||
/// information about a specific position or state in a webxdc app (optional)
|
/// information about a specific position or state in a webxdc app (optional)
|
||||||
async fn get_webxdc_href(
|
async fn get_webxdc_href(&self, account_id: u32, info_msg_id: u32) -> Result<Option<String>> {
|
||||||
&self,
|
|
||||||
account_id: u32,
|
|
||||||
instance_msg_id: u32,
|
|
||||||
) -> Result<Option<String>> {
|
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
let message = Message::load_from_db(&ctx, MsgId::new(instance_msg_id)).await?;
|
let message = Message::load_from_db(&ctx, MsgId::new(info_msg_id)).await?;
|
||||||
Ok(message.get_webxdc_href())
|
Ok(message.get_webxdc_href())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1997,6 +2104,53 @@ impl CommandApi {
|
|||||||
.map(|msg_id| msg_id.to_u32()))
|
.map(|msg_id| msg_id.to_u32()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Starts an outgoing call.
|
||||||
|
async fn place_outgoing_call(
|
||||||
|
&self,
|
||||||
|
account_id: u32,
|
||||||
|
chat_id: u32,
|
||||||
|
place_call_info: String,
|
||||||
|
) -> Result<u32> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
let msg_id = ctx
|
||||||
|
.place_outgoing_call(ChatId::new(chat_id), place_call_info)
|
||||||
|
.await?;
|
||||||
|
Ok(msg_id.to_u32())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Accepts an incoming call.
|
||||||
|
async fn accept_incoming_call(
|
||||||
|
&self,
|
||||||
|
account_id: u32,
|
||||||
|
msg_id: u32,
|
||||||
|
accept_call_info: String,
|
||||||
|
) -> Result<()> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
ctx.accept_incoming_call(MsgId::new(msg_id), accept_call_info)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Ends incoming or outgoing call.
|
||||||
|
async fn end_call(&self, account_id: u32, msg_id: u32) -> Result<()> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
ctx.end_call(MsgId::new(msg_id)).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns information about the call.
|
||||||
|
async fn call_info(&self, account_id: u32, msg_id: u32) -> Result<JsonrpcCallInfo> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
let call_info = JsonrpcCallInfo::from_msg_id(&ctx, MsgId::new(msg_id)).await?;
|
||||||
|
Ok(call_info)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns JSON with ICE servers, to be used for WebRTC video calls.
|
||||||
|
async fn ice_servers(&self, account_id: u32) -> Result<String> {
|
||||||
|
let ctx = self.get_context(account_id).await?;
|
||||||
|
ice_servers(&ctx).await
|
||||||
|
}
|
||||||
|
|
||||||
/// Makes an HTTP GET request and returns a response.
|
/// Makes an HTTP GET request and returns a response.
|
||||||
///
|
///
|
||||||
/// `url` is the HTTP or HTTPS URL.
|
/// `url` is the HTTP or HTTPS URL.
|
||||||
@@ -2077,7 +2231,7 @@ impl CommandApi {
|
|||||||
&self,
|
&self,
|
||||||
account_id: u32,
|
account_id: u32,
|
||||||
message_id: u32,
|
message_id: u32,
|
||||||
) -> Result<Option<JSONRPCReactions>> {
|
) -> Result<Option<JsonrpcReactions>> {
|
||||||
let ctx = self.get_context(account_id).await?;
|
let ctx = self.get_context(account_id).await?;
|
||||||
let reactions = get_msg_reactions(&ctx, MsgId::new(message_id)).await?;
|
let reactions = get_msg_reactions(&ctx, MsgId::new(message_id)).await?;
|
||||||
if reactions.is_empty() {
|
if reactions.is_empty() {
|
||||||
@@ -2148,13 +2302,6 @@ impl CommandApi {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn send_videochat_invitation(&self, account_id: u32, chat_id: u32) -> Result<u32> {
|
|
||||||
let ctx = self.get_context(account_id).await?;
|
|
||||||
chat::send_videochat_invitation(&ctx, ChatId::new(chat_id))
|
|
||||||
.await
|
|
||||||
.map(|msg_id| msg_id.to_u32())
|
|
||||||
}
|
|
||||||
|
|
||||||
// ---------------------------------------------
|
// ---------------------------------------------
|
||||||
// misc prototyping functions
|
// misc prototyping functions
|
||||||
// that might get removed later again
|
// that might get removed later again
|
||||||
@@ -2185,8 +2332,7 @@ impl CommandApi {
|
|||||||
let message = Message::load_from_db(&ctx, MsgId::new(msg_id)).await?;
|
let message = Message::load_from_db(&ctx, MsgId::new(msg_id)).await?;
|
||||||
ensure!(
|
ensure!(
|
||||||
message.get_viewtype() == Viewtype::Sticker,
|
message.get_viewtype() == Viewtype::Sticker,
|
||||||
"message {} is not a sticker",
|
"message {msg_id} is not a sticker"
|
||||||
msg_id
|
|
||||||
);
|
);
|
||||||
let account_folder = ctx
|
let account_folder = ctx
|
||||||
.get_dbfile()
|
.get_dbfile()
|
||||||
@@ -2276,6 +2422,37 @@ impl CommandApi {
|
|||||||
|
|
||||||
// mimics the old desktop call, will get replaced with something better in the composer rewrite,
|
// mimics the old desktop call, will get replaced with something better in the composer rewrite,
|
||||||
// the better version will just be sending the current draft, though there will be probably something similar with more options to this for the corner cases like setting a marker on the map
|
// the better version will just be sending the current draft, though there will be probably something similar with more options to this for the corner cases like setting a marker on the map
|
||||||
|
/// Send a message to a chat.
|
||||||
|
///
|
||||||
|
/// This function returns after the message has been placed in the sending queue.
|
||||||
|
/// This does not imply that the message was really sent out yet.
|
||||||
|
/// However, from your view, you're done with the message.
|
||||||
|
/// Sooner or later it will find its way.
|
||||||
|
///
|
||||||
|
/// **Attaching files:**
|
||||||
|
///
|
||||||
|
/// Pass the file path in the `file` parameter.
|
||||||
|
/// If `file` is not in the blob directory yet,
|
||||||
|
/// it will be copied into the blob directory.
|
||||||
|
/// If you want, you can delete the file immediately after this function returns.
|
||||||
|
///
|
||||||
|
/// You can also write the attachment directly into the blob directory
|
||||||
|
/// and then pass the path as the `file` parameter;
|
||||||
|
/// this will prevent an unnecessary copying of the file.
|
||||||
|
///
|
||||||
|
/// In `filename`, you can pass the original name of the file,
|
||||||
|
/// which will then be shown in the UI.
|
||||||
|
/// in this case the current name of `file` on the filesystem will be ignored.
|
||||||
|
///
|
||||||
|
/// In order to deduplicate files that contain the same data,
|
||||||
|
/// the file will be named `<hash>.<extension>`, e.g. `ce940175885d7b78f7b7e9f1396611f.jpg`.
|
||||||
|
///
|
||||||
|
/// NOTE:
|
||||||
|
/// - This function will rename the file. To get the new file path, call `get_file()`.
|
||||||
|
/// - The file must not be modified after this function was called.
|
||||||
|
/// - Images etc. will NOT be recoded.
|
||||||
|
/// In order to recode images,
|
||||||
|
/// use `misc_set_draft` and pass `Image` as the viewtype.
|
||||||
#[expect(clippy::too_many_arguments)]
|
#[expect(clippy::too_many_arguments)]
|
||||||
async fn misc_send_msg(
|
async fn misc_send_msg(
|
||||||
&self,
|
&self,
|
||||||
@@ -2375,10 +2552,7 @@ impl CommandApi {
|
|||||||
.to_u32();
|
.to_u32();
|
||||||
Ok(msg_id)
|
Ok(msg_id)
|
||||||
} else {
|
} else {
|
||||||
Err(anyhow!(
|
Err(anyhow!("chat with id {chat_id} doesn't have draft message"))
|
||||||
"chat with id {} doesn't have draft message",
|
|
||||||
chat_id
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
97
deltachat-jsonrpc/src/api/types/calls.rs
Normal file
97
deltachat-jsonrpc/src/api/types/calls.rs
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
use anyhow::{Context as _, Result};
|
||||||
|
|
||||||
|
use deltachat::calls::{call_state, sdp_has_video, CallState};
|
||||||
|
use deltachat::context::Context;
|
||||||
|
use deltachat::message::MsgId;
|
||||||
|
use serde::Serialize;
|
||||||
|
use typescript_type_def::TypeDef;
|
||||||
|
|
||||||
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
|
#[serde(rename = "CallInfo", rename_all = "camelCase")]
|
||||||
|
pub struct JsonrpcCallInfo {
|
||||||
|
/// SDP offer.
|
||||||
|
///
|
||||||
|
/// Can be used to manually answer the call
|
||||||
|
/// even if incoming call event was missed.
|
||||||
|
pub sdp_offer: String,
|
||||||
|
|
||||||
|
/// True if SDP offer has a video.
|
||||||
|
pub has_video: bool,
|
||||||
|
|
||||||
|
/// Call state.
|
||||||
|
///
|
||||||
|
/// For example, if the call is accepted, active, canceled, declined etc.
|
||||||
|
pub state: JsonrpcCallState,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl JsonrpcCallInfo {
|
||||||
|
pub async fn from_msg_id(context: &Context, msg_id: MsgId) -> Result<JsonrpcCallInfo> {
|
||||||
|
let call_info = context.load_call_by_id(msg_id).await?.with_context(|| {
|
||||||
|
format!("Attempting to get call state of non-call message {msg_id}")
|
||||||
|
})?;
|
||||||
|
let sdp_offer = call_info.place_call_info.clone();
|
||||||
|
let has_video = sdp_has_video(&sdp_offer).unwrap_or_default();
|
||||||
|
let state = JsonrpcCallState::from_msg_id(context, msg_id).await?;
|
||||||
|
|
||||||
|
Ok(JsonrpcCallInfo {
|
||||||
|
sdp_offer,
|
||||||
|
has_video,
|
||||||
|
state,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
|
#[serde(rename = "CallState", tag = "kind")]
|
||||||
|
pub enum JsonrpcCallState {
|
||||||
|
/// Fresh incoming or outgoing call that is still ringing.
|
||||||
|
///
|
||||||
|
/// There is no separate state for outgoing call
|
||||||
|
/// that has been dialled but not ringing on the other side yet
|
||||||
|
/// as we don't know whether the other side received our call.
|
||||||
|
Alerting,
|
||||||
|
|
||||||
|
/// Active call.
|
||||||
|
Active,
|
||||||
|
|
||||||
|
/// Completed call that was once active
|
||||||
|
/// and then was terminated for any reason.
|
||||||
|
Completed {
|
||||||
|
/// Call duration in seconds.
|
||||||
|
duration: i64,
|
||||||
|
},
|
||||||
|
|
||||||
|
/// Incoming call that was not picked up within a timeout
|
||||||
|
/// or was explicitly ended by the caller before we picked up.
|
||||||
|
Missed,
|
||||||
|
|
||||||
|
/// Incoming call that was explicitly ended on our side
|
||||||
|
/// before picking up or outgoing call
|
||||||
|
/// that was declined before the timeout.
|
||||||
|
Declined,
|
||||||
|
|
||||||
|
/// Outgoing call that has been canceled on our side
|
||||||
|
/// before receiving a response.
|
||||||
|
///
|
||||||
|
/// Incoming calls cannot be canceled,
|
||||||
|
/// on the receiver side canceled calls
|
||||||
|
/// usually result in missed calls.
|
||||||
|
Canceled,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl JsonrpcCallState {
|
||||||
|
pub async fn from_msg_id(context: &Context, msg_id: MsgId) -> Result<JsonrpcCallState> {
|
||||||
|
let call_state = call_state(context, msg_id).await?;
|
||||||
|
|
||||||
|
let jsonrpc_call_state = match call_state {
|
||||||
|
CallState::Alerting => JsonrpcCallState::Alerting,
|
||||||
|
CallState::Active => JsonrpcCallState::Active,
|
||||||
|
CallState::Completed { duration } => JsonrpcCallState::Completed { duration },
|
||||||
|
CallState::Missed => JsonrpcCallState::Missed,
|
||||||
|
CallState::Declined => JsonrpcCallState::Declined,
|
||||||
|
CallState::Canceled => JsonrpcCallState::Canceled,
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(jsonrpc_call_state)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,7 +6,6 @@ use deltachat::chat::{Chat, ChatId};
|
|||||||
use deltachat::constants::Chattype;
|
use deltachat::constants::Chattype;
|
||||||
use deltachat::contact::{Contact, ContactId};
|
use deltachat::contact::{Contact, ContactId};
|
||||||
use deltachat::context::Context;
|
use deltachat::context::Context;
|
||||||
use num_traits::cast::ToPrimitive;
|
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use typescript_type_def::TypeDef;
|
use typescript_type_def::TypeDef;
|
||||||
|
|
||||||
@@ -19,22 +18,34 @@ pub struct FullChat {
|
|||||||
id: u32,
|
id: u32,
|
||||||
name: String,
|
name: String,
|
||||||
|
|
||||||
/// True if the chat is protected.
|
/// True if the chat is encrypted.
|
||||||
|
/// This means that all messages in the chat are encrypted,
|
||||||
|
/// and all contacts in the chat are "key-contacts",
|
||||||
|
/// i.e. identified by the PGP key fingerprint.
|
||||||
///
|
///
|
||||||
/// UI should display a green checkmark
|
/// False if the chat is unencrypted.
|
||||||
/// in the chat title,
|
/// This means that all messages in the chat are unencrypted,
|
||||||
/// in the chat profile title and
|
/// and all contacts in the chat are "address-contacts",
|
||||||
/// in the chatlist item
|
/// i.e. identified by the email address.
|
||||||
/// if chat protection is enabled.
|
/// The UI should mark this chat e.g. with a mail-letter icon.
|
||||||
/// UI should also display a green checkmark
|
///
|
||||||
/// in the contact profile
|
/// Unencrypted groups are called "ad-hoc groups"
|
||||||
/// if 1:1 chat with this contact exists and is protected.
|
/// and the user can't add/remove members,
|
||||||
is_protected: bool,
|
/// create a QR invite code,
|
||||||
|
/// or set an avatar.
|
||||||
|
/// These options should therefore be disabled in the UI.
|
||||||
|
///
|
||||||
|
/// Note that it can happen that an encrypted chat
|
||||||
|
/// contains unencrypted messages that were received in core <= v1.159.*
|
||||||
|
/// and vice versa.
|
||||||
|
///
|
||||||
|
/// See also `is_key_contact` on `Contact`.
|
||||||
|
is_encrypted: bool,
|
||||||
profile_image: Option<String>, //BLOBS ?
|
profile_image: Option<String>, //BLOBS ?
|
||||||
archived: bool,
|
archived: bool,
|
||||||
pinned: bool,
|
pinned: bool,
|
||||||
// subtitle - will be moved to frontend because it uses translation functions
|
// subtitle - will be moved to frontend because it uses translation functions
|
||||||
chat_type: u32,
|
chat_type: JsonrpcChatType,
|
||||||
is_unpromoted: bool,
|
is_unpromoted: bool,
|
||||||
is_self_talk: bool,
|
is_self_talk: bool,
|
||||||
contacts: Vec<ContactObject>,
|
contacts: Vec<ContactObject>,
|
||||||
@@ -47,8 +58,15 @@ pub struct FullChat {
|
|||||||
fresh_message_counter: usize,
|
fresh_message_counter: usize,
|
||||||
// is_group - please check over chat.type in frontend instead
|
// is_group - please check over chat.type in frontend instead
|
||||||
is_contact_request: bool,
|
is_contact_request: bool,
|
||||||
is_protection_broken: bool,
|
|
||||||
is_device_chat: bool,
|
is_device_chat: bool,
|
||||||
|
/// Note that this is different from
|
||||||
|
/// [`ChatListItem::is_self_in_group`](`crate::api::types::chat_list::ChatListItemFetchResult::ChatListItem::is_self_in_group`).
|
||||||
|
/// This property should only be accessed
|
||||||
|
/// when [`FullChat::chat_type`] is [`Chattype::Group`].
|
||||||
|
//
|
||||||
|
// We could utilize [`Chat::is_self_in_chat`],
|
||||||
|
// but that would be an extra DB query.
|
||||||
self_in_group: bool,
|
self_in_group: bool,
|
||||||
is_muted: bool,
|
is_muted: bool,
|
||||||
ephemeral_timer: u32, //TODO look if there are more important properties in newer core versions
|
ephemeral_timer: u32, //TODO look if there are more important properties in newer core versions
|
||||||
@@ -107,11 +125,11 @@ impl FullChat {
|
|||||||
Ok(FullChat {
|
Ok(FullChat {
|
||||||
id: chat_id,
|
id: chat_id,
|
||||||
name: chat.name.clone(),
|
name: chat.name.clone(),
|
||||||
is_protected: chat.is_protected(),
|
is_encrypted: chat.is_encrypted(context).await?,
|
||||||
profile_image, //BLOBS ?
|
profile_image, //BLOBS ?
|
||||||
archived: chat.get_visibility() == chat::ChatVisibility::Archived,
|
archived: chat.get_visibility() == chat::ChatVisibility::Archived,
|
||||||
pinned: chat.get_visibility() == chat::ChatVisibility::Pinned,
|
pinned: chat.get_visibility() == chat::ChatVisibility::Pinned,
|
||||||
chat_type: chat.get_type().to_u32().context("unknown chat type id")?,
|
chat_type: chat.get_type().into(),
|
||||||
is_unpromoted: chat.is_unpromoted(),
|
is_unpromoted: chat.is_unpromoted(),
|
||||||
is_self_talk: chat.is_self_talk(),
|
is_self_talk: chat.is_self_talk(),
|
||||||
contacts,
|
contacts,
|
||||||
@@ -120,7 +138,6 @@ impl FullChat {
|
|||||||
color,
|
color,
|
||||||
fresh_message_counter,
|
fresh_message_counter,
|
||||||
is_contact_request: chat.is_contact_request(),
|
is_contact_request: chat.is_contact_request(),
|
||||||
is_protection_broken: chat.is_protection_broken(),
|
|
||||||
is_device_chat: chat.is_device_talk(),
|
is_device_chat: chat.is_device_talk(),
|
||||||
self_in_group: contact_ids.contains(&ContactId::SELF),
|
self_in_group: contact_ids.contains(&ContactId::SELF),
|
||||||
is_muted: chat.is_muted(),
|
is_muted: chat.is_muted(),
|
||||||
@@ -148,26 +165,38 @@ pub struct BasicChat {
|
|||||||
id: u32,
|
id: u32,
|
||||||
name: String,
|
name: String,
|
||||||
|
|
||||||
/// True if the chat is protected.
|
/// True if the chat is encrypted.
|
||||||
|
/// This means that all messages in the chat are encrypted,
|
||||||
|
/// and all contacts in the chat are "key-contacts",
|
||||||
|
/// i.e. identified by the PGP key fingerprint.
|
||||||
///
|
///
|
||||||
/// UI should display a green checkmark
|
/// False if the chat is unencrypted.
|
||||||
/// in the chat title,
|
/// This means that all messages in the chat are unencrypted,
|
||||||
/// in the chat profile title and
|
/// and all contacts in the chat are "address-contacts",
|
||||||
/// in the chatlist item
|
/// i.e. identified by the email address.
|
||||||
/// if chat protection is enabled.
|
/// The UI should mark this chat e.g. with a mail-letter icon.
|
||||||
/// UI should also display a green checkmark
|
///
|
||||||
/// in the contact profile
|
/// Unencrypted groups are called "ad-hoc groups"
|
||||||
/// if 1:1 chat with this contact exists and is protected.
|
/// and the user can't add/remove members,
|
||||||
is_protected: bool,
|
/// create a QR invite code,
|
||||||
|
/// or set an avatar.
|
||||||
|
/// These options should therefore be disabled in the UI.
|
||||||
|
///
|
||||||
|
/// Note that it can happen that an encrypted chat
|
||||||
|
/// contains unencrypted messages that were received in core <= v1.159.*
|
||||||
|
/// and vice versa.
|
||||||
|
///
|
||||||
|
/// See also `is_key_contact` on `Contact`.
|
||||||
|
is_encrypted: bool,
|
||||||
profile_image: Option<String>, //BLOBS ?
|
profile_image: Option<String>, //BLOBS ?
|
||||||
archived: bool,
|
archived: bool,
|
||||||
pinned: bool,
|
pinned: bool,
|
||||||
chat_type: u32,
|
chat_type: JsonrpcChatType,
|
||||||
is_unpromoted: bool,
|
is_unpromoted: bool,
|
||||||
is_self_talk: bool,
|
is_self_talk: bool,
|
||||||
color: String,
|
color: String,
|
||||||
is_contact_request: bool,
|
is_contact_request: bool,
|
||||||
is_protection_broken: bool,
|
|
||||||
is_device_chat: bool,
|
is_device_chat: bool,
|
||||||
is_muted: bool,
|
is_muted: bool,
|
||||||
}
|
}
|
||||||
@@ -186,16 +215,15 @@ impl BasicChat {
|
|||||||
Ok(BasicChat {
|
Ok(BasicChat {
|
||||||
id: chat_id,
|
id: chat_id,
|
||||||
name: chat.name.clone(),
|
name: chat.name.clone(),
|
||||||
is_protected: chat.is_protected(),
|
is_encrypted: chat.is_encrypted(context).await?,
|
||||||
profile_image, //BLOBS ?
|
profile_image, //BLOBS ?
|
||||||
archived: chat.get_visibility() == chat::ChatVisibility::Archived,
|
archived: chat.get_visibility() == chat::ChatVisibility::Archived,
|
||||||
pinned: chat.get_visibility() == chat::ChatVisibility::Pinned,
|
pinned: chat.get_visibility() == chat::ChatVisibility::Pinned,
|
||||||
chat_type: chat.get_type().to_u32().context("unknown chat type id")?,
|
chat_type: chat.get_type().into(),
|
||||||
is_unpromoted: chat.is_unpromoted(),
|
is_unpromoted: chat.is_unpromoted(),
|
||||||
is_self_talk: chat.is_self_talk(),
|
is_self_talk: chat.is_self_talk(),
|
||||||
color,
|
color,
|
||||||
is_contact_request: chat.is_contact_request(),
|
is_contact_request: chat.is_contact_request(),
|
||||||
is_protection_broken: chat.is_protection_broken(),
|
|
||||||
is_device_chat: chat.is_device_talk(),
|
is_device_chat: chat.is_device_talk(),
|
||||||
is_muted: chat.is_muted(),
|
is_muted: chat.is_muted(),
|
||||||
})
|
})
|
||||||
@@ -230,18 +258,52 @@ impl MuteDuration {
|
|||||||
|
|
||||||
#[derive(Clone, Serialize, Deserialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Clone, Serialize, Deserialize, TypeDef, schemars::JsonSchema)]
|
||||||
#[serde(rename = "ChatVisibility")]
|
#[serde(rename = "ChatVisibility")]
|
||||||
pub enum JSONRPCChatVisibility {
|
pub enum JsonrpcChatVisibility {
|
||||||
Normal,
|
Normal,
|
||||||
Archived,
|
Archived,
|
||||||
Pinned,
|
Pinned,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl JSONRPCChatVisibility {
|
impl JsonrpcChatVisibility {
|
||||||
pub fn into_core_type(self) -> ChatVisibility {
|
pub fn into_core_type(self) -> ChatVisibility {
|
||||||
match self {
|
match self {
|
||||||
JSONRPCChatVisibility::Normal => ChatVisibility::Normal,
|
JsonrpcChatVisibility::Normal => ChatVisibility::Normal,
|
||||||
JSONRPCChatVisibility::Archived => ChatVisibility::Archived,
|
JsonrpcChatVisibility::Archived => ChatVisibility::Archived,
|
||||||
JSONRPCChatVisibility::Pinned => ChatVisibility::Pinned,
|
JsonrpcChatVisibility::Pinned => ChatVisibility::Pinned,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Serialize, Deserialize, PartialEq, TypeDef, schemars::JsonSchema)]
|
||||||
|
#[serde(rename = "ChatType")]
|
||||||
|
pub enum JsonrpcChatType {
|
||||||
|
Single,
|
||||||
|
Group,
|
||||||
|
Mailinglist,
|
||||||
|
OutBroadcast,
|
||||||
|
InBroadcast,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Chattype> for JsonrpcChatType {
|
||||||
|
fn from(chattype: Chattype) -> Self {
|
||||||
|
match chattype {
|
||||||
|
Chattype::Single => JsonrpcChatType::Single,
|
||||||
|
Chattype::Group => JsonrpcChatType::Group,
|
||||||
|
Chattype::Mailinglist => JsonrpcChatType::Mailinglist,
|
||||||
|
Chattype::OutBroadcast => JsonrpcChatType::OutBroadcast,
|
||||||
|
Chattype::InBroadcast => JsonrpcChatType::InBroadcast,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<JsonrpcChatType> for Chattype {
|
||||||
|
fn from(chattype: JsonrpcChatType) -> Self {
|
||||||
|
match chattype {
|
||||||
|
JsonrpcChatType::Single => Chattype::Single,
|
||||||
|
JsonrpcChatType::Group => Chattype::Group,
|
||||||
|
JsonrpcChatType::Mailinglist => Chattype::Mailinglist,
|
||||||
|
JsonrpcChatType::OutBroadcast => Chattype::OutBroadcast,
|
||||||
|
JsonrpcChatType::InBroadcast => Chattype::InBroadcast,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ use anyhow::{Context, Result};
|
|||||||
use deltachat::chat::{Chat, ChatId};
|
use deltachat::chat::{Chat, ChatId};
|
||||||
use deltachat::chatlist::get_last_message_for_chat;
|
use deltachat::chatlist::get_last_message_for_chat;
|
||||||
use deltachat::constants::*;
|
use deltachat::constants::*;
|
||||||
use deltachat::contact::{Contact, ContactId};
|
use deltachat::contact::Contact;
|
||||||
use deltachat::{
|
use deltachat::{
|
||||||
chat::{get_chat_contacts, ChatVisibility},
|
chat::{get_chat_contacts, ChatVisibility},
|
||||||
chatlist::Chatlist,
|
chatlist::Chatlist,
|
||||||
@@ -11,6 +11,7 @@ use num_traits::cast::ToPrimitive;
|
|||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use typescript_type_def::TypeDef;
|
use typescript_type_def::TypeDef;
|
||||||
|
|
||||||
|
use super::chat::JsonrpcChatType;
|
||||||
use super::color_int_to_hex_string;
|
use super::color_int_to_hex_string;
|
||||||
use super::message::MessageViewtype;
|
use super::message::MessageViewtype;
|
||||||
|
|
||||||
@@ -23,13 +24,38 @@ pub enum ChatListItemFetchResult {
|
|||||||
name: String,
|
name: String,
|
||||||
avatar_path: Option<String>,
|
avatar_path: Option<String>,
|
||||||
color: String,
|
color: String,
|
||||||
|
chat_type: JsonrpcChatType,
|
||||||
last_updated: Option<i64>,
|
last_updated: Option<i64>,
|
||||||
summary_text1: String,
|
summary_text1: String,
|
||||||
summary_text2: String,
|
summary_text2: String,
|
||||||
summary_status: u32,
|
summary_status: u32,
|
||||||
/// showing preview if last chat message is image
|
/// showing preview if last chat message is image
|
||||||
summary_preview_image: Option<String>,
|
summary_preview_image: Option<String>,
|
||||||
is_protected: bool,
|
|
||||||
|
/// True if the chat is encrypted.
|
||||||
|
/// This means that all messages in the chat are encrypted,
|
||||||
|
/// and all contacts in the chat are "key-contacts",
|
||||||
|
/// i.e. identified by the PGP key fingerprint.
|
||||||
|
///
|
||||||
|
/// False if the chat is unencrypted.
|
||||||
|
/// This means that all messages in the chat are unencrypted,
|
||||||
|
/// and all contacts in the chat are "address-contacts",
|
||||||
|
/// i.e. identified by the email address.
|
||||||
|
/// The UI should mark this chat e.g. with a mail-letter icon.
|
||||||
|
///
|
||||||
|
/// Unencrypted groups are called "ad-hoc groups"
|
||||||
|
/// and the user can't add/remove members,
|
||||||
|
/// create a QR invite code,
|
||||||
|
/// or set an avatar.
|
||||||
|
/// These options should therefore be disabled in the UI.
|
||||||
|
///
|
||||||
|
/// Note that it can happen that an encrypted chat
|
||||||
|
/// contains unencrypted messages that were received in core <= v1.159.*
|
||||||
|
/// and vice versa.
|
||||||
|
///
|
||||||
|
/// See also `is_key_contact` on `Contact`.
|
||||||
|
is_encrypted: bool,
|
||||||
|
/// deprecated 2025-07, use chat_type instead
|
||||||
is_group: bool,
|
is_group: bool,
|
||||||
fresh_message_counter: usize,
|
fresh_message_counter: usize,
|
||||||
is_self_talk: bool,
|
is_self_talk: bool,
|
||||||
@@ -40,8 +66,6 @@ pub enum ChatListItemFetchResult {
|
|||||||
is_pinned: bool,
|
is_pinned: bool,
|
||||||
is_muted: bool,
|
is_muted: bool,
|
||||||
is_contact_request: bool,
|
is_contact_request: bool,
|
||||||
/// true when chat is a broadcastlist
|
|
||||||
is_broadcast: bool,
|
|
||||||
/// contact id if this is a dm chat (for view profile entry in context menu)
|
/// contact id if this is a dm chat (for view profile entry in context menu)
|
||||||
dm_chat_contact: Option<u32>,
|
dm_chat_contact: Option<u32>,
|
||||||
was_seen_recently: bool,
|
was_seen_recently: bool,
|
||||||
@@ -103,11 +127,8 @@ pub(crate) async fn get_chat_list_item_by_id(
|
|||||||
None => (None, None),
|
None => (None, None),
|
||||||
};
|
};
|
||||||
|
|
||||||
let chat_contacts = get_chat_contacts(ctx, chat_id).await?;
|
|
||||||
|
|
||||||
let self_in_group = chat_contacts.contains(&ContactId::SELF);
|
|
||||||
|
|
||||||
let (dm_chat_contact, was_seen_recently) = if chat.get_type() == Chattype::Single {
|
let (dm_chat_contact, was_seen_recently) = if chat.get_type() == Chattype::Single {
|
||||||
|
let chat_contacts = get_chat_contacts(ctx, chat_id).await?;
|
||||||
let contact = chat_contacts.first();
|
let contact = chat_contacts.first();
|
||||||
let was_seen_recently = match contact {
|
let was_seen_recently = match contact {
|
||||||
Some(contact) => Contact::get_by_id(ctx, *contact)
|
Some(contact) => Contact::get_by_id(ctx, *contact)
|
||||||
@@ -131,23 +152,23 @@ pub(crate) async fn get_chat_list_item_by_id(
|
|||||||
name: chat.get_name().to_owned(),
|
name: chat.get_name().to_owned(),
|
||||||
avatar_path,
|
avatar_path,
|
||||||
color,
|
color,
|
||||||
|
chat_type: chat.get_type().into(),
|
||||||
last_updated,
|
last_updated,
|
||||||
summary_text1,
|
summary_text1,
|
||||||
summary_text2,
|
summary_text2,
|
||||||
summary_status: summary.state.to_u32().expect("impossible"), // idea and a function to transform the constant to strings? or return string enum
|
summary_status: summary.state.to_u32().expect("impossible"), // idea and a function to transform the constant to strings? or return string enum
|
||||||
summary_preview_image,
|
summary_preview_image,
|
||||||
is_protected: chat.is_protected(),
|
is_encrypted: chat.is_encrypted(ctx).await?,
|
||||||
is_group: chat.get_type() == Chattype::Group,
|
is_group: chat.get_type() == Chattype::Group,
|
||||||
fresh_message_counter,
|
fresh_message_counter,
|
||||||
is_self_talk: chat.is_self_talk(),
|
is_self_talk: chat.is_self_talk(),
|
||||||
is_device_talk: chat.is_device_talk(),
|
is_device_talk: chat.is_device_talk(),
|
||||||
is_self_in_group: self_in_group,
|
is_self_in_group: chat.is_self_in_chat(ctx).await?,
|
||||||
is_sending_location: chat.is_sending_locations(),
|
is_sending_location: chat.is_sending_locations(),
|
||||||
is_archived: visibility == ChatVisibility::Archived,
|
is_archived: visibility == ChatVisibility::Archived,
|
||||||
is_pinned: visibility == ChatVisibility::Pinned,
|
is_pinned: visibility == ChatVisibility::Pinned,
|
||||||
is_muted: chat.is_muted(),
|
is_muted: chat.is_muted(),
|
||||||
is_contact_request: chat.is_contact_request(),
|
is_contact_request: chat.is_contact_request(),
|
||||||
is_broadcast: chat.get_type() == Chattype::Broadcast,
|
|
||||||
dm_chat_contact,
|
dm_chat_contact,
|
||||||
was_seen_recently,
|
was_seen_recently,
|
||||||
last_message_type: message_type,
|
last_message_type: message_type,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use deltachat::color;
|
|
||||||
use deltachat::context::Context;
|
use deltachat::context::Context;
|
||||||
|
use deltachat::key::{DcKey, SignedPublicKey};
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use typescript_type_def::TypeDef;
|
use typescript_type_def::TypeDef;
|
||||||
|
|
||||||
@@ -19,29 +19,48 @@ pub struct ContactObject {
|
|||||||
profile_image: Option<String>, // BLOBS
|
profile_image: Option<String>, // BLOBS
|
||||||
name_and_addr: String,
|
name_and_addr: String,
|
||||||
is_blocked: bool,
|
is_blocked: bool,
|
||||||
|
|
||||||
|
/// Is the contact a key contact.
|
||||||
|
is_key_contact: bool,
|
||||||
|
|
||||||
|
/// Is encryption available for this contact.
|
||||||
|
///
|
||||||
|
/// This can only be true for key-contacts.
|
||||||
|
/// However, it is possible to have a key-contact
|
||||||
|
/// for which encryption is not available because we don't have a key yet,
|
||||||
|
/// e.g. if we just scanned the fingerprint from a QR code.
|
||||||
e2ee_avail: bool,
|
e2ee_avail: bool,
|
||||||
|
|
||||||
/// True if the contact can be added to verified groups.
|
/// True if the contact
|
||||||
|
/// can be added to protected chats
|
||||||
|
/// because SELF and contact have verified their fingerprints in both directions.
|
||||||
///
|
///
|
||||||
/// If this is true
|
/// See [`Self::verifier_id`]/`Contact.verifierId` for a guidance how to display these information.
|
||||||
/// UI should display green checkmark after the contact name
|
|
||||||
/// in contact list items,
|
|
||||||
/// in chat member list items
|
|
||||||
/// and in profiles if no chat with the contact exist.
|
|
||||||
is_verified: bool,
|
is_verified: bool,
|
||||||
|
|
||||||
/// True if the contact profile title should have a green checkmark.
|
/// The contact ID that verified a contact.
|
||||||
///
|
///
|
||||||
/// This indicates whether 1:1 chat has a green checkmark
|
/// As verifier may be unknown,
|
||||||
/// or will have a green checkmark if created.
|
/// use [`Self::is_verified`]/`Contact.isVerified` to check if a contact can be added to a protected chat.
|
||||||
is_profile_verified: bool,
|
|
||||||
|
|
||||||
/// The ID of the contact that verified this contact.
|
|
||||||
///
|
///
|
||||||
/// If this is present,
|
/// UI should display the information in the contact's profile as follows:
|
||||||
/// display a green checkmark and "Introduced by ..."
|
///
|
||||||
/// string followed by the verifier contact name and address
|
/// - If `verifierId` != 0,
|
||||||
/// in the contact profile.
|
/// display text "Introduced by ..."
|
||||||
|
/// with the name and address of the contact
|
||||||
|
/// formatted by `name_and_addr`/`nameAndAddr`.
|
||||||
|
/// Prefix the text by a green checkmark.
|
||||||
|
///
|
||||||
|
/// - If `verifierId` == 0 and `isVerified` != 0,
|
||||||
|
/// display "Introduced" prefixed by a green checkmark.
|
||||||
|
///
|
||||||
|
/// - if `verifierId` == 0 and `isVerified` == 0,
|
||||||
|
/// display nothing
|
||||||
|
///
|
||||||
|
/// This contains the contact ID of the verifier.
|
||||||
|
/// If it is `DC_CONTACT_ID_SELF`, we verified the contact ourself.
|
||||||
|
/// If it is None/Null, we don't have verifier information or
|
||||||
|
/// the contact is not verified.
|
||||||
verifier_id: Option<u32>,
|
verifier_id: Option<u32>,
|
||||||
|
|
||||||
/// the contact's last seen timestamp
|
/// the contact's last seen timestamp
|
||||||
@@ -62,11 +81,11 @@ impl ContactObject {
|
|||||||
None => None,
|
None => None,
|
||||||
};
|
};
|
||||||
let is_verified = contact.is_verified(context).await?;
|
let is_verified = contact.is_verified(context).await?;
|
||||||
let is_profile_verified = contact.is_profile_verified(context).await?;
|
|
||||||
|
|
||||||
let verifier_id = contact
|
let verifier_id = contact
|
||||||
.get_verifier_id(context)
|
.get_verifier_id(context)
|
||||||
.await?
|
.await?
|
||||||
|
.flatten()
|
||||||
.map(|contact_id| contact_id.to_u32());
|
.map(|contact_id| contact_id.to_u32());
|
||||||
|
|
||||||
Ok(ContactObject {
|
Ok(ContactObject {
|
||||||
@@ -80,9 +99,9 @@ impl ContactObject {
|
|||||||
profile_image, //BLOBS
|
profile_image, //BLOBS
|
||||||
name_and_addr: contact.get_name_n_addr(),
|
name_and_addr: contact.get_name_n_addr(),
|
||||||
is_blocked: contact.is_blocked(),
|
is_blocked: contact.is_blocked(),
|
||||||
|
is_key_contact: contact.is_key_contact(),
|
||||||
e2ee_avail: contact.e2ee_avail(context).await?,
|
e2ee_avail: contact.e2ee_avail(context).await?,
|
||||||
is_verified,
|
is_verified,
|
||||||
is_profile_verified,
|
|
||||||
verifier_id,
|
verifier_id,
|
||||||
last_seen: contact.last_seen(),
|
last_seen: contact.last_seen(),
|
||||||
was_seen_recently: contact.was_seen_recently(),
|
was_seen_recently: contact.was_seen_recently(),
|
||||||
@@ -111,7 +130,13 @@ pub struct VcardContact {
|
|||||||
impl From<deltachat_contact_tools::VcardContact> for VcardContact {
|
impl From<deltachat_contact_tools::VcardContact> for VcardContact {
|
||||||
fn from(vc: deltachat_contact_tools::VcardContact) -> Self {
|
fn from(vc: deltachat_contact_tools::VcardContact) -> Self {
|
||||||
let display_name = vc.display_name().to_string();
|
let display_name = vc.display_name().to_string();
|
||||||
let color = color::str_to_color(&vc.addr.to_lowercase());
|
let is_self = false;
|
||||||
|
let fpr = vc.key.as_deref().and_then(|k| {
|
||||||
|
SignedPublicKey::from_base64(k)
|
||||||
|
.ok()
|
||||||
|
.map(|k| k.dc_fingerprint())
|
||||||
|
});
|
||||||
|
let color = deltachat::contact::get_color(is_self, &vc.addr, &fpr);
|
||||||
Self {
|
Self {
|
||||||
addr: vc.addr,
|
addr: vc.addr,
|
||||||
display_name,
|
display_name,
|
||||||
|
|||||||
@@ -2,6 +2,8 @@ use deltachat::{Event as CoreEvent, EventType as CoreEventType};
|
|||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use typescript_type_def::TypeDef;
|
use typescript_type_def::TypeDef;
|
||||||
|
|
||||||
|
use super::chat::JsonrpcChatType;
|
||||||
|
|
||||||
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub struct Event {
|
pub struct Event {
|
||||||
@@ -224,7 +226,6 @@ pub enum EventType {
|
|||||||
},
|
},
|
||||||
|
|
||||||
/// Chat changed. The name or the image of a chat group was changed or members were added or removed.
|
/// Chat changed. The name or the image of a chat group was changed or members were added or removed.
|
||||||
/// Or the verify state of a chat has changed.
|
|
||||||
/// See setChatName(), setChatProfileImage(), addContactToChat()
|
/// See setChatName(), setChatProfileImage(), addContactToChat()
|
||||||
/// and removeContactFromChat().
|
/// and removeContactFromChat().
|
||||||
///
|
///
|
||||||
@@ -294,8 +295,8 @@ pub enum EventType {
|
|||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
ImexFileWritten { path: String },
|
ImexFileWritten { path: String },
|
||||||
|
|
||||||
/// Progress information of a secure-join handshake from the view of the inviter
|
/// Progress event sent when SecureJoin protocol has finished
|
||||||
/// (Alice, the person who shows the QR code).
|
/// from the view of the inviter (Alice, the person who shows the QR code).
|
||||||
///
|
///
|
||||||
/// These events are typically sent after a joiner has scanned the QR code
|
/// These events are typically sent after a joiner has scanned the QR code
|
||||||
/// generated by getChatSecurejoinQrCodeSvg().
|
/// generated by getChatSecurejoinQrCodeSvg().
|
||||||
@@ -304,11 +305,14 @@ pub enum EventType {
|
|||||||
/// ID of the contact that wants to join.
|
/// ID of the contact that wants to join.
|
||||||
contact_id: u32,
|
contact_id: u32,
|
||||||
|
|
||||||
/// Progress as:
|
/// The type of the joined chat.
|
||||||
/// 300=vg-/vc-request received, typically shown as "bob@addr joins".
|
/// This can take the same values
|
||||||
/// 600=vg-/vc-request-with-auth received, vg-member-added/vc-contact-confirm sent, typically shown as "bob@addr verified".
|
/// as `BasicChat.chatType` ([`crate::api::types::chat::BasicChat::chat_type`]).
|
||||||
/// 800=contact added to chat, shown as "bob@addr securely joined GROUP". Only for the verified-group-protocol.
|
chat_type: JsonrpcChatType,
|
||||||
/// 1000=Protocol finished for this contact.
|
/// ID of the chat in case of success.
|
||||||
|
chat_id: u32,
|
||||||
|
|
||||||
|
/// Progress, always 1000.
|
||||||
progress: usize,
|
progress: usize,
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -417,6 +421,45 @@ pub enum EventType {
|
|||||||
/// Number of events skipped.
|
/// Number of events skipped.
|
||||||
n: u64,
|
n: u64,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/// Incoming call.
|
||||||
|
IncomingCall {
|
||||||
|
/// ID of the info message referring to the call.
|
||||||
|
msg_id: u32,
|
||||||
|
/// ID of the chat which the message belongs to.
|
||||||
|
chat_id: u32,
|
||||||
|
/// User-defined info as passed to place_outgoing_call()
|
||||||
|
place_call_info: String,
|
||||||
|
/// True if incoming call is a video call.
|
||||||
|
has_video: bool,
|
||||||
|
},
|
||||||
|
|
||||||
|
/// Incoming call accepted.
|
||||||
|
/// This is esp. interesting to stop ringing on other devices.
|
||||||
|
IncomingCallAccepted {
|
||||||
|
/// ID of the info message referring to the call.
|
||||||
|
msg_id: u32,
|
||||||
|
/// ID of the chat which the message belongs to.
|
||||||
|
chat_id: u32,
|
||||||
|
},
|
||||||
|
|
||||||
|
/// Outgoing call accepted.
|
||||||
|
OutgoingCallAccepted {
|
||||||
|
/// ID of the info message referring to the call.
|
||||||
|
msg_id: u32,
|
||||||
|
/// ID of the chat which the message belongs to.
|
||||||
|
chat_id: u32,
|
||||||
|
/// User-defined info passed to dc_accept_incoming_call(
|
||||||
|
accept_call_info: String,
|
||||||
|
},
|
||||||
|
|
||||||
|
/// Call ended.
|
||||||
|
CallEnded {
|
||||||
|
/// ID of the info message referring to the call.
|
||||||
|
msg_id: u32,
|
||||||
|
/// ID of the chat which the message belongs to.
|
||||||
|
chat_id: u32,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<CoreEventType> for EventType {
|
impl From<CoreEventType> for EventType {
|
||||||
@@ -523,9 +566,13 @@ impl From<CoreEventType> for EventType {
|
|||||||
},
|
},
|
||||||
CoreEventType::SecurejoinInviterProgress {
|
CoreEventType::SecurejoinInviterProgress {
|
||||||
contact_id,
|
contact_id,
|
||||||
|
chat_type,
|
||||||
|
chat_id,
|
||||||
progress,
|
progress,
|
||||||
} => SecurejoinInviterProgress {
|
} => SecurejoinInviterProgress {
|
||||||
contact_id: contact_id.to_u32(),
|
contact_id: contact_id.to_u32(),
|
||||||
|
chat_type: chat_type.into(),
|
||||||
|
chat_id: chat_id.to_u32(),
|
||||||
progress,
|
progress,
|
||||||
},
|
},
|
||||||
CoreEventType::SecurejoinJoinerProgress {
|
CoreEventType::SecurejoinJoinerProgress {
|
||||||
@@ -567,6 +614,34 @@ impl From<CoreEventType> for EventType {
|
|||||||
CoreEventType::EventChannelOverflow { n } => EventChannelOverflow { n },
|
CoreEventType::EventChannelOverflow { n } => EventChannelOverflow { n },
|
||||||
CoreEventType::AccountsChanged => AccountsChanged,
|
CoreEventType::AccountsChanged => AccountsChanged,
|
||||||
CoreEventType::AccountsItemChanged => AccountsItemChanged,
|
CoreEventType::AccountsItemChanged => AccountsItemChanged,
|
||||||
|
CoreEventType::IncomingCall {
|
||||||
|
msg_id,
|
||||||
|
chat_id,
|
||||||
|
place_call_info,
|
||||||
|
has_video,
|
||||||
|
} => IncomingCall {
|
||||||
|
msg_id: msg_id.to_u32(),
|
||||||
|
chat_id: chat_id.to_u32(),
|
||||||
|
place_call_info,
|
||||||
|
has_video,
|
||||||
|
},
|
||||||
|
CoreEventType::IncomingCallAccepted { msg_id, chat_id } => IncomingCallAccepted {
|
||||||
|
msg_id: msg_id.to_u32(),
|
||||||
|
chat_id: chat_id.to_u32(),
|
||||||
|
},
|
||||||
|
CoreEventType::OutgoingCallAccepted {
|
||||||
|
msg_id,
|
||||||
|
chat_id,
|
||||||
|
accept_call_info,
|
||||||
|
} => OutgoingCallAccepted {
|
||||||
|
msg_id: msg_id.to_u32(),
|
||||||
|
chat_id: chat_id.to_u32(),
|
||||||
|
accept_call_info,
|
||||||
|
},
|
||||||
|
CoreEventType::CallEnded { msg_id, chat_id } => CallEnded {
|
||||||
|
msg_id: msg_id.to_u32(),
|
||||||
|
chat_id: chat_id.to_u32(),
|
||||||
|
},
|
||||||
#[allow(unreachable_patterns)]
|
#[allow(unreachable_patterns)]
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
_ => unreachable!("This is just to silence a rust_analyzer false-positive"),
|
_ => unreachable!("This is just to silence a rust_analyzer false-positive"),
|
||||||
|
|||||||
@@ -4,83 +4,77 @@ use serde::Deserialize;
|
|||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use yerpc::TypeDef;
|
use yerpc::TypeDef;
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, TypeDef, schemars::JsonSchema)]
|
|
||||||
#[serde(rename_all = "camelCase")]
|
|
||||||
pub struct EnteredServerLoginParam {
|
|
||||||
/// Server hostname or IP address.
|
|
||||||
pub server: String,
|
|
||||||
|
|
||||||
/// Server port.
|
|
||||||
///
|
|
||||||
/// 0 if not specified.
|
|
||||||
pub port: u16,
|
|
||||||
|
|
||||||
/// Socket security.
|
|
||||||
pub security: Socket,
|
|
||||||
|
|
||||||
/// Username.
|
|
||||||
///
|
|
||||||
/// Empty string if not specified.
|
|
||||||
pub user: String,
|
|
||||||
|
|
||||||
/// Password.
|
|
||||||
pub password: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<dc::EnteredServerLoginParam> for EnteredServerLoginParam {
|
|
||||||
fn from(param: dc::EnteredServerLoginParam) -> Self {
|
|
||||||
Self {
|
|
||||||
server: param.server,
|
|
||||||
port: param.port,
|
|
||||||
security: param.security.into(),
|
|
||||||
user: param.user,
|
|
||||||
password: param.password,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<EnteredServerLoginParam> for dc::EnteredServerLoginParam {
|
|
||||||
fn from(param: EnteredServerLoginParam) -> Self {
|
|
||||||
Self {
|
|
||||||
server: param.server,
|
|
||||||
port: param.port,
|
|
||||||
security: param.security.into(),
|
|
||||||
user: param.user,
|
|
||||||
password: param.password,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Login parameters entered by the user.
|
/// Login parameters entered by the user.
|
||||||
|
///
|
||||||
|
/// Usually it will be enough to only set `addr` and `password`,
|
||||||
|
/// and all the other settings will be autoconfigured.
|
||||||
#[derive(Serialize, Deserialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, Deserialize, TypeDef, schemars::JsonSchema)]
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub struct EnteredLoginParam {
|
pub struct EnteredLoginParam {
|
||||||
/// Email address.
|
/// Email address.
|
||||||
pub addr: String,
|
pub addr: String,
|
||||||
|
|
||||||
/// IMAP settings.
|
/// Password.
|
||||||
pub imap: EnteredServerLoginParam,
|
pub password: String,
|
||||||
|
|
||||||
/// SMTP settings.
|
/// Imap server hostname or IP address.
|
||||||
pub smtp: EnteredServerLoginParam,
|
pub imap_server: Option<String>,
|
||||||
|
|
||||||
|
/// Imap server port.
|
||||||
|
pub imap_port: Option<u16>,
|
||||||
|
|
||||||
|
/// Imap socket security.
|
||||||
|
pub imap_security: Option<Socket>,
|
||||||
|
|
||||||
|
/// Imap username.
|
||||||
|
pub imap_user: Option<String>,
|
||||||
|
|
||||||
|
/// SMTP server hostname or IP address.
|
||||||
|
pub smtp_server: Option<String>,
|
||||||
|
|
||||||
|
/// SMTP server port.
|
||||||
|
pub smtp_port: Option<u16>,
|
||||||
|
|
||||||
|
/// SMTP socket security.
|
||||||
|
pub smtp_security: Option<Socket>,
|
||||||
|
|
||||||
|
/// SMTP username.
|
||||||
|
pub smtp_user: Option<String>,
|
||||||
|
|
||||||
|
/// SMTP Password.
|
||||||
|
///
|
||||||
|
/// Only needs to be specified if different than IMAP password.
|
||||||
|
pub smtp_password: Option<String>,
|
||||||
|
|
||||||
/// TLS options: whether to allow invalid certificates and/or
|
/// TLS options: whether to allow invalid certificates and/or
|
||||||
/// invalid hostnames
|
/// invalid hostnames.
|
||||||
pub certificate_checks: EnteredCertificateChecks,
|
/// Default: Automatic
|
||||||
|
pub certificate_checks: Option<EnteredCertificateChecks>,
|
||||||
|
|
||||||
/// If true, login via OAUTH2 (not recommended anymore)
|
/// If true, login via OAUTH2 (not recommended anymore).
|
||||||
pub oauth2: bool,
|
/// Default: false
|
||||||
|
pub oauth2: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<dc::EnteredLoginParam> for EnteredLoginParam {
|
impl From<dc::EnteredLoginParam> for EnteredLoginParam {
|
||||||
fn from(param: dc::EnteredLoginParam) -> Self {
|
fn from(param: dc::EnteredLoginParam) -> Self {
|
||||||
|
let imap_security: Socket = param.imap.security.into();
|
||||||
|
let smtp_security: Socket = param.smtp.security.into();
|
||||||
|
let certificate_checks: EnteredCertificateChecks = param.certificate_checks.into();
|
||||||
Self {
|
Self {
|
||||||
addr: param.addr,
|
addr: param.addr,
|
||||||
imap: param.imap.into(),
|
password: param.imap.password,
|
||||||
smtp: param.smtp.into(),
|
imap_server: param.imap.server.into_option(),
|
||||||
certificate_checks: param.certificate_checks.into(),
|
imap_port: param.imap.port.into_option(),
|
||||||
oauth2: param.oauth2,
|
imap_security: imap_security.into_option(),
|
||||||
|
imap_user: param.imap.user.into_option(),
|
||||||
|
smtp_server: param.smtp.server.into_option(),
|
||||||
|
smtp_port: param.smtp.port.into_option(),
|
||||||
|
smtp_security: smtp_security.into_option(),
|
||||||
|
smtp_user: param.smtp.user.into_option(),
|
||||||
|
smtp_password: param.smtp.password.into_option(),
|
||||||
|
certificate_checks: certificate_checks.into_option(),
|
||||||
|
oauth2: param.oauth2.into_option(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -91,18 +85,31 @@ impl TryFrom<EnteredLoginParam> for dc::EnteredLoginParam {
|
|||||||
fn try_from(param: EnteredLoginParam) -> Result<Self> {
|
fn try_from(param: EnteredLoginParam) -> Result<Self> {
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
addr: param.addr,
|
addr: param.addr,
|
||||||
imap: param.imap.into(),
|
imap: dc::EnteredServerLoginParam {
|
||||||
smtp: param.smtp.into(),
|
server: param.imap_server.unwrap_or_default(),
|
||||||
certificate_checks: param.certificate_checks.into(),
|
port: param.imap_port.unwrap_or_default(),
|
||||||
oauth2: param.oauth2,
|
security: param.imap_security.unwrap_or_default().into(),
|
||||||
|
user: param.imap_user.unwrap_or_default(),
|
||||||
|
password: param.password,
|
||||||
|
},
|
||||||
|
smtp: dc::EnteredServerLoginParam {
|
||||||
|
server: param.smtp_server.unwrap_or_default(),
|
||||||
|
port: param.smtp_port.unwrap_or_default(),
|
||||||
|
security: param.smtp_security.unwrap_or_default().into(),
|
||||||
|
user: param.smtp_user.unwrap_or_default(),
|
||||||
|
password: param.smtp_password.unwrap_or_default(),
|
||||||
|
},
|
||||||
|
certificate_checks: param.certificate_checks.unwrap_or_default().into(),
|
||||||
|
oauth2: param.oauth2.unwrap_or_default(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, Deserialize, TypeDef, schemars::JsonSchema, Default, PartialEq)]
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub enum Socket {
|
pub enum Socket {
|
||||||
/// Unspecified socket security, select automatically.
|
/// Unspecified socket security, select automatically.
|
||||||
|
#[default]
|
||||||
Automatic,
|
Automatic,
|
||||||
|
|
||||||
/// TLS connection.
|
/// TLS connection.
|
||||||
@@ -137,12 +144,13 @@ impl From<Socket> for dc::Socket {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, Deserialize, TypeDef, schemars::JsonSchema, Default, PartialEq)]
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub enum EnteredCertificateChecks {
|
pub enum EnteredCertificateChecks {
|
||||||
/// `Automatic` means that provider database setting should be taken.
|
/// `Automatic` means that provider database setting should be taken.
|
||||||
/// If there is no provider database setting for certificate checks,
|
/// If there is no provider database setting for certificate checks,
|
||||||
/// check certificates strictly.
|
/// check certificates strictly.
|
||||||
|
#[default]
|
||||||
Automatic,
|
Automatic,
|
||||||
|
|
||||||
/// Ensure that TLS certificate is valid for the server hostname.
|
/// Ensure that TLS certificate is valid for the server hostname.
|
||||||
@@ -177,3 +185,19 @@ impl From<EnteredCertificateChecks> for dc::EnteredCertificateChecks {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
trait IntoOption<T> {
|
||||||
|
fn into_option(self) -> Option<T>;
|
||||||
|
}
|
||||||
|
impl<T> IntoOption<T> for T
|
||||||
|
where
|
||||||
|
T: Default + std::cmp::PartialEq,
|
||||||
|
{
|
||||||
|
fn into_option(self) -> Option<T> {
|
||||||
|
if self == T::default() {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -16,13 +16,14 @@ use num_traits::cast::ToPrimitive;
|
|||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use typescript_type_def::TypeDef;
|
use typescript_type_def::TypeDef;
|
||||||
|
|
||||||
|
use super::chat::JsonrpcChatType;
|
||||||
use super::color_int_to_hex_string;
|
use super::color_int_to_hex_string;
|
||||||
use super::contact::ContactObject;
|
use super::contact::ContactObject;
|
||||||
use super::reactions::JSONRPCReactions;
|
use super::reactions::JsonrpcReactions;
|
||||||
use super::webxdc::WebxdcMessageInfo;
|
|
||||||
|
|
||||||
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
#[serde(rename_all = "camelCase", tag = "kind")]
|
#[serde(rename_all = "camelCase", tag = "kind")]
|
||||||
|
#[expect(clippy::large_enum_variant)]
|
||||||
pub enum MessageLoadResult {
|
pub enum MessageLoadResult {
|
||||||
Message(MessageObject),
|
Message(MessageObject),
|
||||||
LoadingError { error: String },
|
LoadingError { error: String },
|
||||||
@@ -59,6 +60,13 @@ pub struct MessageObject {
|
|||||||
|
|
||||||
// summary - use/create another function if you need it
|
// summary - use/create another function if you need it
|
||||||
subject: String,
|
subject: String,
|
||||||
|
|
||||||
|
/// True if the message was correctly encrypted&signed, false otherwise.
|
||||||
|
/// Historically, UIs showed a small padlock on the message then.
|
||||||
|
///
|
||||||
|
/// Today, the UIs should instead show a small email-icon on the message
|
||||||
|
/// if `show_padlock` is `false`,
|
||||||
|
/// and nothing if it is `true`.
|
||||||
show_padlock: bool,
|
show_padlock: bool,
|
||||||
is_setupmessage: bool,
|
is_setupmessage: bool,
|
||||||
is_info: bool,
|
is_info: bool,
|
||||||
@@ -70,13 +78,13 @@ pub struct MessageObject {
|
|||||||
/// when is_info is true this describes what type of system message it is
|
/// when is_info is true this describes what type of system message it is
|
||||||
system_message_type: SystemMessageType,
|
system_message_type: SystemMessageType,
|
||||||
|
|
||||||
|
/// if is_info is set, this refers to the contact profile that should be opened when the info message is tapped.
|
||||||
|
info_contact_id: Option<u32>,
|
||||||
|
|
||||||
duration: i32,
|
duration: i32,
|
||||||
dimensions_height: i32,
|
dimensions_height: i32,
|
||||||
dimensions_width: i32,
|
dimensions_width: i32,
|
||||||
|
|
||||||
videochat_type: Option<u32>,
|
|
||||||
videochat_url: Option<String>,
|
|
||||||
|
|
||||||
override_sender_name: Option<String>,
|
override_sender_name: Option<String>,
|
||||||
sender: ContactObject,
|
sender: ContactObject,
|
||||||
|
|
||||||
@@ -87,8 +95,6 @@ pub struct MessageObject {
|
|||||||
file_bytes: u64,
|
file_bytes: u64,
|
||||||
file_name: Option<String>,
|
file_name: Option<String>,
|
||||||
|
|
||||||
webxdc_info: Option<WebxdcMessageInfo>,
|
|
||||||
|
|
||||||
webxdc_href: Option<String>,
|
webxdc_href: Option<String>,
|
||||||
|
|
||||||
download_state: DownloadState,
|
download_state: DownloadState,
|
||||||
@@ -97,7 +103,7 @@ pub struct MessageObject {
|
|||||||
|
|
||||||
saved_message_id: Option<u32>,
|
saved_message_id: Option<u32>,
|
||||||
|
|
||||||
reactions: Option<JSONRPCReactions>,
|
reactions: Option<JsonrpcReactions>,
|
||||||
|
|
||||||
vcard_contact: Option<VcardContact>,
|
vcard_contact: Option<VcardContact>,
|
||||||
}
|
}
|
||||||
@@ -139,12 +145,6 @@ impl MessageObject {
|
|||||||
let file_bytes = message.get_filebytes(context).await?.unwrap_or_default();
|
let file_bytes = message.get_filebytes(context).await?.unwrap_or_default();
|
||||||
let override_sender_name = message.get_override_sender_name();
|
let override_sender_name = message.get_override_sender_name();
|
||||||
|
|
||||||
let webxdc_info = if message.get_viewtype() == Viewtype::Webxdc {
|
|
||||||
Some(WebxdcMessageInfo::get_for_message(context, msg_id).await?)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
|
||||||
|
|
||||||
let parent_id = message.parent(context).await?.map(|m| m.get_id().to_u32());
|
let parent_id = message.parent(context).await?.map(|m| m.get_id().to_u32());
|
||||||
|
|
||||||
let download_state = message.download_state().into();
|
let download_state = message.download_state().into();
|
||||||
@@ -228,20 +228,15 @@ impl MessageObject {
|
|||||||
is_forwarded: message.is_forwarded(),
|
is_forwarded: message.is_forwarded(),
|
||||||
is_bot: message.is_bot(),
|
is_bot: message.is_bot(),
|
||||||
system_message_type: message.get_info_type().into(),
|
system_message_type: message.get_info_type().into(),
|
||||||
|
info_contact_id: message
|
||||||
|
.get_info_contact_id(context)
|
||||||
|
.await?
|
||||||
|
.map(|id| id.to_u32()),
|
||||||
|
|
||||||
duration: message.get_duration(),
|
duration: message.get_duration(),
|
||||||
dimensions_height: message.get_height(),
|
dimensions_height: message.get_height(),
|
||||||
dimensions_width: message.get_width(),
|
dimensions_width: message.get_width(),
|
||||||
|
|
||||||
videochat_type: match message.get_videochat_type() {
|
|
||||||
Some(vct) => Some(
|
|
||||||
vct.to_u32()
|
|
||||||
.context("videochat type conversion to number failed")?,
|
|
||||||
),
|
|
||||||
None => None,
|
|
||||||
},
|
|
||||||
videochat_url: message.get_videochat_url(),
|
|
||||||
|
|
||||||
override_sender_name,
|
override_sender_name,
|
||||||
sender,
|
sender,
|
||||||
|
|
||||||
@@ -254,7 +249,6 @@ impl MessageObject {
|
|||||||
file_mime: message.get_filemime(),
|
file_mime: message.get_filemime(),
|
||||||
file_bytes,
|
file_bytes,
|
||||||
file_name: message.get_filename(),
|
file_name: message.get_filename(),
|
||||||
webxdc_info,
|
|
||||||
|
|
||||||
// On a WebxdcInfoMessage this might include a hash holding
|
// On a WebxdcInfoMessage this might include a hash holding
|
||||||
// information about a specific position or state in a webxdc app
|
// information about a specific position or state in a webxdc app
|
||||||
@@ -316,8 +310,8 @@ pub enum MessageViewtype {
|
|||||||
/// Message containing any file, eg. a PDF.
|
/// Message containing any file, eg. a PDF.
|
||||||
File,
|
File,
|
||||||
|
|
||||||
/// Message is an invitation to a videochat.
|
/// Message is a call.
|
||||||
VideochatInvitation,
|
Call,
|
||||||
|
|
||||||
/// Message is an webxdc instance.
|
/// Message is an webxdc instance.
|
||||||
Webxdc,
|
Webxdc,
|
||||||
@@ -340,7 +334,7 @@ impl From<Viewtype> for MessageViewtype {
|
|||||||
Viewtype::Voice => MessageViewtype::Voice,
|
Viewtype::Voice => MessageViewtype::Voice,
|
||||||
Viewtype::Video => MessageViewtype::Video,
|
Viewtype::Video => MessageViewtype::Video,
|
||||||
Viewtype::File => MessageViewtype::File,
|
Viewtype::File => MessageViewtype::File,
|
||||||
Viewtype::VideochatInvitation => MessageViewtype::VideochatInvitation,
|
Viewtype::Call => MessageViewtype::Call,
|
||||||
Viewtype::Webxdc => MessageViewtype::Webxdc,
|
Viewtype::Webxdc => MessageViewtype::Webxdc,
|
||||||
Viewtype::Vcard => MessageViewtype::Vcard,
|
Viewtype::Vcard => MessageViewtype::Vcard,
|
||||||
}
|
}
|
||||||
@@ -359,7 +353,7 @@ impl From<MessageViewtype> for Viewtype {
|
|||||||
MessageViewtype::Voice => Viewtype::Voice,
|
MessageViewtype::Voice => Viewtype::Voice,
|
||||||
MessageViewtype::Video => Viewtype::Video,
|
MessageViewtype::Video => Viewtype::Video,
|
||||||
MessageViewtype::File => Viewtype::File,
|
MessageViewtype::File => Viewtype::File,
|
||||||
MessageViewtype::VideochatInvitation => Viewtype::VideochatInvitation,
|
MessageViewtype::Call => Viewtype::Call,
|
||||||
MessageViewtype::Webxdc => Viewtype::Webxdc,
|
MessageViewtype::Webxdc => Viewtype::Webxdc,
|
||||||
MessageViewtype::Vcard => Viewtype::Vcard,
|
MessageViewtype::Vcard => Viewtype::Vcard,
|
||||||
}
|
}
|
||||||
@@ -411,6 +405,9 @@ pub enum SystemMessageType {
|
|||||||
/// Chat ephemeral message timer is changed.
|
/// Chat ephemeral message timer is changed.
|
||||||
EphemeralTimerChanged,
|
EphemeralTimerChanged,
|
||||||
|
|
||||||
|
// Chat is e2ee
|
||||||
|
ChatE2ee,
|
||||||
|
|
||||||
// Chat protection state changed
|
// Chat protection state changed
|
||||||
ChatProtectionEnabled,
|
ChatProtectionEnabled,
|
||||||
ChatProtectionDisabled,
|
ChatProtectionDisabled,
|
||||||
@@ -429,6 +426,9 @@ pub enum SystemMessageType {
|
|||||||
|
|
||||||
/// This message contains a users iroh node address.
|
/// This message contains a users iroh node address.
|
||||||
IrohNodeAddr,
|
IrohNodeAddr,
|
||||||
|
|
||||||
|
CallAccepted,
|
||||||
|
CallEnded,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<deltachat::mimeparser::SystemMessage> for SystemMessageType {
|
impl From<deltachat::mimeparser::SystemMessage> for SystemMessageType {
|
||||||
@@ -445,6 +445,7 @@ impl From<deltachat::mimeparser::SystemMessage> for SystemMessageType {
|
|||||||
SystemMessage::LocationStreamingEnabled => SystemMessageType::LocationStreamingEnabled,
|
SystemMessage::LocationStreamingEnabled => SystemMessageType::LocationStreamingEnabled,
|
||||||
SystemMessage::LocationOnly => SystemMessageType::LocationOnly,
|
SystemMessage::LocationOnly => SystemMessageType::LocationOnly,
|
||||||
SystemMessage::EphemeralTimerChanged => SystemMessageType::EphemeralTimerChanged,
|
SystemMessage::EphemeralTimerChanged => SystemMessageType::EphemeralTimerChanged,
|
||||||
|
SystemMessage::ChatE2ee => SystemMessageType::ChatE2ee,
|
||||||
SystemMessage::ChatProtectionEnabled => SystemMessageType::ChatProtectionEnabled,
|
SystemMessage::ChatProtectionEnabled => SystemMessageType::ChatProtectionEnabled,
|
||||||
SystemMessage::ChatProtectionDisabled => SystemMessageType::ChatProtectionDisabled,
|
SystemMessage::ChatProtectionDisabled => SystemMessageType::ChatProtectionDisabled,
|
||||||
SystemMessage::MultiDeviceSync => SystemMessageType::MultiDeviceSync,
|
SystemMessage::MultiDeviceSync => SystemMessageType::MultiDeviceSync,
|
||||||
@@ -454,6 +455,8 @@ impl From<deltachat::mimeparser::SystemMessage> for SystemMessageType {
|
|||||||
SystemMessage::IrohNodeAddr => SystemMessageType::IrohNodeAddr,
|
SystemMessage::IrohNodeAddr => SystemMessageType::IrohNodeAddr,
|
||||||
SystemMessage::SecurejoinWait => SystemMessageType::SecurejoinWait,
|
SystemMessage::SecurejoinWait => SystemMessageType::SecurejoinWait,
|
||||||
SystemMessage::SecurejoinWaitTimeout => SystemMessageType::SecurejoinWaitTimeout,
|
SystemMessage::SecurejoinWaitTimeout => SystemMessageType::SecurejoinWaitTimeout,
|
||||||
|
SystemMessage::CallAccepted => SystemMessageType::CallAccepted,
|
||||||
|
SystemMessage::CallEnded => SystemMessageType::CallEnded,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -529,8 +532,7 @@ pub struct MessageSearchResult {
|
|||||||
chat_profile_image: Option<String>,
|
chat_profile_image: Option<String>,
|
||||||
chat_color: String,
|
chat_color: String,
|
||||||
chat_name: String,
|
chat_name: String,
|
||||||
chat_type: u32,
|
chat_type: JsonrpcChatType,
|
||||||
is_chat_protected: bool,
|
|
||||||
is_chat_contact_request: bool,
|
is_chat_contact_request: bool,
|
||||||
is_chat_archived: bool,
|
is_chat_archived: bool,
|
||||||
message: String,
|
message: String,
|
||||||
@@ -568,9 +570,8 @@ impl MessageSearchResult {
|
|||||||
chat_id: chat.id.to_u32(),
|
chat_id: chat.id.to_u32(),
|
||||||
chat_name: chat.get_name().to_owned(),
|
chat_name: chat.get_name().to_owned(),
|
||||||
chat_color,
|
chat_color,
|
||||||
chat_type: chat.get_type().to_u32().context("unknown chat type id")?,
|
chat_type: chat.get_type().into(),
|
||||||
chat_profile_image,
|
chat_profile_image,
|
||||||
is_chat_protected: chat.is_protected(),
|
|
||||||
is_chat_contact_request: chat.is_contact_request(),
|
is_chat_contact_request: chat.is_contact_request(),
|
||||||
is_chat_archived: chat.get_visibility() == ChatVisibility::Archived,
|
is_chat_archived: chat.get_visibility() == ChatVisibility::Archived,
|
||||||
message: message.get_text(),
|
message: message.get_text(),
|
||||||
@@ -581,7 +582,7 @@ impl MessageSearchResult {
|
|||||||
|
|
||||||
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
#[serde(rename_all = "camelCase", rename = "MessageListItem", tag = "kind")]
|
#[serde(rename_all = "camelCase", rename = "MessageListItem", tag = "kind")]
|
||||||
pub enum JSONRPCMessageListItem {
|
pub enum JsonrpcMessageListItem {
|
||||||
Message {
|
Message {
|
||||||
msg_id: u32,
|
msg_id: u32,
|
||||||
},
|
},
|
||||||
@@ -594,13 +595,13 @@ pub enum JSONRPCMessageListItem {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<ChatItem> for JSONRPCMessageListItem {
|
impl From<ChatItem> for JsonrpcMessageListItem {
|
||||||
fn from(item: ChatItem) -> Self {
|
fn from(item: ChatItem) -> Self {
|
||||||
match item {
|
match item {
|
||||||
ChatItem::Message { msg_id } => JSONRPCMessageListItem::Message {
|
ChatItem::Message { msg_id } => JsonrpcMessageListItem::Message {
|
||||||
msg_id: msg_id.to_u32(),
|
msg_id: msg_id.to_u32(),
|
||||||
},
|
},
|
||||||
ChatItem::DayMarker { timestamp } => JSONRPCMessageListItem::DayMarker { timestamp },
|
ChatItem::DayMarker { timestamp } => JsonrpcMessageListItem::DayMarker { timestamp },
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -673,7 +674,6 @@ pub struct MessageReadReceipt {
|
|||||||
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub struct MessageInfo {
|
pub struct MessageInfo {
|
||||||
rawtext: String,
|
|
||||||
ephemeral_timer: EphemeralTimer,
|
ephemeral_timer: EphemeralTimer,
|
||||||
/// When message is ephemeral this contains the timestamp of the message expiry
|
/// When message is ephemeral this contains the timestamp of the message expiry
|
||||||
ephemeral_timestamp: Option<i64>,
|
ephemeral_timestamp: Option<i64>,
|
||||||
@@ -686,7 +686,6 @@ pub struct MessageInfo {
|
|||||||
impl MessageInfo {
|
impl MessageInfo {
|
||||||
pub async fn from_msg_id(context: &Context, msg_id: MsgId) -> Result<Self> {
|
pub async fn from_msg_id(context: &Context, msg_id: MsgId) -> Result<Self> {
|
||||||
let message = Message::load_from_db(context, msg_id).await?;
|
let message = Message::load_from_db(context, msg_id).await?;
|
||||||
let rawtext = msg_id.rawtext(context).await?;
|
|
||||||
let ephemeral_timer = message.get_ephemeral_timer().into();
|
let ephemeral_timer = message.get_ephemeral_timer().into();
|
||||||
let ephemeral_timestamp = match message.get_ephemeral_timer() {
|
let ephemeral_timestamp = match message.get_ephemeral_timer() {
|
||||||
deltachat::ephemeral::Timer::Disabled => None,
|
deltachat::ephemeral::Timer::Disabled => None,
|
||||||
@@ -699,7 +698,6 @@ impl MessageInfo {
|
|||||||
let hop_info = msg_id.hop_info(context).await?;
|
let hop_info = msg_id.hop_info(context).await?;
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
rawtext,
|
|
||||||
ephemeral_timer,
|
ephemeral_timer,
|
||||||
ephemeral_timestamp,
|
ephemeral_timestamp,
|
||||||
error: message.error(),
|
error: message.error(),
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
pub mod account;
|
pub mod account;
|
||||||
|
pub mod calls;
|
||||||
pub mod chat;
|
pub mod chat;
|
||||||
pub mod chat_list;
|
pub mod chat_list;
|
||||||
pub mod contact;
|
pub mod contact;
|
||||||
@@ -7,6 +8,7 @@ pub mod http;
|
|||||||
pub mod location;
|
pub mod location;
|
||||||
pub mod login_param;
|
pub mod login_param;
|
||||||
pub mod message;
|
pub mod message;
|
||||||
|
pub mod notify_state;
|
||||||
pub mod provider_info;
|
pub mod provider_info;
|
||||||
pub mod qr;
|
pub mod qr;
|
||||||
pub mod reactions;
|
pub mod reactions;
|
||||||
|
|||||||
26
deltachat-jsonrpc/src/api/types/notify_state.rs
Normal file
26
deltachat-jsonrpc/src/api/types/notify_state.rs
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
use deltachat::push::NotifyState;
|
||||||
|
use serde::Serialize;
|
||||||
|
use typescript_type_def::TypeDef;
|
||||||
|
|
||||||
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
|
#[serde(rename = "NotifyState")]
|
||||||
|
pub enum JsonrpcNotifyState {
|
||||||
|
/// Not subscribed to push notifications.
|
||||||
|
NotConnected,
|
||||||
|
|
||||||
|
/// Subscribed to heartbeat push notifications.
|
||||||
|
Heartbeat,
|
||||||
|
|
||||||
|
/// Subscribed to push notifications for new messages.
|
||||||
|
Connected,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<NotifyState> for JsonrpcNotifyState {
|
||||||
|
fn from(state: NotifyState) -> Self {
|
||||||
|
match state {
|
||||||
|
NotifyState::NotConnected => Self::NotConnected,
|
||||||
|
NotifyState::Heartbeat => Self::Heartbeat,
|
||||||
|
NotifyState::Connected => Self::Connected,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
use deltachat::qr::Qr;
|
use deltachat::qr::Qr;
|
||||||
|
use serde::Deserialize;
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use typescript_type_def::TypeDef;
|
use typescript_type_def::TypeDef;
|
||||||
|
|
||||||
@@ -34,6 +35,26 @@ pub enum QrObject {
|
|||||||
/// Authentication code.
|
/// Authentication code.
|
||||||
authcode: String,
|
authcode: String,
|
||||||
},
|
},
|
||||||
|
/// Ask the user whether to join the broadcast channel.
|
||||||
|
AskJoinBroadcast {
|
||||||
|
/// The user-visible name of this broadcast channel
|
||||||
|
name: String,
|
||||||
|
/// A string of random characters,
|
||||||
|
/// uniquely identifying this broadcast channel across all databases/clients.
|
||||||
|
/// Called `grpid` for historic reasons:
|
||||||
|
/// The id of multi-user chats is always called `grpid` in the database
|
||||||
|
/// because groups were once the only multi-user chats.
|
||||||
|
grpid: String,
|
||||||
|
/// ID of the contact who owns the broadcast channel and created the QR code.
|
||||||
|
contact_id: u32,
|
||||||
|
/// Fingerprint of the broadcast channel owner's key as scanned from the QR code.
|
||||||
|
fingerprint: String,
|
||||||
|
|
||||||
|
/// Invite number.
|
||||||
|
invitenumber: String,
|
||||||
|
/// Authentication code.
|
||||||
|
authcode: String,
|
||||||
|
},
|
||||||
/// Contact fingerprint is verified.
|
/// Contact fingerprint is verified.
|
||||||
///
|
///
|
||||||
/// Ask the user if they want to start chatting.
|
/// Ask the user if they want to start chatting.
|
||||||
@@ -207,6 +228,25 @@ impl From<Qr> for QrObject {
|
|||||||
authcode,
|
authcode,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Qr::AskJoinBroadcast {
|
||||||
|
name,
|
||||||
|
grpid,
|
||||||
|
contact_id,
|
||||||
|
fingerprint,
|
||||||
|
authcode,
|
||||||
|
invitenumber,
|
||||||
|
} => {
|
||||||
|
let contact_id = contact_id.to_u32();
|
||||||
|
let fingerprint = fingerprint.to_string();
|
||||||
|
QrObject::AskJoinBroadcast {
|
||||||
|
name,
|
||||||
|
grpid,
|
||||||
|
contact_id,
|
||||||
|
fingerprint,
|
||||||
|
authcode,
|
||||||
|
invitenumber,
|
||||||
|
}
|
||||||
|
}
|
||||||
Qr::FprOk { contact_id } => {
|
Qr::FprOk { contact_id } => {
|
||||||
let contact_id = contact_id.to_u32();
|
let contact_id = contact_id.to_u32();
|
||||||
QrObject::FprOk { contact_id }
|
QrObject::FprOk { contact_id }
|
||||||
@@ -225,13 +265,6 @@ impl From<Qr> for QrObject {
|
|||||||
auth_token,
|
auth_token,
|
||||||
},
|
},
|
||||||
Qr::BackupTooNew {} => QrObject::BackupTooNew {},
|
Qr::BackupTooNew {} => QrObject::BackupTooNew {},
|
||||||
Qr::WebrtcInstance {
|
|
||||||
domain,
|
|
||||||
instance_pattern,
|
|
||||||
} => QrObject::WebrtcInstance {
|
|
||||||
domain,
|
|
||||||
instance_pattern,
|
|
||||||
},
|
|
||||||
Qr::Proxy { url, host, port } => QrObject::Proxy { url, host, port },
|
Qr::Proxy { url, host, port } => QrObject::Proxy { url, host, port },
|
||||||
Qr::Addr { contact_id, draft } => {
|
Qr::Addr { contact_id, draft } => {
|
||||||
let contact_id = contact_id.to_u32();
|
let contact_id = contact_id.to_u32();
|
||||||
@@ -311,3 +344,53 @@ impl From<Qr> for QrObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, TypeDef, schemars::JsonSchema)]
|
||||||
|
pub enum SecurejoinSource {
|
||||||
|
/// Because of some problem, it is unknown where the QR code came from.
|
||||||
|
Unknown,
|
||||||
|
/// The user opened a link somewhere outside Delta Chat
|
||||||
|
ExternalLink,
|
||||||
|
/// The user clicked on a link in a message inside Delta Chat
|
||||||
|
InternalLink,
|
||||||
|
/// The user clicked "Paste from Clipboard" in the QR scan activity
|
||||||
|
Clipboard,
|
||||||
|
/// The user clicked "Load QR code as image" in the QR scan activity
|
||||||
|
ImageLoaded,
|
||||||
|
/// The user scanned a QR code
|
||||||
|
Scan,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, TypeDef, schemars::JsonSchema)]
|
||||||
|
pub enum SecurejoinUiPath {
|
||||||
|
/// The UI path is unknown, or the user didn't open the QR code screen at all.
|
||||||
|
Unknown,
|
||||||
|
/// The user directly clicked on the QR icon in the main screen
|
||||||
|
QrIcon,
|
||||||
|
/// The user first clicked on the `+` button in the main screen,
|
||||||
|
/// and then on "New Contact"
|
||||||
|
NewContact,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<SecurejoinSource> for deltachat::SecurejoinSource {
|
||||||
|
fn from(value: SecurejoinSource) -> Self {
|
||||||
|
match value {
|
||||||
|
SecurejoinSource::Unknown => deltachat::SecurejoinSource::Unknown,
|
||||||
|
SecurejoinSource::ExternalLink => deltachat::SecurejoinSource::ExternalLink,
|
||||||
|
SecurejoinSource::InternalLink => deltachat::SecurejoinSource::InternalLink,
|
||||||
|
SecurejoinSource::Clipboard => deltachat::SecurejoinSource::Clipboard,
|
||||||
|
SecurejoinSource::ImageLoaded => deltachat::SecurejoinSource::ImageLoaded,
|
||||||
|
SecurejoinSource::Scan => deltachat::SecurejoinSource::Scan,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<SecurejoinUiPath> for deltachat::SecurejoinUiPath {
|
||||||
|
fn from(value: SecurejoinUiPath) -> Self {
|
||||||
|
match value {
|
||||||
|
SecurejoinUiPath::Unknown => deltachat::SecurejoinUiPath::Unknown,
|
||||||
|
SecurejoinUiPath::QrIcon => deltachat::SecurejoinUiPath::QrIcon,
|
||||||
|
SecurejoinUiPath::NewContact => deltachat::SecurejoinUiPath::NewContact,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ use typescript_type_def::TypeDef;
|
|||||||
/// A single reaction emoji.
|
/// A single reaction emoji.
|
||||||
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
#[serde(rename = "Reaction", rename_all = "camelCase")]
|
#[serde(rename = "Reaction", rename_all = "camelCase")]
|
||||||
pub struct JSONRPCReaction {
|
pub struct JsonrpcReaction {
|
||||||
/// Emoji.
|
/// Emoji.
|
||||||
emoji: String,
|
emoji: String,
|
||||||
|
|
||||||
@@ -22,14 +22,14 @@ pub struct JSONRPCReaction {
|
|||||||
/// Structure representing all reactions to a particular message.
|
/// Structure representing all reactions to a particular message.
|
||||||
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
#[derive(Serialize, TypeDef, schemars::JsonSchema)]
|
||||||
#[serde(rename = "Reactions", rename_all = "camelCase")]
|
#[serde(rename = "Reactions", rename_all = "camelCase")]
|
||||||
pub struct JSONRPCReactions {
|
pub struct JsonrpcReactions {
|
||||||
/// Map from a contact to it's reaction to message.
|
/// Map from a contact to it's reaction to message.
|
||||||
reactions_by_contact: BTreeMap<u32, Vec<String>>,
|
reactions_by_contact: BTreeMap<u32, Vec<String>>,
|
||||||
/// Unique reactions and their count, sorted in descending order.
|
/// Unique reactions and their count, sorted in descending order.
|
||||||
reactions: Vec<JSONRPCReaction>,
|
reactions: Vec<JsonrpcReaction>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<Reactions> for JSONRPCReactions {
|
impl From<Reactions> for JsonrpcReactions {
|
||||||
fn from(reactions: Reactions) -> Self {
|
fn from(reactions: Reactions) -> Self {
|
||||||
let mut reactions_by_contact: BTreeMap<u32, Vec<String>> = BTreeMap::new();
|
let mut reactions_by_contact: BTreeMap<u32, Vec<String>> = BTreeMap::new();
|
||||||
|
|
||||||
@@ -56,7 +56,7 @@ impl From<Reactions> for JSONRPCReactions {
|
|||||||
false
|
false
|
||||||
};
|
};
|
||||||
|
|
||||||
let reaction = JSONRPCReaction {
|
let reaction = JsonrpcReaction {
|
||||||
emoji,
|
emoji,
|
||||||
count,
|
count,
|
||||||
is_from_self,
|
is_from_self,
|
||||||
@@ -64,7 +64,7 @@ impl From<Reactions> for JSONRPCReactions {
|
|||||||
reactions_v.push(reaction)
|
reactions_v.push(reaction)
|
||||||
}
|
}
|
||||||
|
|
||||||
JSONRPCReactions {
|
JsonrpcReactions {
|
||||||
reactions_by_contact,
|
reactions_by_contact,
|
||||||
reactions: reactions_v,
|
reactions: reactions_v,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,24 +2,24 @@
|
|||||||
"author": "Delta Chat Developers (ML) <delta@codespeak.net>",
|
"author": "Delta Chat Developers (ML) <delta@codespeak.net>",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@deltachat/tiny-emitter": "3.0.0",
|
"@deltachat/tiny-emitter": "3.0.0",
|
||||||
"isomorphic-ws": "^4.0.1",
|
"isomorphic-ws": "^5.0.0",
|
||||||
"yerpc": "^0.6.2"
|
"yerpc": "^0.6.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/chai": "^4.2.21",
|
"@types/chai": "^4.3.10",
|
||||||
"@types/chai-as-promised": "^7.1.5",
|
"@types/chai-as-promised": "^7.1.8",
|
||||||
"@types/mocha": "^9.0.0",
|
"@types/mocha": "^10.0.4",
|
||||||
"@types/ws": "^7.2.4",
|
"@types/ws": "^8.5.9",
|
||||||
"c8": "^7.10.0",
|
"c8": "^8.0.1",
|
||||||
"chai": "^4.3.4",
|
"chai": "^4.3.4",
|
||||||
"chai-as-promised": "^7.1.1",
|
"chai-as-promised": "^7.1.1",
|
||||||
"esbuild": "^0.17.9",
|
"esbuild": "^0.25.5",
|
||||||
"http-server": "^14.1.1",
|
"http-server": "^14.1.1",
|
||||||
"mocha": "^9.1.1",
|
"mocha": "^10.2.0",
|
||||||
"npm-run-all": "^4.1.5",
|
"npm-run-all": "^4.1.5",
|
||||||
"prettier": "^2.6.2",
|
"prettier": "^3.5.3",
|
||||||
"typedoc": "^0.23.2",
|
"typedoc": "^0.28.5",
|
||||||
"typescript": "^4.5.5",
|
"typescript": "^5.8.3",
|
||||||
"ws": "^8.5.0"
|
"ws": "^8.5.0"
|
||||||
},
|
},
|
||||||
"exports": {
|
"exports": {
|
||||||
@@ -54,5 +54,5 @@
|
|||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"types": "dist/deltachat.d.ts",
|
"types": "dist/deltachat.d.ts",
|
||||||
"version": "1.157.3"
|
"version": "2.25.0"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,24 +5,24 @@ const json = JSON.parse(readFileSync("./coverage/coverage-final.json"));
|
|||||||
const jsonCoverage =
|
const jsonCoverage =
|
||||||
json[Object.keys(json).find((k) => k.includes(generatedFile))];
|
json[Object.keys(json).find((k) => k.includes(generatedFile))];
|
||||||
const fnMap = Object.keys(jsonCoverage.fnMap).map(
|
const fnMap = Object.keys(jsonCoverage.fnMap).map(
|
||||||
(key) => jsonCoverage.fnMap[key]
|
(key) => jsonCoverage.fnMap[key],
|
||||||
);
|
);
|
||||||
const htmlCoverage = readFileSync(
|
const htmlCoverage = readFileSync(
|
||||||
"./coverage/" + generatedFile + ".html",
|
"./coverage/" + generatedFile + ".html",
|
||||||
"utf8"
|
"utf8",
|
||||||
);
|
);
|
||||||
const uncoveredLines = htmlCoverage
|
const uncoveredLines = htmlCoverage
|
||||||
.split("\n")
|
.split("\n")
|
||||||
.filter((line) => line.includes(`"function not covered"`));
|
.filter((line) => line.includes(`"function not covered"`));
|
||||||
const uncoveredFunctions = uncoveredLines.map(
|
const uncoveredFunctions = uncoveredLines.map(
|
||||||
(line) => />([\w_]+)\(/.exec(line)[1]
|
(line) => />([\w_]+)\(/.exec(line)[1],
|
||||||
);
|
);
|
||||||
console.log(
|
console.log(
|
||||||
"\nUncovered api functions:\n" +
|
"\nUncovered api functions:\n" +
|
||||||
uncoveredFunctions
|
uncoveredFunctions
|
||||||
.map((uF) => fnMap.find(({ name }) => name === uF))
|
.map((uF) => fnMap.find(({ name }) => name === uF))
|
||||||
.map(
|
.map(
|
||||||
({ name, line }) => `.${name.padEnd(40)} (${generatedFile}:${line})`
|
({ name, line }) => `.${name.padEnd(40)} (${generatedFile}:${line})`,
|
||||||
)
|
)
|
||||||
.join("\n")
|
.join("\n"),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ while (null != (match = regex.exec(header_data))) {
|
|||||||
|
|
||||||
const constants = data
|
const constants = data
|
||||||
.filter(
|
.filter(
|
||||||
({ key }) => key.toUpperCase()[0] === key[0] // check if define name is uppercase
|
({ key }) => key.toUpperCase()[0] === key[0], // check if define name is uppercase
|
||||||
)
|
)
|
||||||
.sort((lhs, rhs) => {
|
.sort((lhs, rhs) => {
|
||||||
if (lhs.key < rhs.key) return -1;
|
if (lhs.key < rhs.key) return -1;
|
||||||
@@ -40,15 +40,35 @@ const constants = data
|
|||||||
key.startsWith("DC_DOWNLOAD") ||
|
key.startsWith("DC_DOWNLOAD") ||
|
||||||
key.startsWith("DC_INFO_") ||
|
key.startsWith("DC_INFO_") ||
|
||||||
(key.startsWith("DC_MSG") && !key.startsWith("DC_MSG_ID")) ||
|
(key.startsWith("DC_MSG") && !key.startsWith("DC_MSG_ID")) ||
|
||||||
key.startsWith("DC_QR_")
|
key.startsWith("DC_QR_") ||
|
||||||
|
key.startsWith("DC_CERTCK_") ||
|
||||||
|
key.startsWith("DC_SOCKET_") ||
|
||||||
|
key.startsWith("DC_LP_AUTH_") ||
|
||||||
|
key.startsWith("DC_PUSH_") ||
|
||||||
|
key.startsWith("DC_TEXT1_") ||
|
||||||
|
key.startsWith("DC_CHAT_TYPE")
|
||||||
);
|
);
|
||||||
})
|
})
|
||||||
.map((row) => {
|
.map((row) => {
|
||||||
return ` ${row.key}: ${row.value}`;
|
return ` export const ${row.key} = ${row.value};`;
|
||||||
})
|
})
|
||||||
.join(",\n");
|
.join("\n");
|
||||||
|
|
||||||
writeFileSync(
|
writeFileSync(
|
||||||
resolve(__dirname, "../generated/constants.ts"),
|
resolve(__dirname, "../generated/constants.ts"),
|
||||||
`// Generated!\n\nexport enum C {\n${constants.replace(/:/g, " =")},\n}\n`
|
`// Generated!
|
||||||
|
|
||||||
|
export namespace C {
|
||||||
|
${constants}
|
||||||
|
/** @deprecated 10-8-2025 compare string directly with \`== "Group"\` */
|
||||||
|
export const DC_CHAT_TYPE_GROUP = "Group";
|
||||||
|
/** @deprecated 10-8-2025 compare string directly with \`== "InBroadcast"\`*/
|
||||||
|
export const DC_CHAT_TYPE_IN_BROADCAST = "InBroadcast";
|
||||||
|
/** @deprecated 10-8-2025 compare string directly with \`== "Mailinglist"\` */
|
||||||
|
export const DC_CHAT_TYPE_MAILINGLIST = "Mailinglist";
|
||||||
|
/** @deprecated 10-8-2025 compare string directly with \`== "OutBroadcast"\` */
|
||||||
|
export const DC_CHAT_TYPE_OUT_BROADCAST = "OutBroadcast";
|
||||||
|
/** @deprecated 10-8-2025 compare string directly with \`== "Single"\` */
|
||||||
|
export const DC_CHAT_TYPE_SINGLE = "Single";
|
||||||
|
}\n`,
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -8,13 +8,13 @@ import { TinyEmitter } from "@deltachat/tiny-emitter";
|
|||||||
type Events = { ALL: (accountId: number, event: EventType) => void } & {
|
type Events = { ALL: (accountId: number, event: EventType) => void } & {
|
||||||
[Property in EventType["kind"]]: (
|
[Property in EventType["kind"]]: (
|
||||||
accountId: number,
|
accountId: number,
|
||||||
event: Extract<EventType, { kind: Property }>
|
event: Extract<EventType, { kind: Property }>,
|
||||||
) => void;
|
) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
type ContextEvents = { ALL: (event: EventType) => void } & {
|
type ContextEvents = { ALL: (event: EventType) => void } & {
|
||||||
[Property in EventType["kind"]]: (
|
[Property in EventType["kind"]]: (
|
||||||
event: Extract<EventType, { kind: Property }>
|
event: Extract<EventType, { kind: Property }>,
|
||||||
) => void;
|
) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -25,16 +25,22 @@ export type DcEventType<T extends EventType["kind"]> = Extract<
|
|||||||
>;
|
>;
|
||||||
|
|
||||||
export class BaseDeltaChat<
|
export class BaseDeltaChat<
|
||||||
Transport extends BaseTransport<any>
|
Transport extends BaseTransport<any>,
|
||||||
> extends TinyEmitter<Events> {
|
> extends TinyEmitter<Events> {
|
||||||
rpc: RawClient;
|
rpc: RawClient;
|
||||||
account?: T.Account;
|
|
||||||
private contextEmitters: { [key: number]: TinyEmitter<ContextEvents> } = {};
|
private contextEmitters: { [key: number]: TinyEmitter<ContextEvents> } = {};
|
||||||
|
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
private eventTask: Promise<void>;
|
private eventTask: Promise<void>;
|
||||||
|
|
||||||
constructor(public transport: Transport, startEventLoop: boolean) {
|
constructor(
|
||||||
|
public transport: Transport,
|
||||||
|
/**
|
||||||
|
* Whether to start calling {@linkcode RawClient.getNextEvent}
|
||||||
|
* and emitting the respective events on this class.
|
||||||
|
*/
|
||||||
|
startEventLoop: boolean,
|
||||||
|
) {
|
||||||
super();
|
super();
|
||||||
this.rpc = new RawClient(this.transport);
|
this.rpc = new RawClient(this.transport);
|
||||||
if (startEventLoop) {
|
if (startEventLoop) {
|
||||||
@@ -42,6 +48,9 @@ export class BaseDeltaChat<
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see the constructor's `startEventLoop`
|
||||||
|
*/
|
||||||
async eventLoop(): Promise<void> {
|
async eventLoop(): Promise<void> {
|
||||||
while (true) {
|
while (true) {
|
||||||
const event = await this.rpc.getNextEvent();
|
const event = await this.rpc.getNextEvent();
|
||||||
@@ -53,17 +62,24 @@ export class BaseDeltaChat<
|
|||||||
this.contextEmitters[event.contextId].emit(
|
this.contextEmitters[event.contextId].emit(
|
||||||
event.event.kind,
|
event.event.kind,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
event.event as any
|
event.event as any,
|
||||||
);
|
);
|
||||||
this.contextEmitters[event.contextId].emit("ALL", event.event as any);
|
this.contextEmitters[event.contextId].emit("ALL", event.event as any);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated use {@linkcode BaseDeltaChat.rpc.getAllAccounts} instead.
|
||||||
|
*/
|
||||||
async listAccounts(): Promise<T.Account[]> {
|
async listAccounts(): Promise<T.Account[]> {
|
||||||
return await this.rpc.getAllAccounts();
|
return await this.rpc.getAllAccounts();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A convenience function to listen on events binned by `account_id`
|
||||||
|
* (see {@linkcode RawClient.getAllAccounts}).
|
||||||
|
*/
|
||||||
getContextEvents(account_id: number) {
|
getContextEvents(account_id: number) {
|
||||||
if (this.contextEmitters[account_id]) {
|
if (this.contextEmitters[account_id]) {
|
||||||
return this.contextEmitters[account_id];
|
return this.contextEmitters[account_id];
|
||||||
@@ -83,7 +99,10 @@ export class StdioDeltaChat extends BaseDeltaChat<StdioTransport> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export class StdioTransport extends BaseTransport {
|
export class StdioTransport extends BaseTransport {
|
||||||
constructor(public input: any, public output: any) {
|
constructor(
|
||||||
|
public input: any,
|
||||||
|
public output: any,
|
||||||
|
) {
|
||||||
super();
|
super();
|
||||||
|
|
||||||
var buffer = "";
|
var buffer = "";
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import { strictEqual } from "assert";
|
|
||||||
import chai, { assert, expect } from "chai";
|
import chai, { assert, expect } from "chai";
|
||||||
import chaiAsPromised from "chai-as-promised";
|
import chaiAsPromised from "chai-as-promised";
|
||||||
chai.use(chaiAsPromised);
|
chai.use(chaiAsPromised);
|
||||||
@@ -32,14 +31,14 @@ describe("basic tests", () => {
|
|||||||
|
|
||||||
expect(
|
expect(
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
validAddresses.map((email) => dc.rpc.checkEmailValidity(email))
|
validAddresses.map((email) => dc.rpc.checkEmailValidity(email)),
|
||||||
)
|
),
|
||||||
).to.not.contain(false);
|
).to.not.contain(false);
|
||||||
|
|
||||||
expect(
|
expect(
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
invalidAddresses.map((email) => dc.rpc.checkEmailValidity(email))
|
invalidAddresses.map((email) => dc.rpc.checkEmailValidity(email)),
|
||||||
)
|
),
|
||||||
).to.not.contain(true);
|
).to.not.contain(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -85,7 +84,7 @@ describe("basic tests", () => {
|
|||||||
const contactId = await dc.rpc.createContact(
|
const contactId = await dc.rpc.createContact(
|
||||||
accountId,
|
accountId,
|
||||||
"example@delta.chat",
|
"example@delta.chat",
|
||||||
null
|
null,
|
||||||
);
|
);
|
||||||
expect((await dc.rpc.getContact(accountId, contactId)).isBlocked).to.be
|
expect((await dc.rpc.getContact(accountId, contactId)).isBlocked).to.be
|
||||||
.false;
|
.false;
|
||||||
@@ -127,7 +126,7 @@ describe("basic tests", () => {
|
|||||||
await dc.rpc.batchSetConfig(accountId, config);
|
await dc.rpc.batchSetConfig(accountId, config);
|
||||||
const retrieved = await dc.rpc.batchGetConfig(
|
const retrieved = await dc.rpc.batchGetConfig(
|
||||||
accountId,
|
accountId,
|
||||||
Object.keys(config)
|
Object.keys(config),
|
||||||
);
|
);
|
||||||
expect(retrieved).to.deep.equal(config);
|
expect(retrieved).to.deep.equal(config);
|
||||||
});
|
});
|
||||||
@@ -139,7 +138,7 @@ describe("basic tests", () => {
|
|||||||
await dc.rpc.batchSetConfig(accountId, config);
|
await dc.rpc.batchSetConfig(accountId, config);
|
||||||
const retrieved = await dc.rpc.batchGetConfig(
|
const retrieved = await dc.rpc.batchGetConfig(
|
||||||
accountId,
|
accountId,
|
||||||
Object.keys(config)
|
Object.keys(config),
|
||||||
);
|
);
|
||||||
expect(retrieved).to.deep.equal(config);
|
expect(retrieved).to.deep.equal(config);
|
||||||
});
|
});
|
||||||
@@ -153,7 +152,7 @@ describe("basic tests", () => {
|
|||||||
await dc.rpc.batchSetConfig(accountId, config);
|
await dc.rpc.batchSetConfig(accountId, config);
|
||||||
const retrieved = await dc.rpc.batchGetConfig(
|
const retrieved = await dc.rpc.batchGetConfig(
|
||||||
accountId,
|
accountId,
|
||||||
Object.keys(config)
|
Object.keys(config),
|
||||||
);
|
);
|
||||||
expect(retrieved).to.deep.equal(config);
|
expect(retrieved).to.deep.equal(config);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { assert, expect } from "chai";
|
import { assert, expect } from "chai";
|
||||||
import { StdioDeltaChat as DeltaChat, DcEvent } from "../deltachat.js";
|
import { StdioDeltaChat as DeltaChat, DcEvent, C } from "../deltachat.js";
|
||||||
import { RpcServerHandle, createTempUser, startServer } from "./test_base.js";
|
import { RpcServerHandle, createTempUser, startServer } from "./test_base.js";
|
||||||
|
|
||||||
const EVENT_TIMEOUT = 20000;
|
const EVENT_TIMEOUT = 20000;
|
||||||
@@ -17,12 +17,12 @@ describe("online tests", function () {
|
|||||||
if (process.env.COVERAGE && !process.env.COVERAGE_OFFLINE) {
|
if (process.env.COVERAGE && !process.env.COVERAGE_OFFLINE) {
|
||||||
console.error(
|
console.error(
|
||||||
"CAN NOT RUN COVERAGE correctly: Missing CHATMAIL_DOMAIN environment variable!\n\n",
|
"CAN NOT RUN COVERAGE correctly: Missing CHATMAIL_DOMAIN environment variable!\n\n",
|
||||||
"You can set COVERAGE_OFFLINE=1 to circumvent this check and skip the online tests, but those coverage results will be wrong, because some functions can only be tested in the online test"
|
"You can set COVERAGE_OFFLINE=1 to circumvent this check and skip the online tests, but those coverage results will be wrong, because some functions can only be tested in the online test",
|
||||||
);
|
);
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
console.log(
|
console.log(
|
||||||
"Missing CHATMAIL_DOMAIN environment variable!, skip integration tests"
|
"Missing CHATMAIL_DOMAIN environment variable!, skip integration tests",
|
||||||
);
|
);
|
||||||
this.skip();
|
this.skip();
|
||||||
}
|
}
|
||||||
@@ -36,7 +36,7 @@ describe("online tests", function () {
|
|||||||
account1 = createTempUser(process.env.CHATMAIL_DOMAIN);
|
account1 = createTempUser(process.env.CHATMAIL_DOMAIN);
|
||||||
if (!account1 || !account1.email || !account1.password) {
|
if (!account1 || !account1.email || !account1.password) {
|
||||||
console.log(
|
console.log(
|
||||||
"We didn't got back an account from the api, skip integration tests"
|
"We didn't got back an account from the api, skip integration tests",
|
||||||
);
|
);
|
||||||
this.skip();
|
this.skip();
|
||||||
}
|
}
|
||||||
@@ -44,7 +44,7 @@ describe("online tests", function () {
|
|||||||
account2 = createTempUser(process.env.CHATMAIL_DOMAIN);
|
account2 = createTempUser(process.env.CHATMAIL_DOMAIN);
|
||||||
if (!account2 || !account2.email || !account2.password) {
|
if (!account2 || !account2.email || !account2.password) {
|
||||||
console.log(
|
console.log(
|
||||||
"We didn't got back an account2 from the api, skip integration tests"
|
"We didn't got back an account2 from the api, skip integration tests",
|
||||||
);
|
);
|
||||||
this.skip();
|
this.skip();
|
||||||
}
|
}
|
||||||
@@ -80,11 +80,8 @@ describe("online tests", function () {
|
|||||||
}
|
}
|
||||||
this.timeout(15000);
|
this.timeout(15000);
|
||||||
|
|
||||||
const contactId = await dc.rpc.createContact(
|
const vcard = await dc.rpc.makeVcard(accountId2, [C.DC_CONTACT_ID_SELF]);
|
||||||
accountId1,
|
const contactId = (await dc.rpc.importVcardContents(accountId1, vcard))[0];
|
||||||
account2.email,
|
|
||||||
null
|
|
||||||
);
|
|
||||||
const chatId = await dc.rpc.createChatByContactId(accountId1, contactId);
|
const chatId = await dc.rpc.createChatByContactId(accountId1, contactId);
|
||||||
const eventPromise = waitForEvent(dc, "IncomingMsg", accountId2);
|
const eventPromise = waitForEvent(dc, "IncomingMsg", accountId2);
|
||||||
|
|
||||||
@@ -95,26 +92,26 @@ describe("online tests", function () {
|
|||||||
accountId2,
|
accountId2,
|
||||||
chatIdOnAccountB,
|
chatIdOnAccountB,
|
||||||
false,
|
false,
|
||||||
false
|
false,
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(messageList).have.length(1);
|
// There are 2 messages in the chat:
|
||||||
const message = await dc.rpc.getMessage(accountId2, messageList[0]);
|
// 'Messages are end-to-end encrypted' (info message) and 'Hello'
|
||||||
|
expect(messageList).have.length(2);
|
||||||
|
const message = await dc.rpc.getMessage(accountId2, messageList[1]);
|
||||||
expect(message.text).equal("Hello");
|
expect(message.text).equal("Hello");
|
||||||
|
expect(message.showPadlock).equal(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("send and receive text message roundtrip, encrypted on answer onwards", async function () {
|
it("send and receive text message roundtrip", async function () {
|
||||||
if (!accountsConfigured) {
|
if (!accountsConfigured) {
|
||||||
this.skip();
|
this.skip();
|
||||||
}
|
}
|
||||||
this.timeout(10000);
|
this.timeout(10000);
|
||||||
|
|
||||||
// send message from A to B
|
// send message from A to B
|
||||||
const contactId = await dc.rpc.createContact(
|
const vcard = await dc.rpc.makeVcard(accountId2, [C.DC_CONTACT_ID_SELF]);
|
||||||
accountId1,
|
const contactId = (await dc.rpc.importVcardContents(accountId1, vcard))[0];
|
||||||
account2.email,
|
|
||||||
null
|
|
||||||
);
|
|
||||||
const chatId = await dc.rpc.createChatByContactId(accountId1, contactId);
|
const chatId = await dc.rpc.createChatByContactId(accountId1, contactId);
|
||||||
const eventPromise = waitForEvent(dc, "IncomingMsg", accountId2);
|
const eventPromise = waitForEvent(dc, "IncomingMsg", accountId2);
|
||||||
dc.rpc.miscSendTextMessage(accountId1, chatId, "Hello2");
|
dc.rpc.miscSendTextMessage(accountId1, chatId, "Hello2");
|
||||||
@@ -129,11 +126,11 @@ describe("online tests", function () {
|
|||||||
accountId2,
|
accountId2,
|
||||||
chatIdOnAccountB,
|
chatIdOnAccountB,
|
||||||
false,
|
false,
|
||||||
false
|
false,
|
||||||
);
|
);
|
||||||
const message = await dc.rpc.getMessage(
|
const message = await dc.rpc.getMessage(
|
||||||
accountId2,
|
accountId2,
|
||||||
messageList.reverse()[0]
|
messageList.reverse()[0],
|
||||||
);
|
);
|
||||||
expect(message.text).equal("Hello2");
|
expect(message.text).equal("Hello2");
|
||||||
// Send message back from B to A
|
// Send message back from B to A
|
||||||
@@ -155,7 +152,7 @@ describe("online tests", function () {
|
|||||||
const info = await dc.rpc.getProviderInfo(acc, "example.com");
|
const info = await dc.rpc.getProviderInfo(acc, "example.com");
|
||||||
expect(info).to.be.not.null;
|
expect(info).to.be.not.null;
|
||||||
expect(info?.overviewPage).to.equal(
|
expect(info?.overviewPage).to.equal(
|
||||||
"https://providers.delta.chat/example-com"
|
"https://providers.delta.chat/example-com",
|
||||||
);
|
);
|
||||||
expect(info?.status).to.equal(3);
|
expect(info?.status).to.equal(3);
|
||||||
});
|
});
|
||||||
@@ -172,12 +169,12 @@ async function waitForEvent<T extends DcEvent["kind"]>(
|
|||||||
dc: DeltaChat,
|
dc: DeltaChat,
|
||||||
eventType: T,
|
eventType: T,
|
||||||
accountId: number,
|
accountId: number,
|
||||||
timeout: number = EVENT_TIMEOUT
|
timeout: number = EVENT_TIMEOUT,
|
||||||
): Promise<Extract<DcEvent, { kind: T }>> {
|
): Promise<Extract<DcEvent, { kind: T }>> {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
const rejectTimeout = setTimeout(
|
const rejectTimeout = setTimeout(
|
||||||
() => reject(new Error("Timeout reached before event came in")),
|
() => reject(new Error("Timeout reached before event came in")),
|
||||||
timeout
|
timeout,
|
||||||
);
|
);
|
||||||
const callback = (contextId: number, event: DcEvent) => {
|
const callback = (contextId: number, event: DcEvent) => {
|
||||||
if (contextId == accountId) {
|
if (contextId == accountId) {
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ export async function startServer(): Promise<RpcServerHandle> {
|
|||||||
const tmpDir = await mkdtemp(join(tmpdir(), "deltachat-jsonrpc-test"));
|
const tmpDir = await mkdtemp(join(tmpdir(), "deltachat-jsonrpc-test"));
|
||||||
|
|
||||||
const pathToServerBinary = resolve(
|
const pathToServerBinary = resolve(
|
||||||
join(await getTargetDir(), "debug/deltachat-rpc-server")
|
join(await getTargetDir(), "debug/deltachat-rpc-server"),
|
||||||
);
|
);
|
||||||
|
|
||||||
const server = spawn(pathToServerBinary, {
|
const server = spawn(pathToServerBinary, {
|
||||||
@@ -29,7 +29,7 @@ export async function startServer(): Promise<RpcServerHandle> {
|
|||||||
throw new Error(
|
throw new Error(
|
||||||
"Failed to start server executable " +
|
"Failed to start server executable " +
|
||||||
pathToServerBinary +
|
pathToServerBinary +
|
||||||
", make sure you built it first."
|
", make sure you built it first.",
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
let shouldClose = false;
|
let shouldClose = false;
|
||||||
@@ -83,7 +83,7 @@ function getTargetDir(): Promise<string> {
|
|||||||
reject(error);
|
reject(error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "deltachat-repl"
|
name = "deltachat-repl"
|
||||||
version = "1.157.3"
|
version = "2.25.0"
|
||||||
license = "MPL-2.0"
|
license = "MPL-2.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
repository = "https://github.com/chatmail/core"
|
repository = "https://github.com/chatmail/core"
|
||||||
@@ -13,7 +13,7 @@ log = { workspace = true }
|
|||||||
nu-ansi-term = { workspace = true }
|
nu-ansi-term = { workspace = true }
|
||||||
qr2term = "0.3.3"
|
qr2term = "0.3.3"
|
||||||
rusqlite = { workspace = true }
|
rusqlite = { workspace = true }
|
||||||
rustyline = "15"
|
rustyline = "16"
|
||||||
tokio = { workspace = true, features = ["fs", "rt-multi-thread", "macros"] }
|
tokio = { workspace = true, features = ["fs", "rt-multi-thread", "macros"] }
|
||||||
tracing-subscriber = { workspace = true, features = ["env-filter"] }
|
tracing-subscriber = { workspace = true, features = ["env-filter"] }
|
||||||
|
|
||||||
|
|||||||
@@ -6,9 +6,7 @@ use std::str::FromStr;
|
|||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
use anyhow::{bail, ensure, Result};
|
use anyhow::{bail, ensure, Result};
|
||||||
use deltachat::chat::{
|
use deltachat::chat::{self, Chat, ChatId, ChatItem, ChatVisibility, MuteDuration};
|
||||||
self, Chat, ChatId, ChatItem, ChatVisibility, MuteDuration, ProtectionStatus,
|
|
||||||
};
|
|
||||||
use deltachat::chatlist::*;
|
use deltachat::chatlist::*;
|
||||||
use deltachat::constants::*;
|
use deltachat::constants::*;
|
||||||
use deltachat::contact::*;
|
use deltachat::contact::*;
|
||||||
@@ -20,7 +18,6 @@ use deltachat::log::LogExt;
|
|||||||
use deltachat::message::{self, Message, MessageState, MsgId, Viewtype};
|
use deltachat::message::{self, Message, MessageState, MsgId, Viewtype};
|
||||||
use deltachat::mimeparser::SystemMessage;
|
use deltachat::mimeparser::SystemMessage;
|
||||||
use deltachat::peer_channels::{send_webxdc_realtime_advertisement, send_webxdc_realtime_data};
|
use deltachat::peer_channels::{send_webxdc_realtime_advertisement, send_webxdc_realtime_data};
|
||||||
use deltachat::peerstate::*;
|
|
||||||
use deltachat::qr::*;
|
use deltachat::qr::*;
|
||||||
use deltachat::qr_code_generator::create_qr_svg;
|
use deltachat::qr_code_generator::create_qr_svg;
|
||||||
use deltachat::reaction::send_reaction;
|
use deltachat::reaction::send_reaction;
|
||||||
@@ -35,14 +32,6 @@ use tokio::fs;
|
|||||||
/// e.g. bitmask 7 triggers actions defined with bits 1, 2 and 4.
|
/// e.g. bitmask 7 triggers actions defined with bits 1, 2 and 4.
|
||||||
async fn reset_tables(context: &Context, bits: i32) {
|
async fn reset_tables(context: &Context, bits: i32) {
|
||||||
println!("Resetting tables ({bits})...");
|
println!("Resetting tables ({bits})...");
|
||||||
if 0 != bits & 2 {
|
|
||||||
context
|
|
||||||
.sql()
|
|
||||||
.execute("DELETE FROM acpeerstates;", ())
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
println!("(2) Peerstates reset.");
|
|
||||||
}
|
|
||||||
if 0 != bits & 4 {
|
if 0 != bits & 4 {
|
||||||
context
|
context
|
||||||
.sql()
|
.sql()
|
||||||
@@ -81,11 +70,6 @@ async fn reset_tables(context: &Context, bits: i32) {
|
|||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
context.sql().config_cache().write().await.clear();
|
context.sql().config_cache().write().await.clear();
|
||||||
context
|
|
||||||
.sql()
|
|
||||||
.execute("DELETE FROM leftgrps;", ())
|
|
||||||
.await
|
|
||||||
.unwrap();
|
|
||||||
println!("(8) Rest but server config reset.");
|
println!("(8) Rest but server config reset.");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -96,7 +80,7 @@ async fn poke_eml_file(context: &Context, filename: &Path) -> Result<()> {
|
|||||||
let data = read_file(context, filename).await?;
|
let data = read_file(context, filename).await?;
|
||||||
|
|
||||||
if let Err(err) = receive_imf(context, &data, false).await {
|
if let Err(err) = receive_imf(context, &data, false).await {
|
||||||
println!("receive_imf errored: {err:?}");
|
eprintln!("receive_imf errored: {err:?}");
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@@ -120,7 +104,7 @@ async fn poke_spec(context: &Context, spec: Option<&str>) -> bool {
|
|||||||
} else {
|
} else {
|
||||||
let rs = context.sql().get_raw_config("import_spec").await.unwrap();
|
let rs = context.sql().get_raw_config("import_spec").await.unwrap();
|
||||||
if rs.is_none() {
|
if rs.is_none() {
|
||||||
error!(context, "Import: No file or folder given.");
|
eprintln!("Import: No file or folder given.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
real_spec = rs.unwrap();
|
real_spec = rs.unwrap();
|
||||||
@@ -149,7 +133,7 @@ async fn poke_spec(context: &Context, spec: Option<&str>) -> bool {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
error!(context, "Import: Cannot open directory \"{}\".", &real_spec);
|
eprintln!("Import: Cannot open directory \"{}\".", &real_spec);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -219,13 +203,7 @@ async fn log_msg(context: &Context, prefix: impl AsRef<str>, msg: &Message) {
|
|||||||
} else {
|
} else {
|
||||||
""
|
""
|
||||||
},
|
},
|
||||||
if msg.get_viewtype() == Viewtype::VideochatInvitation {
|
if msg.get_viewtype() == Viewtype::Webxdc {
|
||||||
format!(
|
|
||||||
"[VIDEOCHAT-INVITATION: {}, type={}]",
|
|
||||||
msg.get_videochat_url().unwrap_or_default(),
|
|
||||||
msg.get_videochat_type().unwrap_or_default()
|
|
||||||
)
|
|
||||||
} else if msg.get_viewtype() == Viewtype::Webxdc {
|
|
||||||
match msg.get_webxdc_info(context).await {
|
match msg.get_webxdc_info(context).await {
|
||||||
Ok(info) => format!(
|
Ok(info) => format!(
|
||||||
"[WEBXDC: {}, icon={}, document={}, summary={}, source_code_url={}]",
|
"[WEBXDC: {}, icon={}, document={}, summary={}, source_code_url={}]",
|
||||||
@@ -277,7 +255,7 @@ async fn log_msglist(context: &Context, msglist: &[MsgId]) -> Result<()> {
|
|||||||
|
|
||||||
async fn log_contactlist(context: &Context, contacts: &[ContactId]) -> Result<()> {
|
async fn log_contactlist(context: &Context, contacts: &[ContactId]) -> Result<()> {
|
||||||
for contact_id in contacts {
|
for contact_id in contacts {
|
||||||
let mut line2 = "".to_string();
|
let line2 = "".to_string();
|
||||||
let contact = Contact::get_by_id(context, *contact_id).await?;
|
let contact = Contact::get_by_id(context, *contact_id).await?;
|
||||||
let name = contact.get_display_name();
|
let name = contact.get_display_name();
|
||||||
let addr = contact.get_addr();
|
let addr = contact.get_addr();
|
||||||
@@ -296,15 +274,6 @@ async fn log_contactlist(context: &Context, contacts: &[ContactId]) -> Result<()
|
|||||||
verified_str,
|
verified_str,
|
||||||
if !addr.is_empty() { addr } else { "addr unset" }
|
if !addr.is_empty() { addr } else { "addr unset" }
|
||||||
);
|
);
|
||||||
let peerstate = Peerstate::from_addr(context, addr)
|
|
||||||
.await
|
|
||||||
.expect("peerstate error");
|
|
||||||
if peerstate.is_some() && *contact_id != ContactId::SELF {
|
|
||||||
line2 = format!(
|
|
||||||
", prefer-encrypt={}",
|
|
||||||
peerstate.as_ref().unwrap().prefer_encrypt
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
println!("Contact#{}: {}{}", *contact_id, line, line2);
|
println!("Contact#{}: {}{}", *contact_id, line, line2);
|
||||||
}
|
}
|
||||||
@@ -342,7 +311,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
send-backup\n\
|
send-backup\n\
|
||||||
receive-backup <qr>\n\
|
receive-backup <qr>\n\
|
||||||
export-keys\n\
|
export-keys\n\
|
||||||
import-keys\n\
|
import-keys <key-file>\n\
|
||||||
poke [<eml-file>|<folder>|<addr> <key-file>]\n\
|
poke [<eml-file>|<folder>|<addr> <key-file>]\n\
|
||||||
reset <flags>\n\
|
reset <flags>\n\
|
||||||
stop\n\
|
stop\n\
|
||||||
@@ -351,8 +320,6 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
_ => println!(
|
_ => println!(
|
||||||
"==========================Database commands==\n\
|
"==========================Database commands==\n\
|
||||||
info\n\
|
info\n\
|
||||||
open <file to open or create>\n\
|
|
||||||
close\n\
|
|
||||||
set <configuration-key> [<value>]\n\
|
set <configuration-key> [<value>]\n\
|
||||||
get <configuration-key>\n\
|
get <configuration-key>\n\
|
||||||
oauth2\n\
|
oauth2\n\
|
||||||
@@ -367,28 +334,30 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
==============================Chat commands==\n\
|
==============================Chat commands==\n\
|
||||||
listchats [<query>]\n\
|
listchats [<query>]\n\
|
||||||
listarchived\n\
|
listarchived\n\
|
||||||
|
start-realtime <msg-id>\n\
|
||||||
|
send-realtime <msg-id> <data>\n\
|
||||||
chat [<chat-id>|0]\n\
|
chat [<chat-id>|0]\n\
|
||||||
createchat <contact-id>\n\
|
createchat <contact-id>\n\
|
||||||
creategroup <name>\n\
|
creategroup <name>\n\
|
||||||
createbroadcast\n\
|
createbroadcast <name>\n\
|
||||||
createprotected <name>\n\
|
|
||||||
addmember <contact-id>\n\
|
addmember <contact-id>\n\
|
||||||
removemember <contact-id>\n\
|
removemember <contact-id>\n\
|
||||||
groupname <name>\n\
|
groupname <name>\n\
|
||||||
groupimage [<file>]\n\
|
groupimage <image>\n\
|
||||||
chatinfo\n\
|
chatinfo\n\
|
||||||
sendlocations <seconds>\n\
|
sendlocations <seconds>\n\
|
||||||
setlocation <lat> <lng>\n\
|
setlocation <lat> <lng>\n\
|
||||||
dellocations\n\
|
dellocations\n\
|
||||||
getlocations [<contact-id>]\n\
|
getlocations [<contact-id>]\n\
|
||||||
send <text>\n\
|
send <text>\n\
|
||||||
|
send-sync <text>\n\
|
||||||
|
sendempty\n\
|
||||||
sendimage <file> [<text>]\n\
|
sendimage <file> [<text>]\n\
|
||||||
sendsticker <file> [<text>]\n\
|
sendsticker <file> [<text>]\n\
|
||||||
sendfile <file> [<text>]\n\
|
sendfile <file> [<text>]\n\
|
||||||
sendhtml <file for html-part> [<text for plain-part>]\n\
|
sendhtml <file for html-part> [<text for plain-part>]\n\
|
||||||
sendsyncmsg\n\
|
sendsyncmsg\n\
|
||||||
sendupdate <msg-id> <json status update>\n\
|
sendupdate <msg-id> <json status update>\n\
|
||||||
videochat\n\
|
|
||||||
draft [<text>]\n\
|
draft [<text>]\n\
|
||||||
devicemsg <text>\n\
|
devicemsg <text>\n\
|
||||||
listmedia\n\
|
listmedia\n\
|
||||||
@@ -400,7 +369,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
unmute <chat-id>\n\
|
unmute <chat-id>\n\
|
||||||
delchat <chat-id>\n\
|
delchat <chat-id>\n\
|
||||||
accept <chat-id>\n\
|
accept <chat-id>\n\
|
||||||
decline <chat-id>\n\
|
blockchat <chat-id>\n\
|
||||||
===========================Message commands==\n\
|
===========================Message commands==\n\
|
||||||
listmsgs <query>\n\
|
listmsgs <query>\n\
|
||||||
msginfo <msg-id>\n\
|
msginfo <msg-id>\n\
|
||||||
@@ -414,14 +383,14 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
react <msg-id> [<reaction>]\n\
|
react <msg-id> [<reaction>]\n\
|
||||||
===========================Contact commands==\n\
|
===========================Contact commands==\n\
|
||||||
listcontacts [<query>]\n\
|
listcontacts [<query>]\n\
|
||||||
listverified [<query>]\n\
|
|
||||||
addcontact [<name>] <addr>\n\
|
addcontact [<name>] <addr>\n\
|
||||||
contactinfo <contact-id>\n\
|
contactinfo <contact-id>\n\
|
||||||
delcontact <contact-id>\n\
|
delcontact <contact-id>\n\
|
||||||
cleanupcontacts\n\
|
|
||||||
block <contact-id>\n\
|
block <contact-id>\n\
|
||||||
unblock <contact-id>\n\
|
unblock <contact-id>\n\
|
||||||
listblocked\n\
|
listblocked\n\
|
||||||
|
import-vcard <file>\n\
|
||||||
|
make-vcard <file> <contact-id> [contact-id ...]\n\
|
||||||
======================================Misc.==\n\
|
======================================Misc.==\n\
|
||||||
getqr [<chat-id>]\n\
|
getqr [<chat-id>]\n\
|
||||||
getqrsvg [<chat-id>]\n\
|
getqrsvg [<chat-id>]\n\
|
||||||
@@ -442,7 +411,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
Ok(setup_code) => {
|
Ok(setup_code) => {
|
||||||
println!("Setup code for the transferred setup message: {setup_code}",)
|
println!("Setup code for the transferred setup message: {setup_code}",)
|
||||||
}
|
}
|
||||||
Err(err) => bail!("Failed to generate setup code: {}", err),
|
Err(err) => bail!("Failed to generate setup code: {err}"),
|
||||||
},
|
},
|
||||||
"get-setupcodebegin" => {
|
"get-setupcodebegin" => {
|
||||||
ensure!(!arg1.is_empty(), "Argument <msg-id> missing.");
|
ensure!(!arg1.is_empty(), "Argument <msg-id> missing.");
|
||||||
@@ -456,7 +425,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
setupcodebegin.unwrap_or_default(),
|
setupcodebegin.unwrap_or_default(),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
bail!("{} is no setup message.", msg_id,);
|
bail!("{msg_id} is no setup message.",);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"continue-key-transfer" => {
|
"continue-key-transfer" => {
|
||||||
@@ -493,7 +462,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
"send-backup" => {
|
"send-backup" => {
|
||||||
let provider = BackupProvider::prepare(&context).await?;
|
let provider = BackupProvider::prepare(&context).await?;
|
||||||
let qr = format_backup(&provider.qr())?;
|
let qr = format_backup(&provider.qr())?;
|
||||||
println!("QR code: {}", qr);
|
println!("QR code: {qr}");
|
||||||
qr2term::print_qr(qr.as_str())?;
|
qr2term::print_qr(qr.as_str())?;
|
||||||
provider.await?;
|
provider.await?;
|
||||||
}
|
}
|
||||||
@@ -508,13 +477,17 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
println!("Exported to {}.", dir.to_string_lossy());
|
println!("Exported to {}.", dir.to_string_lossy());
|
||||||
}
|
}
|
||||||
"import-keys" => {
|
"import-keys" => {
|
||||||
|
ensure!(!arg1.is_empty(), "Argument <key-file> missing.");
|
||||||
imex(&context, ImexMode::ImportSelfKeys, arg1.as_ref(), None).await?;
|
imex(&context, ImexMode::ImportSelfKeys, arg1.as_ref(), None).await?;
|
||||||
}
|
}
|
||||||
"poke" => {
|
"poke" => {
|
||||||
ensure!(poke_spec(&context, Some(arg1)).await, "Poke failed");
|
ensure!(poke_spec(&context, Some(arg1)).await, "Poke failed");
|
||||||
}
|
}
|
||||||
"reset" => {
|
"reset" => {
|
||||||
ensure!(!arg1.is_empty(), "Argument <bits> missing: 1=jobs, 2=peerstates, 4=private keys, 8=rest but server config");
|
ensure!(
|
||||||
|
!arg1.is_empty(),
|
||||||
|
"Argument <bits> missing: 4=private keys, 8=rest but server config"
|
||||||
|
);
|
||||||
let bits: i32 = arg1.parse()?;
|
let bits: i32 = arg1.parse()?;
|
||||||
ensure!(bits < 16, "<bits> must be lower than 16.");
|
ensure!(bits < 16, "<bits> must be lower than 16.");
|
||||||
reset_tables(&context, bits).await;
|
reset_tables(&context, bits).await;
|
||||||
@@ -547,7 +520,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
println!("Report written to: {file:#?}");
|
println!("Report written to: {file:#?}");
|
||||||
}
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
bail!("Failed to get connectivity html: {}", err);
|
bail!("Failed to get connectivity html: {err}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -582,7 +555,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
for i in (0..cnt).rev() {
|
for i in (0..cnt).rev() {
|
||||||
let chat = Chat::load_from_db(&context, chatlist.get_chat_id(i)?).await?;
|
let chat = Chat::load_from_db(&context, chatlist.get_chat_id(i)?).await?;
|
||||||
println!(
|
println!(
|
||||||
"{}#{}: {} [{} fresh] {}{}{}{}",
|
"{}#{}: {} [{} fresh] {}{}{}",
|
||||||
chat_prefix(&chat),
|
chat_prefix(&chat),
|
||||||
chat.get_id(),
|
chat.get_id(),
|
||||||
chat.get_name(),
|
chat.get_name(),
|
||||||
@@ -593,7 +566,6 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
ChatVisibility::Archived => "📦",
|
ChatVisibility::Archived => "📦",
|
||||||
ChatVisibility::Pinned => "📌",
|
ChatVisibility::Pinned => "📌",
|
||||||
},
|
},
|
||||||
if chat.is_protected() { "🛡️" } else { "" },
|
|
||||||
if chat.is_contact_request() {
|
if chat.is_contact_request() {
|
||||||
"🆕"
|
"🆕"
|
||||||
} else {
|
} else {
|
||||||
@@ -636,7 +608,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
println!("Location streaming enabled.");
|
println!("Location streaming enabled.");
|
||||||
}
|
}
|
||||||
println!("{cnt} chats");
|
println!("{cnt} chats");
|
||||||
println!("{time_needed:?} to create this list");
|
eprintln!("{time_needed:?} to create this list");
|
||||||
}
|
}
|
||||||
"start-realtime" => {
|
"start-realtime" => {
|
||||||
if arg1.is_empty() {
|
if arg1.is_empty() {
|
||||||
@@ -708,7 +680,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
format!("{} member(s)", members.len())
|
format!("{} member(s)", members.len())
|
||||||
};
|
};
|
||||||
println!(
|
println!(
|
||||||
"{}#{}: {} [{}]{}{}{} {}",
|
"{}#{}: {} [{}]{}{}{}",
|
||||||
chat_prefix(sel_chat),
|
chat_prefix(sel_chat),
|
||||||
sel_chat.get_id(),
|
sel_chat.get_id(),
|
||||||
sel_chat.get_name(),
|
sel_chat.get_name(),
|
||||||
@@ -726,11 +698,6 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
},
|
},
|
||||||
_ => "".to_string(),
|
_ => "".to_string(),
|
||||||
},
|
},
|
||||||
if sel_chat.is_protected() {
|
|
||||||
"🛡️"
|
|
||||||
} else {
|
|
||||||
""
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
log_msglist(&context, &msglist).await?;
|
log_msglist(&context, &msglist).await?;
|
||||||
if let Some(draft) = sel_chat.get_id().get_draft(&context).await? {
|
if let Some(draft) = sel_chat.get_id().get_draft(&context).await? {
|
||||||
@@ -746,7 +713,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
chat::marknoticed_chat(&context, sel_chat.get_id()).await?;
|
chat::marknoticed_chat(&context, sel_chat.get_id()).await?;
|
||||||
let time_noticed_needed = time_noticed_start.elapsed().unwrap_or_default();
|
let time_noticed_needed = time_noticed_start.elapsed().unwrap_or_default();
|
||||||
|
|
||||||
println!(
|
eprintln!(
|
||||||
"{time_needed:?} to create this list, {time_noticed_needed:?} to mark all messages as noticed."
|
"{time_needed:?} to create this list, {time_noticed_needed:?} to mark all messages as noticed."
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -759,23 +726,16 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
}
|
}
|
||||||
"creategroup" => {
|
"creategroup" => {
|
||||||
ensure!(!arg1.is_empty(), "Argument <name> missing.");
|
ensure!(!arg1.is_empty(), "Argument <name> missing.");
|
||||||
let chat_id =
|
let chat_id = chat::create_group(&context, arg1).await?;
|
||||||
chat::create_group_chat(&context, ProtectionStatus::Unprotected, arg1).await?;
|
|
||||||
|
|
||||||
println!("Group#{chat_id} created successfully.");
|
println!("Group#{chat_id} created successfully.");
|
||||||
}
|
}
|
||||||
"createbroadcast" => {
|
"createbroadcast" => {
|
||||||
let chat_id = chat::create_broadcast_list(&context).await?;
|
ensure!(!arg1.is_empty(), "Argument <name> missing.");
|
||||||
|
let chat_id = chat::create_broadcast(&context, arg1.to_string()).await?;
|
||||||
|
|
||||||
println!("Broadcast#{chat_id} created successfully.");
|
println!("Broadcast#{chat_id} created successfully.");
|
||||||
}
|
}
|
||||||
"createprotected" => {
|
|
||||||
ensure!(!arg1.is_empty(), "Argument <name> missing.");
|
|
||||||
let chat_id =
|
|
||||||
chat::create_group_chat(&context, ProtectionStatus::Protected, arg1).await?;
|
|
||||||
|
|
||||||
println!("Group#{chat_id} created and protected successfully.");
|
|
||||||
}
|
|
||||||
"addmember" => {
|
"addmember" => {
|
||||||
ensure!(sel_chat.is_some(), "No chat selected");
|
ensure!(sel_chat.is_some(), "No chat selected");
|
||||||
ensure!(!arg1.is_empty(), "Argument <contact-id> missing.");
|
ensure!(!arg1.is_empty(), "Argument <contact-id> missing.");
|
||||||
@@ -927,6 +887,23 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
|
|
||||||
chat::send_text_msg(&context, sel_chat.as_ref().unwrap().get_id(), msg).await?;
|
chat::send_text_msg(&context, sel_chat.as_ref().unwrap().get_id(), msg).await?;
|
||||||
}
|
}
|
||||||
|
"send-sync" => {
|
||||||
|
ensure!(sel_chat.is_some(), "No chat selected.");
|
||||||
|
ensure!(!arg1.is_empty(), "No message text given.");
|
||||||
|
|
||||||
|
// Send message over a dedicated SMTP connection
|
||||||
|
// and measure time.
|
||||||
|
//
|
||||||
|
// This can be used to benchmark SMTP connection establishment.
|
||||||
|
let time_start = std::time::Instant::now();
|
||||||
|
|
||||||
|
let msg = format!("{arg1} {arg2}");
|
||||||
|
let mut msg = Message::new_text(msg);
|
||||||
|
chat::send_msg_sync(&context, sel_chat.as_ref().unwrap().get_id(), &mut msg).await?;
|
||||||
|
|
||||||
|
let time_needed = time_start.elapsed();
|
||||||
|
println!("Sent message in {time_needed:?}.");
|
||||||
|
}
|
||||||
"sendempty" => {
|
"sendempty" => {
|
||||||
ensure!(sel_chat.is_some(), "No chat selected.");
|
ensure!(sel_chat.is_some(), "No chat selected.");
|
||||||
chat::send_text_msg(&context, sel_chat.as_ref().unwrap().get_id(), "".into()).await?;
|
chat::send_text_msg(&context, sel_chat.as_ref().unwrap().get_id(), "".into()).await?;
|
||||||
@@ -974,10 +951,6 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
let msg_id = MsgId::new(arg1.parse()?);
|
let msg_id = MsgId::new(arg1.parse()?);
|
||||||
context.send_webxdc_status_update(msg_id, arg2).await?;
|
context.send_webxdc_status_update(msg_id, arg2).await?;
|
||||||
}
|
}
|
||||||
"videochat" => {
|
|
||||||
ensure!(sel_chat.is_some(), "No chat selected.");
|
|
||||||
chat::send_videochat_invitation(&context, sel_chat.as_ref().unwrap().get_id()).await?;
|
|
||||||
}
|
|
||||||
"listmsgs" => {
|
"listmsgs" => {
|
||||||
ensure!(!arg1.is_empty(), "Argument <query> missing.");
|
ensure!(!arg1.is_empty(), "Argument <query> missing.");
|
||||||
|
|
||||||
@@ -999,7 +972,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
},
|
},
|
||||||
query,
|
query,
|
||||||
);
|
);
|
||||||
println!("{time_needed:?} to create this list");
|
eprintln!("{time_needed:?} to create this list");
|
||||||
}
|
}
|
||||||
"draft" => {
|
"draft" => {
|
||||||
ensure!(sel_chat.is_some(), "No chat selected.");
|
ensure!(sel_chat.is_some(), "No chat selected.");
|
||||||
@@ -1162,19 +1135,13 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
let reaction = arg2;
|
let reaction = arg2;
|
||||||
send_reaction(&context, msg_id, reaction).await?;
|
send_reaction(&context, msg_id, reaction).await?;
|
||||||
}
|
}
|
||||||
"listcontacts" | "contacts" | "listverified" => {
|
"listcontacts" | "contacts" => {
|
||||||
let contacts = Contact::get_all(
|
let contacts = Contact::get_all(&context, DC_GCL_ADD_SELF, Some(arg1)).await?;
|
||||||
&context,
|
|
||||||
if arg0 == "listverified" {
|
|
||||||
DC_GCL_VERIFIED_ONLY | DC_GCL_ADD_SELF
|
|
||||||
} else {
|
|
||||||
DC_GCL_ADD_SELF
|
|
||||||
},
|
|
||||||
Some(arg1),
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
log_contactlist(&context, &contacts).await?;
|
log_contactlist(&context, &contacts).await?;
|
||||||
println!("{} contacts.", contacts.len());
|
println!("{} key contacts.", contacts.len());
|
||||||
|
let addrcontacts = Contact::get_all(&context, DC_GCL_ADDRESS, Some(arg1)).await?;
|
||||||
|
log_contactlist(&context, &addrcontacts).await?;
|
||||||
|
println!("{} address contacts.", addrcontacts.len());
|
||||||
}
|
}
|
||||||
"addcontact" => {
|
"addcontact" => {
|
||||||
ensure!(!arg1.is_empty(), "Arguments [<name>] <addr> expected.");
|
ensure!(!arg1.is_empty(), "Arguments [<name>] <addr> expected.");
|
||||||
@@ -1238,6 +1205,24 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
log_contactlist(&context, &contacts).await?;
|
log_contactlist(&context, &contacts).await?;
|
||||||
println!("{} blocked contacts.", contacts.len());
|
println!("{} blocked contacts.", contacts.len());
|
||||||
}
|
}
|
||||||
|
"import-vcard" => {
|
||||||
|
ensure!(!arg1.is_empty(), "Argument <file> missing.");
|
||||||
|
let vcard_content = fs::read_to_string(&arg1.to_string()).await?;
|
||||||
|
let contacts = import_vcard(&context, &vcard_content).await?;
|
||||||
|
println!("vCard contacts imported:");
|
||||||
|
log_contactlist(&context, &contacts).await?;
|
||||||
|
}
|
||||||
|
"make-vcard" => {
|
||||||
|
ensure!(!arg1.is_empty(), "Argument <file> missing.");
|
||||||
|
ensure!(!arg2.is_empty(), "Argument <contact-id> missing.");
|
||||||
|
let mut contact_ids = vec![];
|
||||||
|
for x in arg2.split_whitespace() {
|
||||||
|
contact_ids.push(ContactId::new(x.parse()?))
|
||||||
|
}
|
||||||
|
let vcard_content = make_vcard(&context, &contact_ids).await?;
|
||||||
|
fs::write(&arg1.to_string(), vcard_content).await?;
|
||||||
|
println!("vCard written to: {arg1}");
|
||||||
|
}
|
||||||
"checkqr" => {
|
"checkqr" => {
|
||||||
ensure!(!arg1.is_empty(), "Argument <qr-content> missing.");
|
ensure!(!arg1.is_empty(), "Argument <qr-content> missing.");
|
||||||
let qr = check_qr(&context, arg1).await?;
|
let qr = check_qr(&context, arg1).await?;
|
||||||
@@ -1247,7 +1232,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
ensure!(!arg1.is_empty(), "Argument <qr-content> missing.");
|
ensure!(!arg1.is_empty(), "Argument <qr-content> missing.");
|
||||||
match set_config_from_qr(&context, arg1).await {
|
match set_config_from_qr(&context, arg1).await {
|
||||||
Ok(()) => println!("Config set from QR code, you can now call 'configure'"),
|
Ok(()) => println!("Config set from QR code, you can now call 'configure'"),
|
||||||
Err(err) => println!("Cannot set config from QR code: {err:?}"),
|
Err(err) => eprintln!("Cannot set config from QR code: {err:?}"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"createqrsvg" => {
|
"createqrsvg" => {
|
||||||
@@ -1259,10 +1244,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
}
|
}
|
||||||
"providerinfo" => {
|
"providerinfo" => {
|
||||||
ensure!(!arg1.is_empty(), "Argument <addr> missing.");
|
ensure!(!arg1.is_empty(), "Argument <addr> missing.");
|
||||||
let proxy_enabled = context
|
match provider::get_provider_info(arg1) {
|
||||||
.get_config_bool(config::Config::ProxyEnabled)
|
|
||||||
.await?;
|
|
||||||
match provider::get_provider_info(&context, arg1, proxy_enabled).await {
|
|
||||||
Some(info) => {
|
Some(info) => {
|
||||||
println!("Information for provider belonging to {arg1}:");
|
println!("Information for provider belonging to {arg1}:");
|
||||||
println!("status: {}", info.status as u32);
|
println!("status: {}", info.status as u32);
|
||||||
@@ -1298,7 +1280,7 @@ pub async fn cmdline(context: Context, line: &str, chat_id: &mut ChatId) -> Resu
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
"" => (),
|
"" => (),
|
||||||
_ => bail!("Unknown command: \"{}\" type ? for help.", arg0),
|
_ => bail!("Unknown command: \"{arg0}\" type ? for help."),
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|||||||
@@ -5,7 +5,6 @@
|
|||||||
//! Usage: cargo run --example repl --release -- <databasefile>
|
//! Usage: cargo run --example repl --release -- <databasefile>
|
||||||
//! All further options can be set using the set-command (type ? for help).
|
//! All further options can be set using the set-command (type ? for help).
|
||||||
|
|
||||||
#[macro_use]
|
|
||||||
extern crate deltachat;
|
extern crate deltachat;
|
||||||
|
|
||||||
use std::borrow::Cow::{self, Borrowed, Owned};
|
use std::borrow::Cow::{self, Borrowed, Owned};
|
||||||
@@ -41,25 +40,25 @@ fn receive_event(event: EventType) {
|
|||||||
match event {
|
match event {
|
||||||
EventType::Info(msg) => {
|
EventType::Info(msg) => {
|
||||||
/* do not show the event as this would fill the screen */
|
/* do not show the event as this would fill the screen */
|
||||||
info!("{}", msg);
|
info!("{msg}");
|
||||||
}
|
}
|
||||||
EventType::SmtpConnected(msg) => {
|
EventType::SmtpConnected(msg) => {
|
||||||
info!("[SMTP_CONNECTED] {}", msg);
|
info!("[SMTP_CONNECTED] {msg}");
|
||||||
}
|
}
|
||||||
EventType::ImapConnected(msg) => {
|
EventType::ImapConnected(msg) => {
|
||||||
info!("[IMAP_CONNECTED] {}", msg);
|
info!("[IMAP_CONNECTED] {msg}");
|
||||||
}
|
}
|
||||||
EventType::SmtpMessageSent(msg) => {
|
EventType::SmtpMessageSent(msg) => {
|
||||||
info!("[SMTP_MESSAGE_SENT] {}", msg);
|
info!("[SMTP_MESSAGE_SENT] {msg}");
|
||||||
}
|
}
|
||||||
EventType::Warning(msg) => {
|
EventType::Warning(msg) => {
|
||||||
warn!("{}", msg);
|
warn!("{msg}");
|
||||||
}
|
}
|
||||||
EventType::Error(msg) => {
|
EventType::Error(msg) => {
|
||||||
error!("{}", msg);
|
error!("{msg}");
|
||||||
}
|
}
|
||||||
EventType::ErrorSelfNotInGroup(msg) => {
|
EventType::ErrorSelfNotInGroup(msg) => {
|
||||||
error!("[SELF_NOT_IN_GROUP] {}", msg);
|
error!("[SELF_NOT_IN_GROUP] {msg}");
|
||||||
}
|
}
|
||||||
EventType::MsgsChanged { chat_id, msg_id } => {
|
EventType::MsgsChanged { chat_id, msg_id } => {
|
||||||
info!(
|
info!(
|
||||||
@@ -124,7 +123,7 @@ fn receive_event(event: EventType) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
info!("Received {:?}", event);
|
info!("Received {event:?}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -180,9 +179,11 @@ const DB_COMMANDS: [&str; 11] = [
|
|||||||
"housekeeping",
|
"housekeeping",
|
||||||
];
|
];
|
||||||
|
|
||||||
const CHAT_COMMANDS: [&str; 36] = [
|
const CHAT_COMMANDS: [&str; 39] = [
|
||||||
"listchats",
|
"listchats",
|
||||||
"listarchived",
|
"listarchived",
|
||||||
|
"start-realtime",
|
||||||
|
"send-realtime",
|
||||||
"chat",
|
"chat",
|
||||||
"createchat",
|
"createchat",
|
||||||
"creategroup",
|
"creategroup",
|
||||||
@@ -198,13 +199,16 @@ const CHAT_COMMANDS: [&str; 36] = [
|
|||||||
"dellocations",
|
"dellocations",
|
||||||
"getlocations",
|
"getlocations",
|
||||||
"send",
|
"send",
|
||||||
|
"send-sync",
|
||||||
|
"sendempty",
|
||||||
"sendimage",
|
"sendimage",
|
||||||
|
"sendsticker",
|
||||||
"sendfile",
|
"sendfile",
|
||||||
"sendhtml",
|
"sendhtml",
|
||||||
"sendsyncmsg",
|
"sendsyncmsg",
|
||||||
"sendupdate",
|
"sendupdate",
|
||||||
"videochat",
|
|
||||||
"draft",
|
"draft",
|
||||||
|
"devicemsg",
|
||||||
"listmedia",
|
"listmedia",
|
||||||
"archive",
|
"archive",
|
||||||
"unarchive",
|
"unarchive",
|
||||||
@@ -212,47 +216,48 @@ const CHAT_COMMANDS: [&str; 36] = [
|
|||||||
"unpin",
|
"unpin",
|
||||||
"mute",
|
"mute",
|
||||||
"unmute",
|
"unmute",
|
||||||
"protect",
|
|
||||||
"unprotect",
|
|
||||||
"delchat",
|
"delchat",
|
||||||
"accept",
|
"accept",
|
||||||
"blockchat",
|
"blockchat",
|
||||||
];
|
];
|
||||||
const MESSAGE_COMMANDS: [&str; 9] = [
|
const MESSAGE_COMMANDS: [&str; 10] = [
|
||||||
"listmsgs",
|
"listmsgs",
|
||||||
"msginfo",
|
"msginfo",
|
||||||
|
"download",
|
||||||
|
"html",
|
||||||
"listfresh",
|
"listfresh",
|
||||||
"forward",
|
"forward",
|
||||||
"resend",
|
"resend",
|
||||||
"markseen",
|
"markseen",
|
||||||
"delmsg",
|
"delmsg",
|
||||||
"download",
|
|
||||||
"react",
|
"react",
|
||||||
];
|
];
|
||||||
const CONTACT_COMMANDS: [&str; 9] = [
|
const CONTACT_COMMANDS: [&str; 9] = [
|
||||||
"listcontacts",
|
"listcontacts",
|
||||||
"listverified",
|
|
||||||
"addcontact",
|
"addcontact",
|
||||||
"contactinfo",
|
"contactinfo",
|
||||||
"delcontact",
|
"delcontact",
|
||||||
"cleanupcontacts",
|
|
||||||
"block",
|
"block",
|
||||||
"unblock",
|
"unblock",
|
||||||
"listblocked",
|
"listblocked",
|
||||||
|
"import-vcard",
|
||||||
|
"make-vcard",
|
||||||
];
|
];
|
||||||
const MISC_COMMANDS: [&str; 12] = [
|
const MISC_COMMANDS: [&str; 14] = [
|
||||||
"getqr",
|
"getqr",
|
||||||
"getqrsvg",
|
"getqrsvg",
|
||||||
"getbadqr",
|
"getbadqr",
|
||||||
"checkqr",
|
"checkqr",
|
||||||
"joinqr",
|
"joinqr",
|
||||||
|
"setqr",
|
||||||
"createqrsvg",
|
"createqrsvg",
|
||||||
|
"providerinfo",
|
||||||
"fileinfo",
|
"fileinfo",
|
||||||
|
"estimatedeletion",
|
||||||
"clear",
|
"clear",
|
||||||
"exit",
|
"exit",
|
||||||
"quit",
|
"quit",
|
||||||
"help",
|
"help",
|
||||||
"estimatedeletion",
|
|
||||||
];
|
];
|
||||||
|
|
||||||
impl Hinter for DcHelper {
|
impl Hinter for DcHelper {
|
||||||
@@ -308,7 +313,7 @@ impl Validator for DcHelper {}
|
|||||||
|
|
||||||
async fn start(args: Vec<String>) -> Result<(), Error> {
|
async fn start(args: Vec<String>) -> Result<(), Error> {
|
||||||
if args.len() < 2 {
|
if args.len() < 2 {
|
||||||
println!("Error: Bad arguments, expected [db-name].");
|
eprintln!("Error: Bad arguments, expected [db-name].");
|
||||||
bail!("No db-name specified");
|
bail!("No db-name specified");
|
||||||
}
|
}
|
||||||
let context = ContextBuilder::new(args[1].clone().into())
|
let context = ContextBuilder::new(args[1].clone().into())
|
||||||
@@ -323,7 +328,7 @@ async fn start(args: Vec<String>) -> Result<(), Error> {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
println!("Delta Chat Core is awaiting your commands.");
|
println!("Chatmail is awaiting your commands.");
|
||||||
|
|
||||||
let config = Config::builder()
|
let config = Config::builder()
|
||||||
.history_ignore_space(true)
|
.history_ignore_space(true)
|
||||||
@@ -363,7 +368,7 @@ async fn start(args: Vec<String>) -> Result<(), Error> {
|
|||||||
false
|
false
|
||||||
}
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
println!("Error: {err:#}");
|
eprintln!("Error: {err:#}");
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -378,7 +383,7 @@ async fn start(args: Vec<String>) -> Result<(), Error> {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
println!("Error: {err:#}");
|
eprintln!("Error: {err:#}");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -462,7 +467,7 @@ async fn handle_cmd(
|
|||||||
println!("QR code svg written to: {file:#?}");
|
println!("QR code svg written to: {file:#?}");
|
||||||
}
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
bail!("Failed to get QR code svg: {}", err);
|
bail!("Failed to get QR code svg: {err}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "deltachat-rpc-client"
|
name = "deltachat-rpc-client"
|
||||||
version = "1.157.3"
|
version = "2.25.0"
|
||||||
description = "Python client for Delta Chat core JSON-RPC interface"
|
description = "Python client for Delta Chat core JSON-RPC interface"
|
||||||
classifiers = [
|
classifiers = [
|
||||||
"Development Status :: 5 - Production/Stable",
|
"Development Status :: 5 - Production/Stable",
|
||||||
@@ -19,6 +19,7 @@ classifiers = [
|
|||||||
"Programming Language :: Python :: 3.11",
|
"Programming Language :: Python :: 3.11",
|
||||||
"Programming Language :: Python :: 3.12",
|
"Programming Language :: Python :: 3.12",
|
||||||
"Programming Language :: Python :: 3.13",
|
"Programming Language :: Python :: 3.13",
|
||||||
|
"Programming Language :: Python :: 3.14",
|
||||||
"Topic :: Communications :: Chat",
|
"Topic :: Communications :: Chat",
|
||||||
"Topic :: Communications :: Email"
|
"Topic :: Communications :: Email"
|
||||||
]
|
]
|
||||||
@@ -66,7 +67,18 @@ lint.select = [
|
|||||||
|
|
||||||
"RUF006" # asyncio-dangling-task
|
"RUF006" # asyncio-dangling-task
|
||||||
]
|
]
|
||||||
|
lint.ignore = [
|
||||||
|
"PLC0415" # `import` should be at the top-level of a file
|
||||||
|
]
|
||||||
line-length = 120
|
line-length = 120
|
||||||
|
|
||||||
[tool.isort]
|
[tool.isort]
|
||||||
profile = "black"
|
profile = "black"
|
||||||
|
|
||||||
|
[dependency-groups]
|
||||||
|
dev = [
|
||||||
|
"imap-tools",
|
||||||
|
"pytest",
|
||||||
|
"pytest-timeout",
|
||||||
|
"pytest-xdist",
|
||||||
|
]
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
"""Delta Chat JSON-RPC high-level API"""
|
"""Delta Chat JSON-RPC high-level API."""
|
||||||
|
|
||||||
from ._utils import AttrDict, run_bot_cli, run_client_cli
|
from ._utils import AttrDict, run_bot_cli, run_client_cli
|
||||||
from .account import Account
|
from .account import Account
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import argparse
|
import argparse
|
||||||
|
import os
|
||||||
import re
|
import re
|
||||||
import sys
|
import sys
|
||||||
from threading import Thread
|
from threading import Thread
|
||||||
@@ -89,8 +90,8 @@ def _run_cli(
|
|||||||
help="accounts folder (default: current working directory)",
|
help="accounts folder (default: current working directory)",
|
||||||
nargs="?",
|
nargs="?",
|
||||||
)
|
)
|
||||||
parser.add_argument("--email", action="store", help="email address")
|
parser.add_argument("--email", action="store", help="email address", default=os.getenv("DELTACHAT_EMAIL"))
|
||||||
parser.add_argument("--password", action="store", help="password")
|
parser.add_argument("--password", action="store", help="password", default=os.getenv("DELTACHAT_PASSWORD"))
|
||||||
args = parser.parse_args(argv[1:])
|
args = parser.parse_args(argv[1:])
|
||||||
|
|
||||||
with Rpc(accounts_dir=args.accounts_dir, **kwargs) as rpc:
|
with Rpc(accounts_dir=args.accounts_dir, **kwargs) as rpc:
|
||||||
@@ -114,7 +115,7 @@ def _run_cli(
|
|||||||
|
|
||||||
|
|
||||||
def extract_addr(text: str) -> str:
|
def extract_addr(text: str) -> str:
|
||||||
"""extract email address from the given text."""
|
"""Extract email address from the given text."""
|
||||||
match = re.match(r".*\((.+@.+)\)", text)
|
match = re.match(r".*\((.+@.+)\)", text)
|
||||||
if match:
|
if match:
|
||||||
text = match.group(1)
|
text = match.group(1)
|
||||||
@@ -123,7 +124,7 @@ def extract_addr(text: str) -> str:
|
|||||||
|
|
||||||
|
|
||||||
def parse_system_image_changed(text: str) -> Optional[Tuple[str, bool]]:
|
def parse_system_image_changed(text: str) -> Optional[Tuple[str, bool]]:
|
||||||
"""return image changed/deleted info from parsing the given system message text."""
|
"""Return image changed/deleted info from parsing the given system message text."""
|
||||||
text = text.lower()
|
text = text.lower()
|
||||||
match = re.match(r"group image (changed|deleted) by (.+).", text)
|
match = re.match(r"group image (changed|deleted) by (.+).", text)
|
||||||
if match:
|
if match:
|
||||||
@@ -142,7 +143,7 @@ def parse_system_title_changed(text: str) -> Optional[Tuple[str, str]]:
|
|||||||
|
|
||||||
|
|
||||||
def parse_system_add_remove(text: str) -> Optional[Tuple[str, str, str]]:
|
def parse_system_add_remove(text: str) -> Optional[Tuple[str, str, str]]:
|
||||||
"""return add/remove info from parsing the given system message text.
|
"""Return add/remove info from parsing the given system message text.
|
||||||
|
|
||||||
returns a (action, affected, actor) tuple.
|
returns a (action, affected, actor) tuple.
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
|
"""Account module."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import json
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from typing import TYPE_CHECKING, Optional, Union
|
from typing import TYPE_CHECKING, Optional, Union
|
||||||
from warnings import warn
|
from warnings import warn
|
||||||
@@ -34,7 +37,10 @@ class Account:
|
|||||||
return next_event
|
return next_event
|
||||||
|
|
||||||
def clear_all_events(self):
|
def clear_all_events(self):
|
||||||
"""Removes all queued-up events for a given account. Useful for tests."""
|
"""Remove all queued-up events for a given account.
|
||||||
|
|
||||||
|
Useful for tests.
|
||||||
|
"""
|
||||||
self._rpc.clear_all_events(self.id)
|
self._rpc.clear_all_events(self.id)
|
||||||
|
|
||||||
def remove(self) -> None:
|
def remove(self) -> None:
|
||||||
@@ -43,7 +49,9 @@ class Account:
|
|||||||
|
|
||||||
def clone(self) -> "Account":
|
def clone(self) -> "Account":
|
||||||
"""Clone given account.
|
"""Clone given account.
|
||||||
This uses backup-transfer via iroh, i.e. the 'Add second device' feature."""
|
|
||||||
|
This uses backup-transfer via iroh, i.e. the 'Add second device' feature.
|
||||||
|
"""
|
||||||
future = self._rpc.provide_backup.future(self.id)
|
future = self._rpc.provide_backup.future(self.id)
|
||||||
qr = self._rpc.get_backup_qr(self.id)
|
qr = self._rpc.get_backup_qr(self.id)
|
||||||
new_account = self.manager.add_account()
|
new_account = self.manager.add_account()
|
||||||
@@ -80,7 +88,7 @@ class Account:
|
|||||||
return self._rpc.get_config(self.id, key)
|
return self._rpc.get_config(self.id, key)
|
||||||
|
|
||||||
def update_config(self, **kwargs) -> None:
|
def update_config(self, **kwargs) -> None:
|
||||||
"""update config values."""
|
"""Update config values."""
|
||||||
for key, value in kwargs.items():
|
for key, value in kwargs.items():
|
||||||
self.set_config(key, value)
|
self.set_config(key, value)
|
||||||
|
|
||||||
@@ -99,10 +107,12 @@ class Account:
|
|||||||
"""Parse QR code contents.
|
"""Parse QR code contents.
|
||||||
|
|
||||||
This function takes the raw text scanned
|
This function takes the raw text scanned
|
||||||
and checks what can be done with it."""
|
and checks what can be done with it.
|
||||||
|
"""
|
||||||
return self._rpc.check_qr(self.id, qr)
|
return self._rpc.check_qr(self.id, qr)
|
||||||
|
|
||||||
def set_config_from_qr(self, qr: str):
|
def set_config_from_qr(self, qr: str):
|
||||||
|
"""Set configuration values from a QR code."""
|
||||||
self._rpc.set_config_from_qr(self.id, qr)
|
self._rpc.set_config_from_qr(self.id, qr)
|
||||||
|
|
||||||
@futuremethod
|
@futuremethod
|
||||||
@@ -110,12 +120,28 @@ class Account:
|
|||||||
"""Configure an account."""
|
"""Configure an account."""
|
||||||
yield self._rpc.configure.future(self.id)
|
yield self._rpc.configure.future(self.id)
|
||||||
|
|
||||||
|
@futuremethod
|
||||||
|
def add_or_update_transport(self, params):
|
||||||
|
"""Add a new transport."""
|
||||||
|
yield self._rpc.add_or_update_transport.future(self.id, params)
|
||||||
|
|
||||||
|
@futuremethod
|
||||||
|
def add_transport_from_qr(self, qr: str):
|
||||||
|
"""Add a new transport using a QR code."""
|
||||||
|
yield self._rpc.add_transport_from_qr.future(self.id, qr)
|
||||||
|
|
||||||
|
@futuremethod
|
||||||
|
def list_transports(self):
|
||||||
|
"""Return the list of all email accounts that are used as a transport in the current profile."""
|
||||||
|
transports = yield self._rpc.list_transports.future(self.id)
|
||||||
|
return transports
|
||||||
|
|
||||||
def bring_online(self):
|
def bring_online(self):
|
||||||
"""Start I/O and wait until IMAP becomes IDLE."""
|
"""Start I/O and wait until IMAP becomes IDLE."""
|
||||||
self.start_io()
|
self.start_io()
|
||||||
self.wait_for_event(EventType.IMAP_INBOX_IDLE)
|
self.wait_for_event(EventType.IMAP_INBOX_IDLE)
|
||||||
|
|
||||||
def create_contact(self, obj: Union[int, str, Contact], name: Optional[str] = None) -> Contact:
|
def create_contact(self, obj: Union[int, str, Contact, "Account"], name: Optional[str] = None) -> Contact:
|
||||||
"""Create a new Contact or return an existing one.
|
"""Create a new Contact or return an existing one.
|
||||||
|
|
||||||
Calling this method will always result in the same
|
Calling this method will always result in the same
|
||||||
@@ -123,9 +149,15 @@ class Account:
|
|||||||
with that e-mail address, it is unblocked and its display
|
with that e-mail address, it is unblocked and its display
|
||||||
name is updated if specified.
|
name is updated if specified.
|
||||||
|
|
||||||
:param obj: email-address or contact id.
|
:param obj: email-address, contact id or account.
|
||||||
:param name: (optional) display name for this contact.
|
:param name: (optional) display name for this contact.
|
||||||
"""
|
"""
|
||||||
|
if isinstance(obj, Account):
|
||||||
|
vcard = obj.self_contact.make_vcard()
|
||||||
|
[contact] = self.import_vcard(vcard)
|
||||||
|
if name:
|
||||||
|
contact.set_name(name)
|
||||||
|
return contact
|
||||||
if isinstance(obj, int):
|
if isinstance(obj, int):
|
||||||
obj = Contact(self, obj)
|
obj = Contact(self, obj)
|
||||||
if isinstance(obj, Contact):
|
if isinstance(obj, Contact):
|
||||||
@@ -141,14 +173,14 @@ class Account:
|
|||||||
def import_vcard(self, vcard: str) -> list[Contact]:
|
def import_vcard(self, vcard: str) -> list[Contact]:
|
||||||
"""Import vCard.
|
"""Import vCard.
|
||||||
|
|
||||||
Return created or modified contacts in the order they appear in vCard."""
|
Return created or modified contacts in the order they appear in vCard.
|
||||||
|
"""
|
||||||
contact_ids = self._rpc.import_vcard_contents(self.id, vcard)
|
contact_ids = self._rpc.import_vcard_contents(self.id, vcard)
|
||||||
return [Contact(self, contact_id) for contact_id in contact_ids]
|
return [Contact(self, contact_id) for contact_id in contact_ids]
|
||||||
|
|
||||||
def create_chat(self, account: "Account") -> Chat:
|
def create_chat(self, account: "Account") -> Chat:
|
||||||
vcard = account.self_contact.make_vcard()
|
"""Create a 1:1 chat with another account."""
|
||||||
[contact] = self.import_vcard(vcard)
|
return self.create_contact(account).create_chat()
|
||||||
return contact.create_chat()
|
|
||||||
|
|
||||||
def get_device_chat(self) -> Chat:
|
def get_device_chat(self) -> Chat:
|
||||||
"""Return device chat."""
|
"""Return device chat."""
|
||||||
@@ -159,7 +191,21 @@ class Account:
|
|||||||
return Contact(self, contact_id)
|
return Contact(self, contact_id)
|
||||||
|
|
||||||
def get_contact_by_addr(self, address: str) -> Optional[Contact]:
|
def get_contact_by_addr(self, address: str) -> Optional[Contact]:
|
||||||
"""Check if an e-mail address belongs to a known and unblocked contact."""
|
"""Looks up a known and unblocked contact with a given e-mail address.
|
||||||
|
To get a list of all known and unblocked contacts, use contacts_get_contacts().
|
||||||
|
|
||||||
|
**POTENTIAL SECURITY ISSUE**: If there are multiple contacts with this address
|
||||||
|
(e.g. an address-contact and a key-contact),
|
||||||
|
this looks up the most recently seen contact,
|
||||||
|
i.e. which contact is returned depends on which contact last sent a message.
|
||||||
|
If the user just clicked on a mailto: link, then this is the best thing you can do.
|
||||||
|
But **DO NOT** internally represent contacts by their email address
|
||||||
|
and do not use this function to look them up;
|
||||||
|
otherwise this function will sometimes look up the wrong contact.
|
||||||
|
Instead, you should internally represent contacts by their ids.
|
||||||
|
|
||||||
|
To validate an e-mail address independently of the contact database
|
||||||
|
use check_email_validity()."""
|
||||||
contact_id = self._rpc.lookup_contact_id_by_addr(self.id, address)
|
contact_id = self._rpc.lookup_contact_id_by_addr(self.id, address)
|
||||||
return contact_id and Contact(self, contact_id)
|
return contact_id and Contact(self, contact_id)
|
||||||
|
|
||||||
@@ -185,8 +231,8 @@ class Account:
|
|||||||
def get_contacts(
|
def get_contacts(
|
||||||
self,
|
self,
|
||||||
query: Optional[str] = None,
|
query: Optional[str] = None,
|
||||||
|
*,
|
||||||
with_self: bool = False,
|
with_self: bool = False,
|
||||||
verified_only: bool = False,
|
|
||||||
snapshot: bool = False,
|
snapshot: bool = False,
|
||||||
) -> Union[list[Contact], list[AttrDict]]:
|
) -> Union[list[Contact], list[AttrDict]]:
|
||||||
"""Get a filtered list of contacts.
|
"""Get a filtered list of contacts.
|
||||||
@@ -194,12 +240,9 @@ class Account:
|
|||||||
:param query: if a string is specified, only return contacts
|
:param query: if a string is specified, only return contacts
|
||||||
whose name or e-mail matches query.
|
whose name or e-mail matches query.
|
||||||
:param with_self: if True the self-contact is also included if it matches the query.
|
:param with_self: if True the self-contact is also included if it matches the query.
|
||||||
:param only_verified: if True only return verified contacts.
|
|
||||||
:param snapshot: If True return a list of contact snapshots instead of Contact instances.
|
:param snapshot: If True return a list of contact snapshots instead of Contact instances.
|
||||||
"""
|
"""
|
||||||
flags = 0
|
flags = 0
|
||||||
if verified_only:
|
|
||||||
flags |= ContactFlag.VERIFIED_ONLY
|
|
||||||
if with_self:
|
if with_self:
|
||||||
flags |= ContactFlag.ADD_SELF
|
flags |= ContactFlag.ADD_SELF
|
||||||
|
|
||||||
@@ -211,12 +254,12 @@ class Account:
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def self_contact(self) -> Contact:
|
def self_contact(self) -> Contact:
|
||||||
"""This account's identity as a Contact."""
|
"""Account's identity as a Contact."""
|
||||||
return Contact(self, SpecialContactId.SELF)
|
return Contact(self, SpecialContactId.SELF)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def device_contact(self) -> Chat:
|
def device_contact(self) -> Chat:
|
||||||
"""This account's device contact."""
|
"""Account's device contact."""
|
||||||
return Contact(self, SpecialContactId.DEVICE)
|
return Contact(self, SpecialContactId.DEVICE)
|
||||||
|
|
||||||
def get_chatlist(
|
def get_chatlist(
|
||||||
@@ -262,20 +305,51 @@ class Account:
|
|||||||
chats.append(AttrDict(item))
|
chats.append(AttrDict(item))
|
||||||
return chats
|
return chats
|
||||||
|
|
||||||
def create_group(self, name: str, protect: bool = False) -> Chat:
|
def create_group(self, name: str) -> Chat:
|
||||||
"""Create a new group chat.
|
"""Create a new group chat.
|
||||||
|
|
||||||
After creation, the group has only self-contact as member and is in unpromoted state.
|
After creation,
|
||||||
|
the group has only self-contact as member one member (see `SpecialContactId.SELF`)
|
||||||
|
and is in _unpromoted_ state.
|
||||||
|
This means, you can add or remove members, change the name,
|
||||||
|
the group image and so on without messages being sent to all group members.
|
||||||
|
|
||||||
|
This changes as soon as the first message is sent to the group members
|
||||||
|
and the group becomes _promoted_.
|
||||||
|
After that, all changes are synced with all group members
|
||||||
|
by sending status message.
|
||||||
|
|
||||||
|
To check, if a chat is still unpromoted, you can look at the `is_unpromoted` property of a chat
|
||||||
|
(see `get_full_snapshot()` / `get_basic_snapshot()`).
|
||||||
|
This may be useful if you want to show some help for just created groups.
|
||||||
"""
|
"""
|
||||||
return Chat(self, self._rpc.create_group_chat(self.id, name, protect))
|
return Chat(self, self._rpc.create_group_chat(self.id, name, False))
|
||||||
|
|
||||||
|
def create_broadcast(self, name: str) -> Chat:
|
||||||
|
"""Create a new, outgoing **broadcast channel**
|
||||||
|
(called "Channel" in the UI).
|
||||||
|
|
||||||
|
Broadcast channels are similar to groups on the sending device,
|
||||||
|
however, recipients get the messages in a read-only chat
|
||||||
|
and will not see who the other members are.
|
||||||
|
|
||||||
|
Called `broadcast` here rather than `channel`,
|
||||||
|
because the word "channel" already appears a lot in the code,
|
||||||
|
which would make it hard to grep for it.
|
||||||
|
|
||||||
|
After creation, the chat contains no recipients and is in _unpromoted_ state;
|
||||||
|
see `create_group()` for more information on the unpromoted state.
|
||||||
|
|
||||||
|
Returns the created chat.
|
||||||
|
"""
|
||||||
|
return Chat(self, self._rpc.create_broadcast(self.id, name))
|
||||||
|
|
||||||
def get_chat_by_id(self, chat_id: int) -> Chat:
|
def get_chat_by_id(self, chat_id: int) -> Chat:
|
||||||
"""Return the Chat instance with the given ID."""
|
"""Return the Chat instance with the given ID."""
|
||||||
return Chat(self, chat_id)
|
return Chat(self, chat_id)
|
||||||
|
|
||||||
def secure_join(self, qrdata: str) -> Chat:
|
def secure_join(self, qrdata: str) -> Chat:
|
||||||
"""Continue a Setup-Contact or Verified-Group-Invite protocol started on
|
"""Continue a Setup-Contact or Verified-Group-Invite protocol started on another device.
|
||||||
another device.
|
|
||||||
|
|
||||||
The function returns immediately and the handshake runs in background, sending
|
The function returns immediately and the handshake runs in background, sending
|
||||||
and receiving several messages.
|
and receiving several messages.
|
||||||
@@ -325,9 +399,10 @@ class Account:
|
|||||||
next_msg_ids = self._rpc.get_next_msgs(self.id)
|
next_msg_ids = self._rpc.get_next_msgs(self.id)
|
||||||
return [Message(self, msg_id) for msg_id in next_msg_ids]
|
return [Message(self, msg_id) for msg_id in next_msg_ids]
|
||||||
|
|
||||||
|
@futuremethod
|
||||||
def wait_next_messages(self) -> list[Message]:
|
def wait_next_messages(self) -> list[Message]:
|
||||||
"""Wait for new messages and return a list of them."""
|
"""Wait for new messages and return a list of them."""
|
||||||
next_msg_ids = self._rpc.wait_next_msgs(self.id)
|
next_msg_ids = yield self._rpc.wait_next_msgs.future(self.id)
|
||||||
return [Message(self, msg_id) for msg_id in next_msg_ids]
|
return [Message(self, msg_id) for msg_id in next_msg_ids]
|
||||||
|
|
||||||
def wait_for_incoming_msg_event(self):
|
def wait_for_incoming_msg_event(self):
|
||||||
@@ -345,22 +420,26 @@ class Account:
|
|||||||
def wait_for_incoming_msg(self):
|
def wait_for_incoming_msg(self):
|
||||||
"""Wait for incoming message and return it.
|
"""Wait for incoming message and return it.
|
||||||
|
|
||||||
Consumes all events before the next incoming message event."""
|
Consumes all events before the next incoming message event.
|
||||||
|
"""
|
||||||
return self.get_message_by_id(self.wait_for_incoming_msg_event().msg_id)
|
return self.get_message_by_id(self.wait_for_incoming_msg_event().msg_id)
|
||||||
|
|
||||||
def wait_for_securejoin_inviter_success(self):
|
def wait_for_securejoin_inviter_success(self):
|
||||||
|
"""Wait until SecureJoin process finishes successfully on the inviter side."""
|
||||||
while True:
|
while True:
|
||||||
event = self.wait_for_event()
|
event = self.wait_for_event()
|
||||||
if event["kind"] == "SecurejoinInviterProgress" and event["progress"] == 1000:
|
if event["kind"] == "SecurejoinInviterProgress" and event["progress"] == 1000:
|
||||||
break
|
break
|
||||||
|
|
||||||
def wait_for_securejoin_joiner_success(self):
|
def wait_for_securejoin_joiner_success(self):
|
||||||
|
"""Wait until SecureJoin process finishes successfully on the joiner side."""
|
||||||
while True:
|
while True:
|
||||||
event = self.wait_for_event()
|
event = self.wait_for_event()
|
||||||
if event["kind"] == "SecurejoinJoinerProgress" and event["progress"] == 1000:
|
if event["kind"] == "SecurejoinJoinerProgress" and event["progress"] == 1000:
|
||||||
break
|
break
|
||||||
|
|
||||||
def wait_for_reactions_changed(self):
|
def wait_for_reactions_changed(self):
|
||||||
|
"""Wait for reaction change event."""
|
||||||
return self.wait_for_event(EventType.REACTIONS_CHANGED)
|
return self.wait_for_event(EventType.REACTIONS_CHANGED)
|
||||||
|
|
||||||
def get_fresh_messages_in_arrival_order(self) -> list[Message]:
|
def get_fresh_messages_in_arrival_order(self) -> list[Message]:
|
||||||
@@ -394,3 +473,8 @@ class Account:
|
|||||||
def initiate_autocrypt_key_transfer(self) -> None:
|
def initiate_autocrypt_key_transfer(self) -> None:
|
||||||
"""Send Autocrypt Setup Message."""
|
"""Send Autocrypt Setup Message."""
|
||||||
return self._rpc.initiate_autocrypt_key_transfer(self.id)
|
return self._rpc.initiate_autocrypt_key_transfer(self.id)
|
||||||
|
|
||||||
|
def ice_servers(self) -> list:
|
||||||
|
"""Return ICE servers for WebRTC configuration."""
|
||||||
|
ice_servers_json = self._rpc.ice_servers(self.id)
|
||||||
|
return json.loads(ice_servers_json)
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
"""Chat module."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import calendar
|
import calendar
|
||||||
@@ -89,7 +91,8 @@ class Chat:
|
|||||||
def set_ephemeral_timer(self, timer: int) -> None:
|
def set_ephemeral_timer(self, timer: int) -> None:
|
||||||
"""Set ephemeral timer of this chat in seconds.
|
"""Set ephemeral timer of this chat in seconds.
|
||||||
|
|
||||||
0 means the timer is disabled, use 1 for immediate deletion."""
|
0 means the timer is disabled, use 1 for immediate deletion.
|
||||||
|
"""
|
||||||
self._rpc.set_chat_ephemeral_timer(self.account.id, self.id, timer)
|
self._rpc.set_chat_ephemeral_timer(self.account.id, self.id, timer)
|
||||||
|
|
||||||
def get_encryption_info(self) -> str:
|
def get_encryption_info(self) -> str:
|
||||||
@@ -165,6 +168,11 @@ class Chat:
|
|||||||
msg_id = self._rpc.send_sticker(self.account.id, self.id, path)
|
msg_id = self._rpc.send_sticker(self.account.id, self.id, path)
|
||||||
return Message(self.account, msg_id)
|
return Message(self.account, msg_id)
|
||||||
|
|
||||||
|
def resend_messages(self, messages: list[Message]) -> None:
|
||||||
|
"""Resend a list of messages to this chat."""
|
||||||
|
msg_ids = [msg.id for msg in messages]
|
||||||
|
self._rpc.resend_messages(self.account.id, msg_ids)
|
||||||
|
|
||||||
def forward_messages(self, messages: list[Message]) -> None:
|
def forward_messages(self, messages: list[Message]) -> None:
|
||||||
"""Forward a list of messages to this chat."""
|
"""Forward a list of messages to this chat."""
|
||||||
msg_ids = [msg.id for msg in messages]
|
msg_ids = [msg.id for msg in messages]
|
||||||
@@ -199,12 +207,12 @@ class Chat:
|
|||||||
return snapshot
|
return snapshot
|
||||||
|
|
||||||
def get_messages(self, info_only: bool = False, add_daymarker: bool = False) -> list[Message]:
|
def get_messages(self, info_only: bool = False, add_daymarker: bool = False) -> list[Message]:
|
||||||
"""get the list of messages in this chat."""
|
"""Get the list of messages in this chat."""
|
||||||
msgs = self._rpc.get_message_ids(self.account.id, self.id, info_only, add_daymarker)
|
msgs = self._rpc.get_message_ids(self.account.id, self.id, info_only, add_daymarker)
|
||||||
return [Message(self.account, msg_id) for msg_id in msgs]
|
return [Message(self.account, msg_id) for msg_id in msgs]
|
||||||
|
|
||||||
def get_fresh_message_count(self) -> int:
|
def get_fresh_message_count(self) -> int:
|
||||||
"""Get number of fresh messages in this chat"""
|
"""Get number of fresh messages in this chat."""
|
||||||
return self._rpc.get_fresh_msg_cnt(self.account.id, self.id)
|
return self._rpc.get_fresh_msg_cnt(self.account.id, self.id)
|
||||||
|
|
||||||
def mark_noticed(self) -> None:
|
def mark_noticed(self) -> None:
|
||||||
@@ -286,3 +294,8 @@ class Chat:
|
|||||||
f.write(vcard.encode())
|
f.write(vcard.encode())
|
||||||
f.flush()
|
f.flush()
|
||||||
self._rpc.send_msg(self.account.id, self.id, {"viewtype": ViewType.VCARD, "file": f.name})
|
self._rpc.send_msg(self.account.id, self.id, {"viewtype": ViewType.VCARD, "file": f.name})
|
||||||
|
|
||||||
|
def place_outgoing_call(self, place_call_info: str) -> Message:
|
||||||
|
"""Starts an outgoing call."""
|
||||||
|
msg_id = self._rpc.place_outgoing_call(self.account.id, self.id, place_call_info)
|
||||||
|
return Message(self.account, msg_id)
|
||||||
|
|||||||
@@ -48,6 +48,7 @@ class Client:
|
|||||||
self.add_hooks(hooks or [])
|
self.add_hooks(hooks or [])
|
||||||
|
|
||||||
def add_hooks(self, hooks: Iterable[tuple[Callable, Union[type, EventFilter]]]) -> None:
|
def add_hooks(self, hooks: Iterable[tuple[Callable, Union[type, EventFilter]]]) -> None:
|
||||||
|
"""Register multiple hooks."""
|
||||||
for hook, event in hooks:
|
for hook, event in hooks:
|
||||||
self.add_hook(hook, event)
|
self.add_hook(hook, event)
|
||||||
|
|
||||||
@@ -77,14 +78,15 @@ class Client:
|
|||||||
self._hooks.get(type(event), set()).remove((hook, event))
|
self._hooks.get(type(event), set()).remove((hook, event))
|
||||||
|
|
||||||
def is_configured(self) -> bool:
|
def is_configured(self) -> bool:
|
||||||
|
"""Return True if the client is configured."""
|
||||||
return self.account.is_configured()
|
return self.account.is_configured()
|
||||||
|
|
||||||
def configure(self, email: str, password: str, **kwargs) -> None:
|
def configure(self, email: str, password: str, **kwargs) -> None:
|
||||||
self.account.set_config("addr", email)
|
"""Configure the client."""
|
||||||
self.account.set_config("mail_pw", password)
|
|
||||||
for key, value in kwargs.items():
|
for key, value in kwargs.items():
|
||||||
self.account.set_config(key, value)
|
self.account.set_config(key, value)
|
||||||
self.account.configure()
|
params = {"addr": email, "password": password}
|
||||||
|
self.account.add_or_update_transport(params)
|
||||||
self.logger.debug("Account configured")
|
self.logger.debug("Account configured")
|
||||||
|
|
||||||
def run_forever(self) -> None:
|
def run_forever(self) -> None:
|
||||||
@@ -198,5 +200,6 @@ class Bot(Client):
|
|||||||
"""Simple bot implementation that listens to events of a single account."""
|
"""Simple bot implementation that listens to events of a single account."""
|
||||||
|
|
||||||
def configure(self, email: str, password: str, **kwargs) -> None:
|
def configure(self, email: str, password: str, **kwargs) -> None:
|
||||||
|
"""Configure the bot."""
|
||||||
kwargs.setdefault("bot", "1")
|
kwargs.setdefault("bot", "1")
|
||||||
super().configure(email, password, **kwargs)
|
super().configure(email, password, **kwargs)
|
||||||
|
|||||||
@@ -1,14 +1,20 @@
|
|||||||
|
"""Constants module."""
|
||||||
|
|
||||||
from enum import Enum, IntEnum
|
from enum import Enum, IntEnum
|
||||||
|
|
||||||
COMMAND_PREFIX = "/"
|
COMMAND_PREFIX = "/"
|
||||||
|
|
||||||
|
|
||||||
class ContactFlag(IntEnum):
|
class ContactFlag(IntEnum):
|
||||||
VERIFIED_ONLY = 0x01
|
"""Bit flags for get_contacts() method."""
|
||||||
|
|
||||||
ADD_SELF = 0x02
|
ADD_SELF = 0x02
|
||||||
|
ADDRESS = 0x04
|
||||||
|
|
||||||
|
|
||||||
class ChatlistFlag(IntEnum):
|
class ChatlistFlag(IntEnum):
|
||||||
|
"""Bit flags for get_chatlist() method."""
|
||||||
|
|
||||||
ARCHIVED_ONLY = 0x01
|
ARCHIVED_ONLY = 0x01
|
||||||
NO_SPECIALS = 0x02
|
NO_SPECIALS = 0x02
|
||||||
ADD_ALLDONE_HINT = 0x04
|
ADD_ALLDONE_HINT = 0x04
|
||||||
@@ -16,6 +22,8 @@ class ChatlistFlag(IntEnum):
|
|||||||
|
|
||||||
|
|
||||||
class SpecialContactId(IntEnum):
|
class SpecialContactId(IntEnum):
|
||||||
|
"""Special contact IDs."""
|
||||||
|
|
||||||
SELF = 1
|
SELF = 1
|
||||||
INFO = 2 # centered messages as "member added", used in all chats
|
INFO = 2 # centered messages as "member added", used in all chats
|
||||||
DEVICE = 5 # messages "update info" in the device-chat
|
DEVICE = 5 # messages "update info" in the device-chat
|
||||||
@@ -23,7 +31,7 @@ class SpecialContactId(IntEnum):
|
|||||||
|
|
||||||
|
|
||||||
class EventType(str, Enum):
|
class EventType(str, Enum):
|
||||||
"""Core event types"""
|
"""Core event types."""
|
||||||
|
|
||||||
INFO = "Info"
|
INFO = "Info"
|
||||||
SMTP_CONNECTED = "SmtpConnected"
|
SMTP_CONNECTED = "SmtpConnected"
|
||||||
@@ -48,6 +56,7 @@ class EventType(str, Enum):
|
|||||||
MSG_READ = "MsgRead"
|
MSG_READ = "MsgRead"
|
||||||
MSG_DELETED = "MsgDeleted"
|
MSG_DELETED = "MsgDeleted"
|
||||||
CHAT_MODIFIED = "ChatModified"
|
CHAT_MODIFIED = "ChatModified"
|
||||||
|
CHAT_DELETED = "ChatDeleted"
|
||||||
CHAT_EPHEMERAL_TIMER_MODIFIED = "ChatEphemeralTimerModified"
|
CHAT_EPHEMERAL_TIMER_MODIFIED = "ChatEphemeralTimerModified"
|
||||||
CONTACTS_CHANGED = "ContactsChanged"
|
CONTACTS_CHANGED = "ContactsChanged"
|
||||||
LOCATION_CHANGED = "LocationChanged"
|
LOCATION_CHANGED = "LocationChanged"
|
||||||
@@ -64,13 +73,17 @@ class EventType(str, Enum):
|
|||||||
CHATLIST_ITEM_CHANGED = "ChatlistItemChanged"
|
CHATLIST_ITEM_CHANGED = "ChatlistItemChanged"
|
||||||
ACCOUNTS_CHANGED = "AccountsChanged"
|
ACCOUNTS_CHANGED = "AccountsChanged"
|
||||||
ACCOUNTS_ITEM_CHANGED = "AccountsItemChanged"
|
ACCOUNTS_ITEM_CHANGED = "AccountsItemChanged"
|
||||||
|
INCOMING_CALL = "IncomingCall"
|
||||||
|
INCOMING_CALL_ACCEPTED = "IncomingCallAccepted"
|
||||||
|
OUTGOING_CALL_ACCEPTED = "OutgoingCallAccepted"
|
||||||
|
CALL_ENDED = "CallEnded"
|
||||||
CONFIG_SYNCED = "ConfigSynced"
|
CONFIG_SYNCED = "ConfigSynced"
|
||||||
WEBXDC_REALTIME_DATA = "WebxdcRealtimeData"
|
WEBXDC_REALTIME_DATA = "WebxdcRealtimeData"
|
||||||
WEBXDC_REALTIME_ADVERTISEMENT_RECEIVED = "WebxdcRealtimeAdvertisementReceived"
|
WEBXDC_REALTIME_ADVERTISEMENT_RECEIVED = "WebxdcRealtimeAdvertisementReceived"
|
||||||
|
|
||||||
|
|
||||||
class ChatId(IntEnum):
|
class ChatId(IntEnum):
|
||||||
"""Special chat ids"""
|
"""Special chat IDs."""
|
||||||
|
|
||||||
TRASH = 3
|
TRASH = 3
|
||||||
ARCHIVED_LINK = 6
|
ARCHIVED_LINK = 6
|
||||||
@@ -78,18 +91,46 @@ class ChatId(IntEnum):
|
|||||||
LAST_SPECIAL = 9
|
LAST_SPECIAL = 9
|
||||||
|
|
||||||
|
|
||||||
class ChatType(IntEnum):
|
class ChatType(str, Enum):
|
||||||
"""Chat types"""
|
"""Chat type."""
|
||||||
|
|
||||||
UNDEFINED = 0
|
SINGLE = "Single"
|
||||||
SINGLE = 100
|
"""1:1 chat, i.e. a direct chat with a single contact"""
|
||||||
GROUP = 120
|
|
||||||
MAILINGLIST = 140
|
GROUP = "Group"
|
||||||
BROADCAST = 160
|
|
||||||
|
MAILINGLIST = "Mailinglist"
|
||||||
|
|
||||||
|
OUT_BROADCAST = "OutBroadcast"
|
||||||
|
"""Outgoing broadcast channel, called "Channel" in the UI.
|
||||||
|
|
||||||
|
The user can send into this channel,
|
||||||
|
and all recipients will receive messages
|
||||||
|
in an `IN_BROADCAST`.
|
||||||
|
|
||||||
|
Called `broadcast` here rather than `channel`,
|
||||||
|
because the word "channel" already appears a lot in the code,
|
||||||
|
which would make it hard to grep for it.
|
||||||
|
"""
|
||||||
|
|
||||||
|
IN_BROADCAST = "InBroadcast"
|
||||||
|
"""Incoming broadcast channel, called "Channel" in the UI.
|
||||||
|
|
||||||
|
This channel is read-only,
|
||||||
|
and we do not know who the other recipients are.
|
||||||
|
|
||||||
|
This is similar to a `MAILINGLIST`,
|
||||||
|
with the main difference being that
|
||||||
|
`IN_BROADCAST`s are encrypted.
|
||||||
|
|
||||||
|
Called `broadcast` here rather than `channel`,
|
||||||
|
because the word "channel" already appears a lot in the code,
|
||||||
|
which would make it hard to grep for it.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
class ChatVisibility(str, Enum):
|
class ChatVisibility(str, Enum):
|
||||||
"""Chat visibility types"""
|
"""Chat visibility types."""
|
||||||
|
|
||||||
NORMAL = "Normal"
|
NORMAL = "Normal"
|
||||||
ARCHIVED = "Archived"
|
ARCHIVED = "Archived"
|
||||||
@@ -97,7 +138,7 @@ class ChatVisibility(str, Enum):
|
|||||||
|
|
||||||
|
|
||||||
class DownloadState(str, Enum):
|
class DownloadState(str, Enum):
|
||||||
"""Message download state"""
|
"""Message download state."""
|
||||||
|
|
||||||
DONE = "Done"
|
DONE = "Done"
|
||||||
AVAILABLE = "Available"
|
AVAILABLE = "Available"
|
||||||
@@ -117,7 +158,6 @@ class ViewType(str, Enum):
|
|||||||
VOICE = "Voice"
|
VOICE = "Voice"
|
||||||
VIDEO = "Video"
|
VIDEO = "Video"
|
||||||
FILE = "File"
|
FILE = "File"
|
||||||
VIDEOCHAT_INVITATION = "VideochatInvitation"
|
|
||||||
WEBXDC = "Webxdc"
|
WEBXDC = "Webxdc"
|
||||||
VCARD = "Vcard"
|
VCARD = "Vcard"
|
||||||
|
|
||||||
@@ -158,14 +198,14 @@ class MessageState(IntEnum):
|
|||||||
|
|
||||||
|
|
||||||
class MessageId(IntEnum):
|
class MessageId(IntEnum):
|
||||||
"""Special message ids"""
|
"""Special message IDs."""
|
||||||
|
|
||||||
DAYMARKER = 9
|
DAYMARKER = 9
|
||||||
LAST_SPECIAL = 9
|
LAST_SPECIAL = 9
|
||||||
|
|
||||||
|
|
||||||
class CertificateChecks(IntEnum):
|
class CertificateChecks(IntEnum):
|
||||||
"""Certificate checks mode"""
|
"""Certificate checks mode."""
|
||||||
|
|
||||||
AUTOMATIC = 0
|
AUTOMATIC = 0
|
||||||
STRICT = 1
|
STRICT = 1
|
||||||
@@ -173,7 +213,7 @@ class CertificateChecks(IntEnum):
|
|||||||
|
|
||||||
|
|
||||||
class Connectivity(IntEnum):
|
class Connectivity(IntEnum):
|
||||||
"""Connectivity states"""
|
"""Connectivity states."""
|
||||||
|
|
||||||
NOT_CONNECTED = 1000
|
NOT_CONNECTED = 1000
|
||||||
CONNECTING = 2000
|
CONNECTING = 2000
|
||||||
@@ -182,7 +222,7 @@ class Connectivity(IntEnum):
|
|||||||
|
|
||||||
|
|
||||||
class KeyGenType(IntEnum):
|
class KeyGenType(IntEnum):
|
||||||
"""Type of the key to generate"""
|
"""Type of the key to generate."""
|
||||||
|
|
||||||
DEFAULT = 0
|
DEFAULT = 0
|
||||||
RSA2048 = 1
|
RSA2048 = 1
|
||||||
@@ -192,21 +232,21 @@ class KeyGenType(IntEnum):
|
|||||||
|
|
||||||
# "Lp" means "login parameters"
|
# "Lp" means "login parameters"
|
||||||
class LpAuthFlag(IntEnum):
|
class LpAuthFlag(IntEnum):
|
||||||
"""Authorization flags"""
|
"""Authorization flags."""
|
||||||
|
|
||||||
OAUTH2 = 0x2
|
OAUTH2 = 0x2
|
||||||
NORMAL = 0x4
|
NORMAL = 0x4
|
||||||
|
|
||||||
|
|
||||||
class MediaQuality(IntEnum):
|
class MediaQuality(IntEnum):
|
||||||
"""Media quality setting"""
|
"""Media quality setting."""
|
||||||
|
|
||||||
BALANCED = 0
|
BALANCED = 0
|
||||||
WORSE = 1
|
WORSE = 1
|
||||||
|
|
||||||
|
|
||||||
class ProviderStatus(IntEnum):
|
class ProviderStatus(IntEnum):
|
||||||
"""Provider status according to manual testing"""
|
"""Provider status according to manual testing."""
|
||||||
|
|
||||||
OK = 1
|
OK = 1
|
||||||
PREPARATION = 2
|
PREPARATION = 2
|
||||||
@@ -214,7 +254,7 @@ class ProviderStatus(IntEnum):
|
|||||||
|
|
||||||
|
|
||||||
class PushNotifyState(IntEnum):
|
class PushNotifyState(IntEnum):
|
||||||
"""Push notifications state"""
|
"""Push notifications state."""
|
||||||
|
|
||||||
NOT_CONNECTED = 0
|
NOT_CONNECTED = 0
|
||||||
HEARTBEAT = 1
|
HEARTBEAT = 1
|
||||||
@@ -222,7 +262,7 @@ class PushNotifyState(IntEnum):
|
|||||||
|
|
||||||
|
|
||||||
class ShowEmails(IntEnum):
|
class ShowEmails(IntEnum):
|
||||||
"""Show emails mode"""
|
"""Show emails mode."""
|
||||||
|
|
||||||
OFF = 0
|
OFF = 0
|
||||||
ACCEPTED_CONTACTS = 1
|
ACCEPTED_CONTACTS = 1
|
||||||
@@ -230,17 +270,9 @@ class ShowEmails(IntEnum):
|
|||||||
|
|
||||||
|
|
||||||
class SocketSecurity(IntEnum):
|
class SocketSecurity(IntEnum):
|
||||||
"""Socket security"""
|
"""Socket security."""
|
||||||
|
|
||||||
AUTOMATIC = 0
|
AUTOMATIC = 0
|
||||||
SSL = 1
|
SSL = 1
|
||||||
STARTTLS = 2
|
STARTTLS = 2
|
||||||
PLAIN = 3
|
PLAIN = 3
|
||||||
|
|
||||||
|
|
||||||
class VideochatType(IntEnum):
|
|
||||||
"""Video chat URL type"""
|
|
||||||
|
|
||||||
UNKNOWN = 0
|
|
||||||
BASICWEBRTC = 1
|
|
||||||
JITSI = 2
|
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
"""Contact module."""
|
||||||
|
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
@@ -11,8 +13,7 @@ if TYPE_CHECKING:
|
|||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class Contact:
|
class Contact:
|
||||||
"""
|
"""Contact API.
|
||||||
Contact API.
|
|
||||||
|
|
||||||
Essentially a wrapper for RPC, account ID and a contact ID.
|
Essentially a wrapper for RPC, account ID and a contact ID.
|
||||||
"""
|
"""
|
||||||
@@ -36,17 +37,14 @@ class Contact:
|
|||||||
"""Delete contact."""
|
"""Delete contact."""
|
||||||
self._rpc.delete_contact(self.account.id, self.id)
|
self._rpc.delete_contact(self.account.id, self.id)
|
||||||
|
|
||||||
def reset_encryption(self) -> None:
|
|
||||||
"""Reset contact encryption."""
|
|
||||||
self._rpc.reset_contact_encryption(self.account.id, self.id)
|
|
||||||
|
|
||||||
def set_name(self, name: str) -> None:
|
def set_name(self, name: str) -> None:
|
||||||
"""Change the name of this contact."""
|
"""Change the name of this contact."""
|
||||||
self._rpc.change_contact_name(self.account.id, self.id, name)
|
self._rpc.change_contact_name(self.account.id, self.id, name)
|
||||||
|
|
||||||
def get_encryption_info(self) -> str:
|
def get_encryption_info(self) -> str:
|
||||||
"""Get a multi-line encryption info, containing your fingerprint and
|
"""Get a multi-line encryption info.
|
||||||
the fingerprint of the contact.
|
|
||||||
|
Encryption info contains your fingerprint and the fingerprint of the contact.
|
||||||
"""
|
"""
|
||||||
return self._rpc.get_contact_encryption_info(self.account.id, self.id)
|
return self._rpc.get_contact_encryption_info(self.account.id, self.id)
|
||||||
|
|
||||||
@@ -66,4 +64,5 @@ class Contact:
|
|||||||
)
|
)
|
||||||
|
|
||||||
def make_vcard(self) -> str:
|
def make_vcard(self) -> str:
|
||||||
|
"""Make a vCard for the contact."""
|
||||||
return self.account.make_vcard([self])
|
return self.account.make_vcard([self])
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
"""Account manager module."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING
|
||||||
@@ -10,12 +12,13 @@ if TYPE_CHECKING:
|
|||||||
|
|
||||||
|
|
||||||
class DeltaChat:
|
class DeltaChat:
|
||||||
"""
|
"""Delta Chat accounts manager.
|
||||||
Delta Chat accounts manager.
|
|
||||||
This is the root of the object oriented API.
|
This is the root of the object oriented API.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, rpc: "Rpc") -> None:
|
def __init__(self, rpc: "Rpc") -> None:
|
||||||
|
"""Initialize account manager."""
|
||||||
self.rpc = rpc
|
self.rpc = rpc
|
||||||
|
|
||||||
def add_account(self) -> Account:
|
def add_account(self) -> Account:
|
||||||
@@ -37,9 +40,7 @@ class DeltaChat:
|
|||||||
self.rpc.stop_io_for_all_accounts()
|
self.rpc.stop_io_for_all_accounts()
|
||||||
|
|
||||||
def maybe_network(self) -> None:
|
def maybe_network(self) -> None:
|
||||||
"""Indicate that the network likely has come back or just that the network
|
"""Indicate that the network conditions might have changed."""
|
||||||
conditions might have changed.
|
|
||||||
"""
|
|
||||||
self.rpc.maybe_network()
|
self.rpc.maybe_network()
|
||||||
|
|
||||||
def get_system_info(self) -> AttrDict:
|
def get_system_info(self) -> AttrDict:
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ class EventFilter(ABC):
|
|||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def __hash__(self) -> int:
|
def __hash__(self) -> int:
|
||||||
"""Object's unique hash"""
|
"""Object's unique hash."""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def __eq__(self, other) -> bool:
|
def __eq__(self, other) -> bool:
|
||||||
@@ -52,9 +52,7 @@ class EventFilter(ABC):
|
|||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def filter(self, event):
|
def filter(self, event):
|
||||||
"""Return True-like value if the event passed the filter and should be
|
"""Return True-like value if the event passed the filter."""
|
||||||
used, or False-like value otherwise.
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class RawEvent(EventFilter):
|
class RawEvent(EventFilter):
|
||||||
@@ -82,31 +80,17 @@ class RawEvent(EventFilter):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
def filter(self, event: "AttrDict") -> bool:
|
def filter(self, event: "AttrDict") -> bool:
|
||||||
|
"""Filter an event.
|
||||||
|
|
||||||
|
Return true if the event should be processed.
|
||||||
|
"""
|
||||||
if self.types and event.kind not in self.types:
|
if self.types and event.kind not in self.types:
|
||||||
return False
|
return False
|
||||||
return self._call_func(event)
|
return self._call_func(event)
|
||||||
|
|
||||||
|
|
||||||
class NewMessage(EventFilter):
|
class NewMessage(EventFilter):
|
||||||
"""Matches whenever a new message arrives.
|
"""Matches whenever a new message arrives."""
|
||||||
|
|
||||||
Warning: registering a handler for this event will cause the messages
|
|
||||||
to be marked as read. Its usage is mainly intended for bots.
|
|
||||||
|
|
||||||
:param pattern: if set, this Pattern will be used to filter the message by its text
|
|
||||||
content.
|
|
||||||
:param command: If set, only match messages with the given command (ex. /help).
|
|
||||||
Setting this property implies `is_info==False`.
|
|
||||||
:param is_bot: If set to True only match messages sent by bots, if set to None
|
|
||||||
match messages from bots and users. If omitted or set to False
|
|
||||||
only messages from users will be matched.
|
|
||||||
:param is_info: If set to True only match info/system messages, if set to False
|
|
||||||
only match messages that are not info/system messages. If omitted
|
|
||||||
info/system messages as well as normal messages will be matched.
|
|
||||||
:param func: A Callable function that should accept the event as input
|
|
||||||
parameter, and return a bool value indicating whether the event
|
|
||||||
should be dispatched or not.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self,
|
||||||
@@ -121,6 +105,25 @@ class NewMessage(EventFilter):
|
|||||||
is_info: Optional[bool] = None,
|
is_info: Optional[bool] = None,
|
||||||
func: Optional[Callable[["AttrDict"], bool]] = None,
|
func: Optional[Callable[["AttrDict"], bool]] = None,
|
||||||
) -> None:
|
) -> None:
|
||||||
|
"""Initialize a new message filter.
|
||||||
|
|
||||||
|
Warning: registering a handler for this event will cause the messages
|
||||||
|
to be marked as read. Its usage is mainly intended for bots.
|
||||||
|
|
||||||
|
:param pattern: if set, this Pattern will be used to filter the message by its text
|
||||||
|
content.
|
||||||
|
:param command: If set, only match messages with the given command (ex. /help).
|
||||||
|
Setting this property implies `is_info==False`.
|
||||||
|
:param is_bot: If set to True only match messages sent by bots, if set to None
|
||||||
|
match messages from bots and users. If omitted or set to False
|
||||||
|
only messages from users will be matched.
|
||||||
|
:param is_info: If set to True only match info/system messages, if set to False
|
||||||
|
only match messages that are not info/system messages. If omitted
|
||||||
|
info/system messages as well as normal messages will be matched.
|
||||||
|
:param func: A Callable function that should accept the event as input
|
||||||
|
parameter, and return a bool value indicating whether the event
|
||||||
|
should be dispatched or not.
|
||||||
|
"""
|
||||||
super().__init__(func=func)
|
super().__init__(func=func)
|
||||||
self.is_bot = is_bot
|
self.is_bot = is_bot
|
||||||
self.is_info = is_info
|
self.is_info = is_info
|
||||||
@@ -159,6 +162,7 @@ class NewMessage(EventFilter):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
def filter(self, event: "AttrDict") -> bool:
|
def filter(self, event: "AttrDict") -> bool:
|
||||||
|
"""Return true if if the event is a new message event."""
|
||||||
if self.is_bot is not None and self.is_bot != event.message_snapshot.is_bot:
|
if self.is_bot is not None and self.is_bot != event.message_snapshot.is_bot:
|
||||||
return False
|
return False
|
||||||
if self.is_info is not None and self.is_info != event.message_snapshot.is_info:
|
if self.is_info is not None and self.is_info != event.message_snapshot.is_info:
|
||||||
@@ -199,6 +203,7 @@ class MemberListChanged(EventFilter):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
def filter(self, event: "AttrDict") -> bool:
|
def filter(self, event: "AttrDict") -> bool:
|
||||||
|
"""Return true if if the event is a member addition event."""
|
||||||
if self.added is not None and self.added != event.member_added:
|
if self.added is not None and self.added != event.member_added:
|
||||||
return False
|
return False
|
||||||
return self._call_func(event)
|
return self._call_func(event)
|
||||||
@@ -231,6 +236,7 @@ class GroupImageChanged(EventFilter):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
def filter(self, event: "AttrDict") -> bool:
|
def filter(self, event: "AttrDict") -> bool:
|
||||||
|
"""Return True if event is matched."""
|
||||||
if self.deleted is not None and self.deleted != event.image_deleted:
|
if self.deleted is not None and self.deleted != event.image_deleted:
|
||||||
return False
|
return False
|
||||||
return self._call_func(event)
|
return self._call_func(event)
|
||||||
@@ -256,13 +262,12 @@ class GroupNameChanged(EventFilter):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
def filter(self, event: "AttrDict") -> bool:
|
def filter(self, event: "AttrDict") -> bool:
|
||||||
|
"""Return True if event is matched."""
|
||||||
return self._call_func(event)
|
return self._call_func(event)
|
||||||
|
|
||||||
|
|
||||||
class HookCollection:
|
class HookCollection:
|
||||||
"""
|
"""Helper class to collect event hooks that can later be added to a Delta Chat client."""
|
||||||
Helper class to collect event hooks that can later be added to a Delta Chat client.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self) -> None:
|
def __init__(self) -> None:
|
||||||
self._hooks: set[tuple[Callable, Union[type, EventFilter]]] = set()
|
self._hooks: set[tuple[Callable, Union[type, EventFilter]]] = set()
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
|
"""Message module."""
|
||||||
|
|
||||||
import json
|
import json
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from typing import TYPE_CHECKING, Optional, Union
|
from typing import TYPE_CHECKING, List, Optional, Union
|
||||||
|
|
||||||
from ._utils import AttrDict, futuremethod
|
from ._utils import AttrDict, futuremethod
|
||||||
from .const import EventType
|
from .const import EventType
|
||||||
@@ -37,6 +39,11 @@ class Message:
|
|||||||
snapshot["message"] = self
|
snapshot["message"] = self
|
||||||
return snapshot
|
return snapshot
|
||||||
|
|
||||||
|
def get_read_receipts(self) -> List[AttrDict]:
|
||||||
|
"""Get message read receipts."""
|
||||||
|
read_receipts = self._rpc.get_message_read_receipts(self.account.id, self.id)
|
||||||
|
return [AttrDict(read_receipt) for read_receipt in read_receipts]
|
||||||
|
|
||||||
def get_reactions(self) -> Optional[AttrDict]:
|
def get_reactions(self) -> Optional[AttrDict]:
|
||||||
"""Get message reactions."""
|
"""Get message reactions."""
|
||||||
reactions = self._rpc.get_message_reactions(self.account.id, self.id)
|
reactions = self._rpc.get_message_reactions(self.account.id, self.id)
|
||||||
@@ -45,6 +52,7 @@ class Message:
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
def get_sender_contact(self) -> Contact:
|
def get_sender_contact(self) -> Contact:
|
||||||
|
"""Return sender contact."""
|
||||||
from_id = self.get_snapshot().from_id
|
from_id = self.get_snapshot().from_id
|
||||||
return self.account.get_contact_by_id(from_id)
|
return self.account.get_contact_by_id(from_id)
|
||||||
|
|
||||||
@@ -53,6 +61,11 @@ class Message:
|
|||||||
self._rpc.markseen_msgs(self.account.id, [self.id])
|
self._rpc.markseen_msgs(self.account.id, [self.id])
|
||||||
|
|
||||||
def continue_autocrypt_key_transfer(self, setup_code: str) -> None:
|
def continue_autocrypt_key_transfer(self, setup_code: str) -> None:
|
||||||
|
"""Continue the Autocrypt Setup Message key transfer.
|
||||||
|
|
||||||
|
This function can be called on received Autocrypt Setup Message
|
||||||
|
to import the key encrypted with the provided setup code.
|
||||||
|
"""
|
||||||
self._rpc.continue_autocrypt_key_transfer(self.account.id, self.id, setup_code)
|
self._rpc.continue_autocrypt_key_transfer(self.account.id, self.id, setup_code)
|
||||||
|
|
||||||
def send_webxdc_status_update(self, update: Union[dict, str], description: str) -> None:
|
def send_webxdc_status_update(self, update: Union[dict, str], description: str) -> None:
|
||||||
@@ -62,9 +75,15 @@ class Message:
|
|||||||
self._rpc.send_webxdc_status_update(self.account.id, self.id, update, description)
|
self._rpc.send_webxdc_status_update(self.account.id, self.id, update, description)
|
||||||
|
|
||||||
def get_webxdc_status_updates(self, last_known_serial: int = 0) -> list:
|
def get_webxdc_status_updates(self, last_known_serial: int = 0) -> list:
|
||||||
|
"""Return a list of Webxdc status updates for Webxdc instance message."""
|
||||||
return json.loads(self._rpc.get_webxdc_status_updates(self.account.id, self.id, last_known_serial))
|
return json.loads(self._rpc.get_webxdc_status_updates(self.account.id, self.id, last_known_serial))
|
||||||
|
|
||||||
|
def get_info(self) -> str:
|
||||||
|
"""Return message info."""
|
||||||
|
return self._rpc.get_message_info(self.account.id, self.id)
|
||||||
|
|
||||||
def get_webxdc_info(self) -> dict:
|
def get_webxdc_info(self) -> dict:
|
||||||
|
"""Get info from a Webxdc message in JSON format."""
|
||||||
return self._rpc.get_webxdc_info(self.account.id, self.id)
|
return self._rpc.get_webxdc_info(self.account.id, self.id)
|
||||||
|
|
||||||
def wait_until_delivered(self) -> None:
|
def wait_until_delivered(self) -> None:
|
||||||
@@ -74,10 +93,35 @@ class Message:
|
|||||||
if event.kind == EventType.MSG_DELIVERED and event.msg_id == self.id:
|
if event.kind == EventType.MSG_DELIVERED and event.msg_id == self.id:
|
||||||
break
|
break
|
||||||
|
|
||||||
|
def resend(self) -> None:
|
||||||
|
"""Resend messages and make information available for newly added chat members.
|
||||||
|
Resending sends out the original message, however, recipients and webxdc-status may differ.
|
||||||
|
Clients that already have the original message can still ignore the resent message as
|
||||||
|
they have tracked the state by dedicated updates.
|
||||||
|
|
||||||
|
Some messages cannot be resent, eg. info-messages, drafts, already pending messages,
|
||||||
|
or messages that are not sent by SELF.
|
||||||
|
"""
|
||||||
|
self._rpc.resend_messages(self.account.id, [self.id])
|
||||||
|
|
||||||
@futuremethod
|
@futuremethod
|
||||||
def send_webxdc_realtime_advertisement(self):
|
def send_webxdc_realtime_advertisement(self):
|
||||||
|
"""Send an advertisement to join the realtime channel."""
|
||||||
yield self._rpc.send_webxdc_realtime_advertisement.future(self.account.id, self.id)
|
yield self._rpc.send_webxdc_realtime_advertisement.future(self.account.id, self.id)
|
||||||
|
|
||||||
@futuremethod
|
@futuremethod
|
||||||
def send_webxdc_realtime_data(self, data) -> None:
|
def send_webxdc_realtime_data(self, data) -> None:
|
||||||
|
"""Send data to the realtime channel."""
|
||||||
yield self._rpc.send_webxdc_realtime_data.future(self.account.id, self.id, list(data))
|
yield self._rpc.send_webxdc_realtime_data.future(self.account.id, self.id, list(data))
|
||||||
|
|
||||||
|
def accept_incoming_call(self, accept_call_info):
|
||||||
|
"""Accepts an incoming call."""
|
||||||
|
self._rpc.accept_incoming_call(self.account.id, self.id, accept_call_info)
|
||||||
|
|
||||||
|
def end_call(self):
|
||||||
|
"""Ends incoming or outgoing call."""
|
||||||
|
self._rpc.end_call(self.account.id, self.id)
|
||||||
|
|
||||||
|
def get_call_info(self) -> AttrDict:
|
||||||
|
"""Return information about the call."""
|
||||||
|
return AttrDict(self._rpc.call_info(self.account.id, self.id))
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
"""Pytest plugin module."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import os
|
import os
|
||||||
@@ -11,70 +13,77 @@ from . import Account, AttrDict, Bot, Chat, Client, DeltaChat, EventType, Messag
|
|||||||
from ._utils import futuremethod
|
from ._utils import futuremethod
|
||||||
from .rpc import Rpc
|
from .rpc import Rpc
|
||||||
|
|
||||||
|
E2EE_INFO_MSGS = 1
|
||||||
def get_temp_credentials() -> dict:
|
"""
|
||||||
domain = os.getenv("CHATMAIL_DOMAIN")
|
The number of info messages added to new e2ee chats.
|
||||||
username = "ci-" + "".join(random.choice("2345789acdefghjkmnpqrstuvwxyz") for i in range(6))
|
Currently this is "End-to-end encryption available".
|
||||||
password = f"{username}${username}"
|
"""
|
||||||
addr = f"{username}@{domain}"
|
|
||||||
return {"email": addr, "password": password}
|
|
||||||
|
|
||||||
|
|
||||||
class ACFactory:
|
class ACFactory:
|
||||||
|
"""Test account factory."""
|
||||||
|
|
||||||
def __init__(self, deltachat: DeltaChat) -> None:
|
def __init__(self, deltachat: DeltaChat) -> None:
|
||||||
self.deltachat = deltachat
|
self.deltachat = deltachat
|
||||||
|
|
||||||
def get_unconfigured_account(self) -> Account:
|
def get_unconfigured_account(self) -> Account:
|
||||||
account = self.deltachat.add_account()
|
"""Create a new unconfigured account."""
|
||||||
account.set_config("verified_one_on_one_chats", "1")
|
return self.deltachat.add_account()
|
||||||
return account
|
|
||||||
|
|
||||||
def get_unconfigured_bot(self) -> Bot:
|
def get_unconfigured_bot(self) -> Bot:
|
||||||
|
"""Create a new unconfigured bot."""
|
||||||
return Bot(self.get_unconfigured_account())
|
return Bot(self.get_unconfigured_account())
|
||||||
|
|
||||||
def new_preconfigured_account(self) -> Account:
|
def get_credentials(self) -> (str, str):
|
||||||
"""Make a new account with configuration options set, but configuration not started."""
|
"""Generate new credentials for chatmail account."""
|
||||||
credentials = get_temp_credentials()
|
domain = os.getenv("CHATMAIL_DOMAIN")
|
||||||
account = self.get_unconfigured_account()
|
username = "ci-" + "".join(random.choice("2345789acdefghjkmnpqrstuvwxyz") for i in range(6))
|
||||||
account.set_config("addr", credentials["email"])
|
return f"{username}@{domain}", f"{username}${username}"
|
||||||
account.set_config("mail_pw", credentials["password"])
|
|
||||||
assert not account.is_configured()
|
|
||||||
return account
|
|
||||||
|
|
||||||
@futuremethod
|
@futuremethod
|
||||||
def new_configured_account(self):
|
def new_configured_account(self):
|
||||||
account = self.new_preconfigured_account()
|
"""Create a new configured account."""
|
||||||
yield account.configure.future()
|
account = self.get_unconfigured_account()
|
||||||
|
domain = os.getenv("CHATMAIL_DOMAIN")
|
||||||
|
yield account.add_transport_from_qr.future(f"dcaccount:{domain}")
|
||||||
|
|
||||||
assert account.is_configured()
|
assert account.is_configured()
|
||||||
return account
|
return account
|
||||||
|
|
||||||
def new_configured_bot(self) -> Bot:
|
def new_configured_bot(self) -> Bot:
|
||||||
credentials = get_temp_credentials()
|
"""Create a new configured bot."""
|
||||||
|
addr, password = self.get_credentials()
|
||||||
bot = self.get_unconfigured_bot()
|
bot = self.get_unconfigured_bot()
|
||||||
bot.configure(credentials["email"], credentials["password"])
|
bot.configure(addr, password)
|
||||||
return bot
|
return bot
|
||||||
|
|
||||||
@futuremethod
|
@futuremethod
|
||||||
def get_online_account(self):
|
def get_online_account(self):
|
||||||
|
"""Create a new account and start I/O."""
|
||||||
account = yield self.new_configured_account.future()
|
account = yield self.new_configured_account.future()
|
||||||
account.bring_online()
|
account.bring_online()
|
||||||
return account
|
return account
|
||||||
|
|
||||||
def get_online_accounts(self, num: int) -> list[Account]:
|
def get_online_accounts(self, num: int) -> list[Account]:
|
||||||
|
"""Create multiple online accounts."""
|
||||||
futures = [self.get_online_account.future() for _ in range(num)]
|
futures = [self.get_online_account.future() for _ in range(num)]
|
||||||
return [f() for f in futures]
|
return [f() for f in futures]
|
||||||
|
|
||||||
def resetup_account(self, ac: Account) -> Account:
|
def resetup_account(self, ac: Account) -> Account:
|
||||||
"""Resetup account from scratch, losing the encryption key."""
|
"""Resetup account from scratch, losing the encryption key."""
|
||||||
ac.stop_io()
|
ac.stop_io()
|
||||||
ac_clone = self.get_unconfigured_account()
|
transports = ac.list_transports()
|
||||||
for i in ["addr", "mail_pw"]:
|
|
||||||
ac_clone.set_config(i, ac.get_config(i))
|
|
||||||
ac.remove()
|
ac.remove()
|
||||||
ac_clone.configure()
|
ac_clone = self.get_unconfigured_account()
|
||||||
|
for transport in transports:
|
||||||
|
ac_clone.add_or_update_transport(transport)
|
||||||
return ac_clone
|
return ac_clone
|
||||||
|
|
||||||
def get_accepted_chat(self, ac1: Account, ac2: Account) -> Chat:
|
def get_accepted_chat(self, ac1: Account, ac2: Account) -> Chat:
|
||||||
|
"""Create a new 1:1 chat between ac1 and ac2 accepted on both sides.
|
||||||
|
|
||||||
|
Returned chat is a chat with ac2 from ac1 point of view.
|
||||||
|
"""
|
||||||
ac2.create_chat(ac1)
|
ac2.create_chat(ac1)
|
||||||
return ac1.create_chat(ac2)
|
return ac1.create_chat(ac2)
|
||||||
|
|
||||||
@@ -86,9 +95,10 @@ class ACFactory:
|
|||||||
file: Optional[str] = None,
|
file: Optional[str] = None,
|
||||||
group: Optional[str] = None,
|
group: Optional[str] = None,
|
||||||
) -> Message:
|
) -> Message:
|
||||||
|
"""Send a message."""
|
||||||
if not from_account:
|
if not from_account:
|
||||||
from_account = (self.get_online_accounts(1))[0]
|
from_account = (self.get_online_accounts(1))[0]
|
||||||
to_contact = from_account.create_contact(to_account.get_config("addr"))
|
to_contact = from_account.create_contact(to_account)
|
||||||
if group:
|
if group:
|
||||||
to_chat = from_account.create_group(group)
|
to_chat = from_account.create_group(group)
|
||||||
to_chat.add_contact(to_contact)
|
to_chat.add_contact(to_contact)
|
||||||
@@ -104,6 +114,7 @@ class ACFactory:
|
|||||||
file: Optional[str] = None,
|
file: Optional[str] = None,
|
||||||
group: Optional[str] = None,
|
group: Optional[str] = None,
|
||||||
) -> AttrDict:
|
) -> AttrDict:
|
||||||
|
"""Send a message and wait until recipient processes it."""
|
||||||
self.send_message(
|
self.send_message(
|
||||||
to_account=to_client.account,
|
to_account=to_client.account,
|
||||||
from_account=from_account,
|
from_account=from_account,
|
||||||
@@ -117,6 +128,7 @@ class ACFactory:
|
|||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def rpc(tmp_path) -> AsyncGenerator:
|
def rpc(tmp_path) -> AsyncGenerator:
|
||||||
|
"""RPC client fixture."""
|
||||||
rpc_server = Rpc(accounts_dir=str(tmp_path / "accounts"))
|
rpc_server = Rpc(accounts_dir=str(tmp_path / "accounts"))
|
||||||
with rpc_server:
|
with rpc_server:
|
||||||
yield rpc_server
|
yield rpc_server
|
||||||
@@ -124,6 +136,7 @@ def rpc(tmp_path) -> AsyncGenerator:
|
|||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def acfactory(rpc) -> AsyncGenerator:
|
def acfactory(rpc) -> AsyncGenerator:
|
||||||
|
"""Return account factory fixture."""
|
||||||
return ACFactory(DeltaChat(rpc))
|
return ACFactory(DeltaChat(rpc))
|
||||||
|
|
||||||
|
|
||||||
@@ -141,7 +154,7 @@ def data():
|
|||||||
raise Exception("Data path cannot be found")
|
raise Exception("Data path cannot be found")
|
||||||
|
|
||||||
def get_path(self, bn):
|
def get_path(self, bn):
|
||||||
"""return path of file or None if it doesn't exist."""
|
"""Return path of file or None if it doesn't exist."""
|
||||||
fn = os.path.join(self.path, *bn.split("/"))
|
fn = os.path.join(self.path, *bn.split("/"))
|
||||||
assert os.path.exists(fn)
|
assert os.path.exists(fn)
|
||||||
return fn
|
return fn
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
"""JSON-RPC client module."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import itertools
|
import itertools
|
||||||
@@ -12,16 +14,19 @@ from typing import Any, Iterator, Optional
|
|||||||
|
|
||||||
|
|
||||||
class JsonRpcError(Exception):
|
class JsonRpcError(Exception):
|
||||||
pass
|
"""JSON-RPC error."""
|
||||||
|
|
||||||
|
|
||||||
class RpcFuture:
|
class RpcFuture:
|
||||||
|
"""RPC future waiting for RPC call result."""
|
||||||
|
|
||||||
def __init__(self, rpc: "Rpc", request_id: int, event: Event):
|
def __init__(self, rpc: "Rpc", request_id: int, event: Event):
|
||||||
self.rpc = rpc
|
self.rpc = rpc
|
||||||
self.request_id = request_id
|
self.request_id = request_id
|
||||||
self.event = event
|
self.event = event
|
||||||
|
|
||||||
def __call__(self):
|
def __call__(self):
|
||||||
|
"""Wait for the future to return the result."""
|
||||||
self.event.wait()
|
self.event.wait()
|
||||||
response = self.rpc.request_results.pop(self.request_id)
|
response = self.rpc.request_results.pop(self.request_id)
|
||||||
if "error" in response:
|
if "error" in response:
|
||||||
@@ -32,17 +37,19 @@ class RpcFuture:
|
|||||||
|
|
||||||
|
|
||||||
class RpcMethod:
|
class RpcMethod:
|
||||||
|
"""RPC method."""
|
||||||
|
|
||||||
def __init__(self, rpc: "Rpc", name: str):
|
def __init__(self, rpc: "Rpc", name: str):
|
||||||
self.rpc = rpc
|
self.rpc = rpc
|
||||||
self.name = name
|
self.name = name
|
||||||
|
|
||||||
def __call__(self, *args) -> Any:
|
def __call__(self, *args) -> Any:
|
||||||
"""Synchronously calls JSON-RPC method."""
|
"""Call JSON-RPC method synchronously."""
|
||||||
future = self.future(*args)
|
future = self.future(*args)
|
||||||
return future()
|
return future()
|
||||||
|
|
||||||
def future(self, *args) -> Any:
|
def future(self, *args) -> Any:
|
||||||
"""Asynchronously calls JSON-RPC method."""
|
"""Call JSON-RPC method asynchronously."""
|
||||||
request_id = next(self.rpc.id_iterator)
|
request_id = next(self.rpc.id_iterator)
|
||||||
request = {
|
request = {
|
||||||
"jsonrpc": "2.0",
|
"jsonrpc": "2.0",
|
||||||
@@ -58,8 +65,13 @@ class RpcMethod:
|
|||||||
|
|
||||||
|
|
||||||
class Rpc:
|
class Rpc:
|
||||||
|
"""RPC client."""
|
||||||
|
|
||||||
def __init__(self, accounts_dir: Optional[str] = None, **kwargs):
|
def __init__(self, accounts_dir: Optional[str] = None, **kwargs):
|
||||||
"""The given arguments will be passed to subprocess.Popen()"""
|
"""Initialize RPC client.
|
||||||
|
|
||||||
|
The given arguments will be passed to subprocess.Popen().
|
||||||
|
"""
|
||||||
if accounts_dir:
|
if accounts_dir:
|
||||||
kwargs["env"] = {
|
kwargs["env"] = {
|
||||||
**kwargs.get("env", os.environ),
|
**kwargs.get("env", os.environ),
|
||||||
@@ -81,6 +93,7 @@ class Rpc:
|
|||||||
self.events_thread: Thread
|
self.events_thread: Thread
|
||||||
|
|
||||||
def start(self) -> None:
|
def start(self) -> None:
|
||||||
|
"""Start RPC server subprocess."""
|
||||||
if sys.version_info >= (3, 11):
|
if sys.version_info >= (3, 11):
|
||||||
self.process = subprocess.Popen(
|
self.process = subprocess.Popen(
|
||||||
"deltachat-rpc-server",
|
"deltachat-rpc-server",
|
||||||
@@ -130,6 +143,7 @@ class Rpc:
|
|||||||
self.close()
|
self.close()
|
||||||
|
|
||||||
def reader_loop(self) -> None:
|
def reader_loop(self) -> None:
|
||||||
|
"""Process JSON-RPC responses from the RPC server process output."""
|
||||||
try:
|
try:
|
||||||
while line := self.process.stdout.readline():
|
while line := self.process.stdout.readline():
|
||||||
response = json.loads(line)
|
response = json.loads(line)
|
||||||
@@ -157,12 +171,13 @@ class Rpc:
|
|||||||
logging.exception("Exception in the writer loop")
|
logging.exception("Exception in the writer loop")
|
||||||
|
|
||||||
def get_queue(self, account_id: int) -> Queue:
|
def get_queue(self, account_id: int) -> Queue:
|
||||||
|
"""Get event queue corresponding to the given account ID."""
|
||||||
if account_id not in self.event_queues:
|
if account_id not in self.event_queues:
|
||||||
self.event_queues[account_id] = Queue()
|
self.event_queues[account_id] = Queue()
|
||||||
return self.event_queues[account_id]
|
return self.event_queues[account_id]
|
||||||
|
|
||||||
def events_loop(self) -> None:
|
def events_loop(self) -> None:
|
||||||
"""Requests new events and distributes them between queues."""
|
"""Request new events and distributes them between queues."""
|
||||||
try:
|
try:
|
||||||
while True:
|
while True:
|
||||||
if self.closing:
|
if self.closing:
|
||||||
@@ -178,12 +193,12 @@ class Rpc:
|
|||||||
logging.exception("Exception in the event loop")
|
logging.exception("Exception in the event loop")
|
||||||
|
|
||||||
def wait_for_event(self, account_id: int) -> Optional[dict]:
|
def wait_for_event(self, account_id: int) -> Optional[dict]:
|
||||||
"""Waits for the next event from the given account and returns it."""
|
"""Wait for the next event from the given account and returns it."""
|
||||||
queue = self.get_queue(account_id)
|
queue = self.get_queue(account_id)
|
||||||
return queue.get()
|
return queue.get()
|
||||||
|
|
||||||
def clear_all_events(self, account_id: int):
|
def clear_all_events(self, account_id: int):
|
||||||
"""Removes all queued-up events for a given account. Useful for tests."""
|
"""Remove all queued-up events for a given account. Useful for tests."""
|
||||||
queue = self.get_queue(account_id)
|
queue = self.get_queue(account_id)
|
||||||
try:
|
try:
|
||||||
while True:
|
while True:
|
||||||
|
|||||||
@@ -13,10 +13,11 @@ def test_event_on_configuration(acfactory: ACFactory) -> None:
|
|||||||
Test if ACCOUNTS_ITEM_CHANGED event is emitted on configure
|
Test if ACCOUNTS_ITEM_CHANGED event is emitted on configure
|
||||||
"""
|
"""
|
||||||
|
|
||||||
account = acfactory.new_preconfigured_account()
|
addr, password = acfactory.get_credentials()
|
||||||
|
account = acfactory.get_unconfigured_account()
|
||||||
account.clear_all_events()
|
account.clear_all_events()
|
||||||
assert not account.is_configured()
|
assert not account.is_configured()
|
||||||
future = account.configure.future()
|
future = account.add_or_update_transport.future({"addr": addr, "password": password})
|
||||||
while True:
|
while True:
|
||||||
event = account.wait_for_event()
|
event = account.wait_for_event()
|
||||||
if event.kind == EventType.ACCOUNTS_ITEM_CHANGED:
|
if event.kind == EventType.ACCOUNTS_ITEM_CHANGED:
|
||||||
|
|||||||
109
deltachat-rpc-client/tests/test_calls.py
Normal file
109
deltachat-rpc-client/tests/test_calls.py
Normal file
@@ -0,0 +1,109 @@
|
|||||||
|
from deltachat_rpc_client import EventType, Message
|
||||||
|
|
||||||
|
|
||||||
|
def test_calls(acfactory) -> None:
|
||||||
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
|
place_call_info = "offer"
|
||||||
|
accept_call_info = "answer"
|
||||||
|
|
||||||
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
|
bob.create_chat(alice) # Accept the chat so incoming call causes a notification.
|
||||||
|
outgoing_call_message = alice_chat_bob.place_outgoing_call(place_call_info)
|
||||||
|
assert outgoing_call_message.get_call_info().state.kind == "Alerting"
|
||||||
|
|
||||||
|
incoming_call_event = bob.wait_for_event(EventType.INCOMING_CALL)
|
||||||
|
assert incoming_call_event.place_call_info == place_call_info
|
||||||
|
assert not incoming_call_event.has_video # Cannot be parsed as SDP, so false by default
|
||||||
|
incoming_call_message = Message(bob, incoming_call_event.msg_id)
|
||||||
|
assert incoming_call_message.get_call_info().state.kind == "Alerting"
|
||||||
|
assert not incoming_call_message.get_call_info().has_video
|
||||||
|
|
||||||
|
incoming_call_message.accept_incoming_call(accept_call_info)
|
||||||
|
assert incoming_call_message.get_call_info().sdp_offer == place_call_info
|
||||||
|
assert incoming_call_message.get_call_info().state.kind == "Active"
|
||||||
|
outgoing_call_accepted_event = alice.wait_for_event(EventType.OUTGOING_CALL_ACCEPTED)
|
||||||
|
assert outgoing_call_accepted_event.accept_call_info == accept_call_info
|
||||||
|
assert outgoing_call_message.get_call_info().state.kind == "Active"
|
||||||
|
|
||||||
|
outgoing_call_message.end_call()
|
||||||
|
assert outgoing_call_message.get_call_info().state.kind == "Completed"
|
||||||
|
|
||||||
|
end_call_event = bob.wait_for_event(EventType.CALL_ENDED)
|
||||||
|
assert end_call_event.msg_id == outgoing_call_message.id
|
||||||
|
assert incoming_call_message.get_call_info().state.kind == "Completed"
|
||||||
|
|
||||||
|
|
||||||
|
def test_video_call(acfactory) -> None:
|
||||||
|
# Example from <https://datatracker.ietf.org/doc/rfc9143/>
|
||||||
|
# with `s= ` replaced with `s=-`.
|
||||||
|
#
|
||||||
|
# `s=` cannot be empty according to RFC 3264,
|
||||||
|
# so it is more clear as `s=-`.
|
||||||
|
place_call_info = """v=0\r
|
||||||
|
o=alice 2890844526 2890844526 IN IP6 2001:db8::3\r
|
||||||
|
s=-\r
|
||||||
|
c=IN IP6 2001:db8::3\r
|
||||||
|
t=0 0\r
|
||||||
|
a=group:BUNDLE foo bar\r
|
||||||
|
\r
|
||||||
|
m=audio 10000 RTP/AVP 0 8 97\r
|
||||||
|
b=AS:200\r
|
||||||
|
a=mid:foo\r
|
||||||
|
a=rtcp-mux\r
|
||||||
|
a=rtpmap:0 PCMU/8000\r
|
||||||
|
a=rtpmap:8 PCMA/8000\r
|
||||||
|
a=rtpmap:97 iLBC/8000\r
|
||||||
|
a=extmap:1 urn:ietf:params:rtp-hdrext:sdes:mid\r
|
||||||
|
\r
|
||||||
|
m=video 10002 RTP/AVP 31 32\r
|
||||||
|
b=AS:1000\r
|
||||||
|
a=mid:bar\r
|
||||||
|
a=rtcp-mux\r
|
||||||
|
a=rtpmap:31 H261/90000\r
|
||||||
|
a=rtpmap:32 MPV/90000\r
|
||||||
|
a=extmap:1 urn:ietf:params:rtp-hdrext:sdes:mid\r
|
||||||
|
"""
|
||||||
|
|
||||||
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
|
bob.create_chat(alice) # Accept the chat so incoming call causes a notification.
|
||||||
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
|
alice_chat_bob.place_outgoing_call(place_call_info)
|
||||||
|
|
||||||
|
incoming_call_event = bob.wait_for_event(EventType.INCOMING_CALL)
|
||||||
|
assert incoming_call_event.place_call_info == place_call_info
|
||||||
|
assert incoming_call_event.has_video
|
||||||
|
|
||||||
|
incoming_call_message = Message(bob, incoming_call_event.msg_id)
|
||||||
|
assert incoming_call_message.get_call_info().has_video
|
||||||
|
|
||||||
|
|
||||||
|
def test_ice_servers(acfactory) -> None:
|
||||||
|
alice = acfactory.get_online_account()
|
||||||
|
|
||||||
|
ice_servers = alice.ice_servers()
|
||||||
|
assert len(ice_servers) == 1
|
||||||
|
|
||||||
|
|
||||||
|
def test_no_contact_request_call(acfactory) -> None:
|
||||||
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
|
alice_chat_bob = alice.create_chat(bob)
|
||||||
|
alice_chat_bob.place_outgoing_call("offer")
|
||||||
|
alice_chat_bob.send_text("Hello!")
|
||||||
|
|
||||||
|
# Notification for "Hello!" message should arrive
|
||||||
|
# without the call ringing.
|
||||||
|
while True:
|
||||||
|
event = bob.wait_for_event()
|
||||||
|
|
||||||
|
# There should be no incoming call notification.
|
||||||
|
assert event.kind != EventType.INCOMING_CALL
|
||||||
|
|
||||||
|
if event.kind == EventType.MSGS_CHANGED:
|
||||||
|
msg = bob.get_message_by_id(event.msg_id)
|
||||||
|
if msg.get_snapshot().text == "Hello!":
|
||||||
|
break
|
||||||
@@ -48,8 +48,7 @@ def test_delivery_status(acfactory: ACFactory) -> None:
|
|||||||
"""
|
"""
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
|
|
||||||
alice.clear_all_events()
|
alice.clear_all_events()
|
||||||
@@ -119,8 +118,7 @@ def test_download_on_demand(acfactory: ACFactory) -> None:
|
|||||||
"""
|
"""
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_text("hi")
|
alice_chat_bob.send_text("hi")
|
||||||
|
|
||||||
@@ -150,8 +148,7 @@ def test_download_on_demand(acfactory: ACFactory) -> None:
|
|||||||
def get_multi_account_test_setup(acfactory: ACFactory) -> [Account, Account, Account]:
|
def get_multi_account_test_setup(acfactory: ACFactory) -> [Account, Account, Account]:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_text("hi")
|
alice_chat_bob.send_text("hi")
|
||||||
|
|
||||||
@@ -172,6 +169,8 @@ def test_imap_sync_seen_msgs(acfactory: ACFactory) -> None:
|
|||||||
"""
|
"""
|
||||||
alice, alice_second_device, bob, alice_chat_bob = get_multi_account_test_setup(acfactory)
|
alice, alice_second_device, bob, alice_chat_bob = get_multi_account_test_setup(acfactory)
|
||||||
|
|
||||||
|
bob.create_chat(alice)
|
||||||
|
|
||||||
alice_chat_bob.send_text("hello")
|
alice_chat_bob.send_text("hello")
|
||||||
|
|
||||||
msg = bob.wait_for_incoming_msg()
|
msg = bob.wait_for_incoming_msg()
|
||||||
|
|||||||
@@ -214,7 +214,9 @@ def test_advertisement_after_chatting(acfactory, path_to_webxdc):
|
|||||||
ac1_ac2_chat = ac1.create_chat(ac2)
|
ac1_ac2_chat = ac1.create_chat(ac2)
|
||||||
ac1_webxdc_msg = ac1_ac2_chat.send_message(text="WebXDC", file=path_to_webxdc)
|
ac1_webxdc_msg = ac1_ac2_chat.send_message(text="WebXDC", file=path_to_webxdc)
|
||||||
ac2_webxdc_msg = ac2.wait_for_incoming_msg()
|
ac2_webxdc_msg = ac2.wait_for_incoming_msg()
|
||||||
assert ac2_webxdc_msg.get_snapshot().text == "WebXDC"
|
ac2_webxdc_msg_snapshot = ac2_webxdc_msg.get_snapshot()
|
||||||
|
assert ac2_webxdc_msg_snapshot.text == "WebXDC"
|
||||||
|
ac2_webxdc_msg_snapshot.chat.accept()
|
||||||
|
|
||||||
ac1_ac2_chat.send_text("Hello!")
|
ac1_ac2_chat.send_text("Hello!")
|
||||||
ac2_hello_msg = ac2.wait_for_incoming_msg()
|
ac2_hello_msg = ac2.wait_for_incoming_msg()
|
||||||
|
|||||||
@@ -18,9 +18,7 @@ def test_autocrypt_setup_message_key_transfer(acfactory):
|
|||||||
alice1 = acfactory.get_online_account()
|
alice1 = acfactory.get_online_account()
|
||||||
|
|
||||||
alice2 = acfactory.get_unconfigured_account()
|
alice2 = acfactory.get_unconfigured_account()
|
||||||
alice2.set_config("addr", alice1.get_config("addr"))
|
alice2.add_or_update_transport({"addr": alice1.get_config("addr"), "password": alice1.get_config("mail_pw")})
|
||||||
alice2.set_config("mail_pw", alice1.get_config("mail_pw"))
|
|
||||||
alice2.configure()
|
|
||||||
alice2.bring_online()
|
alice2.bring_online()
|
||||||
|
|
||||||
setup_code = alice1.initiate_autocrypt_key_transfer()
|
setup_code = alice1.initiate_autocrypt_key_transfer()
|
||||||
@@ -37,9 +35,7 @@ def test_ac_setup_message_twice(acfactory):
|
|||||||
alice1 = acfactory.get_online_account()
|
alice1 = acfactory.get_online_account()
|
||||||
|
|
||||||
alice2 = acfactory.get_unconfigured_account()
|
alice2 = acfactory.get_unconfigured_account()
|
||||||
alice2.set_config("addr", alice1.get_config("addr"))
|
alice2.add_or_update_transport({"addr": alice1.get_config("addr"), "password": alice1.get_config("mail_pw")})
|
||||||
alice2.set_config("mail_pw", alice1.get_config("mail_pw"))
|
|
||||||
alice2.configure()
|
|
||||||
alice2.bring_online()
|
alice2.bring_online()
|
||||||
|
|
||||||
# Send the first Autocrypt Setup Message and ignore it.
|
# Send the first Autocrypt Setup Message and ignore it.
|
||||||
|
|||||||
113
deltachat-rpc-client/tests/test_multidevice.py
Normal file
113
deltachat-rpc-client/tests/test_multidevice.py
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
from imap_tools import AND
|
||||||
|
|
||||||
|
from deltachat_rpc_client import EventType
|
||||||
|
from deltachat_rpc_client.const import MessageState
|
||||||
|
|
||||||
|
|
||||||
|
def test_one_account_send_bcc_setting(acfactory, log, direct_imap):
|
||||||
|
ac1, ac2 = acfactory.get_online_accounts(2)
|
||||||
|
ac1_clone = ac1.clone()
|
||||||
|
ac1_clone.bring_online()
|
||||||
|
|
||||||
|
log.section("send out message without bcc to ourselves")
|
||||||
|
ac1.set_config("bcc_self", "0")
|
||||||
|
chat = ac1.create_chat(ac2)
|
||||||
|
self_addr = ac1.get_config("addr")
|
||||||
|
other_addr = ac2.get_config("addr")
|
||||||
|
|
||||||
|
msg_out = chat.send_text("message1")
|
||||||
|
assert not msg_out.get_snapshot().is_forwarded
|
||||||
|
|
||||||
|
# wait for send out (no BCC)
|
||||||
|
ev = ac1.wait_for_event(EventType.SMTP_MESSAGE_SENT)
|
||||||
|
assert ac1.get_config("bcc_self") == "0"
|
||||||
|
|
||||||
|
assert self_addr not in ev.msg
|
||||||
|
assert other_addr in ev.msg
|
||||||
|
|
||||||
|
log.section("ac1: setting bcc_self=1")
|
||||||
|
ac1.set_config("bcc_self", "1")
|
||||||
|
|
||||||
|
log.section("send out message with bcc to ourselves")
|
||||||
|
msg_out = chat.send_text("message2")
|
||||||
|
|
||||||
|
# wait for send out (BCC)
|
||||||
|
ev = ac1.wait_for_event(EventType.SMTP_MESSAGE_SENT)
|
||||||
|
assert ac1.get_config("bcc_self") == "1"
|
||||||
|
|
||||||
|
# Second client receives only second message, but not the first.
|
||||||
|
ev_msg = ac1_clone.wait_for_event(EventType.MSGS_CHANGED)
|
||||||
|
assert ac1_clone.get_message_by_id(ev_msg.msg_id).get_snapshot().text == "Messages are end-to-end encrypted."
|
||||||
|
|
||||||
|
ev_msg = ac1_clone.wait_for_event(EventType.MSGS_CHANGED)
|
||||||
|
assert ac1_clone.get_message_by_id(ev_msg.msg_id).get_snapshot().text == msg_out.get_snapshot().text
|
||||||
|
|
||||||
|
# now make sure we are sending message to ourselves too
|
||||||
|
assert self_addr in ev.msg
|
||||||
|
assert self_addr in ev.msg
|
||||||
|
|
||||||
|
# BCC-self messages are marked as seen by the sender device.
|
||||||
|
while True:
|
||||||
|
event = ac1.wait_for_event()
|
||||||
|
if event.kind == EventType.INFO and event.msg.endswith("Marked messages 1 in folder INBOX as seen."):
|
||||||
|
break
|
||||||
|
|
||||||
|
# Check that the message is marked as seen on IMAP.
|
||||||
|
ac1_direct_imap = direct_imap(ac1)
|
||||||
|
ac1_direct_imap.connect()
|
||||||
|
ac1_direct_imap.select_folder("Inbox")
|
||||||
|
assert len(list(ac1_direct_imap.conn.fetch(AND(seen=True)))) == 1
|
||||||
|
|
||||||
|
|
||||||
|
def test_multidevice_sync_seen(acfactory, log):
|
||||||
|
"""Test that message marked as seen on one device is marked as seen on another."""
|
||||||
|
ac1, ac2 = acfactory.get_online_accounts(2)
|
||||||
|
ac1_clone = ac1.clone()
|
||||||
|
ac1_clone.bring_online()
|
||||||
|
|
||||||
|
ac1.set_config("bcc_self", "1")
|
||||||
|
ac1_clone.set_config("bcc_self", "1")
|
||||||
|
|
||||||
|
ac1_chat = ac1.create_chat(ac2)
|
||||||
|
ac1_clone_chat = ac1_clone.create_chat(ac2)
|
||||||
|
ac2_chat = ac2.create_chat(ac1)
|
||||||
|
|
||||||
|
log.section("Send a message from ac2 to ac1 and check that it's 'fresh'")
|
||||||
|
ac2_chat.send_text("Hi")
|
||||||
|
ac1_message = ac1.wait_for_incoming_msg()
|
||||||
|
ac1_clone_message = ac1_clone.wait_for_incoming_msg()
|
||||||
|
assert ac1_chat.get_fresh_message_count() == 1
|
||||||
|
assert ac1_clone_chat.get_fresh_message_count() == 1
|
||||||
|
assert ac1_message.get_snapshot().state == MessageState.IN_FRESH
|
||||||
|
assert ac1_clone_message.get_snapshot().state == MessageState.IN_FRESH
|
||||||
|
|
||||||
|
log.section("ac1 marks message as seen on the first device")
|
||||||
|
ac1.mark_seen_messages([ac1_message])
|
||||||
|
assert ac1_message.get_snapshot().state == MessageState.IN_SEEN
|
||||||
|
|
||||||
|
log.section("ac1 clone detects that message is marked as seen")
|
||||||
|
ev = ac1_clone.wait_for_event(EventType.MSGS_NOTICED)
|
||||||
|
assert ev.chat_id == ac1_clone_chat.id
|
||||||
|
|
||||||
|
log.section("Send an ephemeral message from ac2 to ac1")
|
||||||
|
ac2_chat.set_ephemeral_timer(60)
|
||||||
|
ac1.wait_for_event(EventType.CHAT_EPHEMERAL_TIMER_MODIFIED)
|
||||||
|
ac1.wait_for_incoming_msg()
|
||||||
|
ac1_clone.wait_for_event(EventType.CHAT_EPHEMERAL_TIMER_MODIFIED)
|
||||||
|
ac1_clone.wait_for_incoming_msg()
|
||||||
|
|
||||||
|
ac2_chat.send_text("Foobar")
|
||||||
|
ac1_message = ac1.wait_for_incoming_msg()
|
||||||
|
ac1_clone_message = ac1_clone.wait_for_incoming_msg()
|
||||||
|
assert "Ephemeral timer: 60\n" in ac1_message.get_info()
|
||||||
|
assert "Expires: " not in ac1_clone_message.get_info()
|
||||||
|
assert "Ephemeral timer: 60\n" in ac1_message.get_info()
|
||||||
|
assert "Expires: " not in ac1_clone_message.get_info()
|
||||||
|
|
||||||
|
ac1_message.mark_seen()
|
||||||
|
assert "Expires: " in ac1_message.get_info()
|
||||||
|
ev = ac1_clone.wait_for_event(EventType.MSGS_NOTICED)
|
||||||
|
assert ev.chat_id == ac1_clone_chat.id
|
||||||
|
assert ac1_clone_message.get_snapshot().state == MessageState.IN_SEEN
|
||||||
|
# Test that the timer is started on the second device after synchronizing the seen status.
|
||||||
|
assert "Expires: " in ac1_clone_message.get_info()
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
import logging
|
import logging
|
||||||
import time
|
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from deltachat_rpc_client import Chat, EventType, SpecialContactId
|
from deltachat_rpc_client import Chat, EventType, SpecialContactId
|
||||||
|
from deltachat_rpc_client.const import ChatType
|
||||||
from deltachat_rpc_client.rpc import JsonRpcError
|
from deltachat_rpc_client.rpc import JsonRpcError
|
||||||
|
|
||||||
|
|
||||||
@@ -16,14 +16,14 @@ def test_qr_setup_contact(acfactory, tmp_path) -> None:
|
|||||||
alice.wait_for_securejoin_inviter_success()
|
alice.wait_for_securejoin_inviter_success()
|
||||||
|
|
||||||
# Test that Alice verified Bob's profile.
|
# Test that Alice verified Bob's profile.
|
||||||
alice_contact_bob = alice.get_contact_by_addr(bob.get_config("addr"))
|
alice_contact_bob = alice.create_contact(bob)
|
||||||
alice_contact_bob_snapshot = alice_contact_bob.get_snapshot()
|
alice_contact_bob_snapshot = alice_contact_bob.get_snapshot()
|
||||||
assert alice_contact_bob_snapshot.is_verified
|
assert alice_contact_bob_snapshot.is_verified
|
||||||
|
|
||||||
bob.wait_for_securejoin_joiner_success()
|
bob.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
# Test that Bob verified Alice's profile.
|
# Test that Bob verified Alice's profile.
|
||||||
bob_contact_alice = bob.get_contact_by_addr(alice.get_config("addr"))
|
bob_contact_alice = bob.create_contact(alice)
|
||||||
bob_contact_alice_snapshot = bob_contact_alice.get_snapshot()
|
bob_contact_alice_snapshot = bob_contact_alice.get_snapshot()
|
||||||
assert bob_contact_alice_snapshot.is_verified
|
assert bob_contact_alice_snapshot.is_verified
|
||||||
|
|
||||||
@@ -59,8 +59,7 @@ def test_qr_setup_contact_svg(acfactory) -> None:
|
|||||||
assert "Alice" in svg
|
assert "Alice" in svg
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("protect", [True, False])
|
def test_qr_securejoin(acfactory):
|
||||||
def test_qr_securejoin(acfactory, protect):
|
|
||||||
alice, bob, fiona = acfactory.get_online_accounts(3)
|
alice, bob, fiona = acfactory.get_online_accounts(3)
|
||||||
|
|
||||||
# Setup second device for Alice
|
# Setup second device for Alice
|
||||||
@@ -68,8 +67,7 @@ def test_qr_securejoin(acfactory, protect):
|
|||||||
alice2 = alice.clone()
|
alice2 = alice.clone()
|
||||||
|
|
||||||
logging.info("Alice creates a group")
|
logging.info("Alice creates a group")
|
||||||
alice_chat = alice.create_group("Group", protect=protect)
|
alice_chat = alice.create_group("Group")
|
||||||
assert alice_chat.get_basic_snapshot().is_protected == protect
|
|
||||||
|
|
||||||
logging.info("Bob joins the group")
|
logging.info("Bob joins the group")
|
||||||
qr_code = alice_chat.get_qr_code()
|
qr_code = alice_chat.get_qr_code()
|
||||||
@@ -84,16 +82,15 @@ def test_qr_securejoin(acfactory, protect):
|
|||||||
bob.wait_for_securejoin_joiner_success()
|
bob.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
# Test that Alice verified Bob's profile.
|
# Test that Alice verified Bob's profile.
|
||||||
alice_contact_bob = alice.get_contact_by_addr(bob.get_config("addr"))
|
alice_contact_bob = alice.create_contact(bob)
|
||||||
alice_contact_bob_snapshot = alice_contact_bob.get_snapshot()
|
alice_contact_bob_snapshot = alice_contact_bob.get_snapshot()
|
||||||
assert alice_contact_bob_snapshot.is_verified
|
assert alice_contact_bob_snapshot.is_verified
|
||||||
|
|
||||||
snapshot = bob.get_message_by_id(bob.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
snapshot = bob.get_message_by_id(bob.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
assert snapshot.text == "Member Me ({}) added by {}.".format(bob.get_config("addr"), alice.get_config("addr"))
|
assert snapshot.text == "Member Me added by {}.".format(alice.get_config("addr"))
|
||||||
assert snapshot.chat.get_basic_snapshot().is_protected == protect
|
|
||||||
|
|
||||||
# Test that Bob verified Alice's profile.
|
# Test that Bob verified Alice's profile.
|
||||||
bob_contact_alice = bob.get_contact_by_addr(alice.get_config("addr"))
|
bob_contact_alice = bob.create_contact(alice)
|
||||||
bob_contact_alice_snapshot = bob_contact_alice.get_snapshot()
|
bob_contact_alice_snapshot = bob_contact_alice.get_snapshot()
|
||||||
assert bob_contact_alice_snapshot.is_verified
|
assert bob_contact_alice_snapshot.is_verified
|
||||||
|
|
||||||
@@ -101,7 +98,7 @@ def test_qr_securejoin(acfactory, protect):
|
|||||||
# Alice observes securejoin protocol and verifies Bob on second device.
|
# Alice observes securejoin protocol and verifies Bob on second device.
|
||||||
alice2.start_io()
|
alice2.start_io()
|
||||||
alice2.wait_for_securejoin_inviter_success()
|
alice2.wait_for_securejoin_inviter_success()
|
||||||
alice2_contact_bob = alice2.get_contact_by_addr(bob.get_config("addr"))
|
alice2_contact_bob = alice2.create_contact(bob)
|
||||||
alice2_contact_bob_snapshot = alice2_contact_bob.get_snapshot()
|
alice2_contact_bob_snapshot = alice2_contact_bob.get_snapshot()
|
||||||
assert alice2_contact_bob_snapshot.is_verified
|
assert alice2_contact_bob_snapshot.is_verified
|
||||||
|
|
||||||
@@ -113,12 +110,148 @@ def test_qr_securejoin(acfactory, protect):
|
|||||||
fiona.wait_for_securejoin_joiner_success()
|
fiona.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("all_devices_online", [True, False])
|
||||||
|
def test_qr_securejoin_broadcast(acfactory, all_devices_online):
|
||||||
|
alice, bob, fiona = acfactory.get_online_accounts(3)
|
||||||
|
|
||||||
|
alice2 = alice.clone()
|
||||||
|
bob2 = bob.clone()
|
||||||
|
|
||||||
|
if all_devices_online:
|
||||||
|
alice2.start_io()
|
||||||
|
bob2.start_io()
|
||||||
|
|
||||||
|
logging.info("===================== Alice creates a broadcast =====================")
|
||||||
|
alice_chat = alice.create_broadcast("Broadcast channel!")
|
||||||
|
snapshot = alice_chat.get_basic_snapshot()
|
||||||
|
assert not snapshot.is_unpromoted # Broadcast channels are never unpromoted
|
||||||
|
|
||||||
|
logging.info("===================== Bob joins the broadcast =====================")
|
||||||
|
|
||||||
|
qr_code = alice_chat.get_qr_code()
|
||||||
|
bob.secure_join(qr_code)
|
||||||
|
alice.wait_for_securejoin_inviter_success()
|
||||||
|
bob.wait_for_securejoin_joiner_success()
|
||||||
|
alice_chat.send_text("Hello everyone!")
|
||||||
|
|
||||||
|
def get_broadcast(ac):
|
||||||
|
chat = ac.get_chatlist(query="Broadcast channel!")[0]
|
||||||
|
assert chat.get_basic_snapshot().name == "Broadcast channel!"
|
||||||
|
return chat
|
||||||
|
|
||||||
|
def wait_for_broadcast_messages(ac):
|
||||||
|
chat = get_broadcast(ac)
|
||||||
|
|
||||||
|
snapshot = ac.wait_for_incoming_msg().get_snapshot()
|
||||||
|
assert snapshot.text == "You joined the channel."
|
||||||
|
assert snapshot.chat_id == chat.id
|
||||||
|
|
||||||
|
snapshot = ac.wait_for_incoming_msg().get_snapshot()
|
||||||
|
assert snapshot.text == "Hello everyone!"
|
||||||
|
assert snapshot.chat_id == chat.id
|
||||||
|
|
||||||
|
def check_account(ac, contact, inviter_side, please_wait_info_msg=False):
|
||||||
|
# Check that the chat partner is verified.
|
||||||
|
contact_snapshot = contact.get_snapshot()
|
||||||
|
assert contact_snapshot.is_verified
|
||||||
|
|
||||||
|
chat = get_broadcast(ac)
|
||||||
|
chat_msgs = chat.get_messages()
|
||||||
|
|
||||||
|
if please_wait_info_msg:
|
||||||
|
first_msg = chat_msgs.pop(0).get_snapshot()
|
||||||
|
assert first_msg.text == "Establishing guaranteed end-to-end encryption, please wait…"
|
||||||
|
assert first_msg.is_info
|
||||||
|
|
||||||
|
encrypted_msg = chat_msgs[0].get_snapshot()
|
||||||
|
assert encrypted_msg.text == "Messages are end-to-end encrypted."
|
||||||
|
assert encrypted_msg.is_info
|
||||||
|
|
||||||
|
member_added_msg = chat_msgs[1].get_snapshot()
|
||||||
|
if inviter_side:
|
||||||
|
assert member_added_msg.text == f"Member {contact_snapshot.display_name} added."
|
||||||
|
else:
|
||||||
|
assert member_added_msg.text == "You joined the channel."
|
||||||
|
assert member_added_msg.is_info
|
||||||
|
|
||||||
|
hello_msg = chat_msgs[2].get_snapshot()
|
||||||
|
assert hello_msg.text == "Hello everyone!"
|
||||||
|
assert not hello_msg.is_info
|
||||||
|
assert hello_msg.show_padlock
|
||||||
|
assert hello_msg.error is None
|
||||||
|
|
||||||
|
assert len(chat_msgs) == 3
|
||||||
|
|
||||||
|
chat_snapshot = chat.get_full_snapshot()
|
||||||
|
assert chat_snapshot.is_encrypted
|
||||||
|
assert chat_snapshot.name == "Broadcast channel!"
|
||||||
|
if inviter_side:
|
||||||
|
assert chat_snapshot.chat_type == ChatType.OUT_BROADCAST
|
||||||
|
else:
|
||||||
|
assert chat_snapshot.chat_type == ChatType.IN_BROADCAST
|
||||||
|
assert chat_snapshot.can_send == inviter_side
|
||||||
|
|
||||||
|
chat_contacts = chat_snapshot.contact_ids
|
||||||
|
assert contact.id in chat_contacts
|
||||||
|
if inviter_side:
|
||||||
|
assert len(chat_contacts) == 1
|
||||||
|
else:
|
||||||
|
assert len(chat_contacts) == 2
|
||||||
|
assert SpecialContactId.SELF in chat_contacts
|
||||||
|
assert chat_snapshot.self_in_group
|
||||||
|
|
||||||
|
wait_for_broadcast_messages(bob)
|
||||||
|
|
||||||
|
check_account(alice, alice.create_contact(bob), inviter_side=True)
|
||||||
|
check_account(bob, bob.create_contact(alice), inviter_side=False, please_wait_info_msg=True)
|
||||||
|
|
||||||
|
logging.info("===================== Test Alice's second device =====================")
|
||||||
|
|
||||||
|
# Start second Alice device, if it wasn't started already.
|
||||||
|
alice2.start_io()
|
||||||
|
|
||||||
|
while True:
|
||||||
|
msg_id = alice2.wait_for_msgs_changed_event().msg_id
|
||||||
|
if msg_id:
|
||||||
|
snapshot = alice2.get_message_by_id(msg_id).get_snapshot()
|
||||||
|
if snapshot.text == "Hello everyone!":
|
||||||
|
break
|
||||||
|
|
||||||
|
check_account(alice2, alice2.create_contact(bob), inviter_side=True)
|
||||||
|
|
||||||
|
logging.info("===================== Test Bob's second device =====================")
|
||||||
|
|
||||||
|
# Start second Bob device, if it wasn't started already.
|
||||||
|
bob2.start_io()
|
||||||
|
bob2.wait_for_securejoin_joiner_success()
|
||||||
|
wait_for_broadcast_messages(bob2)
|
||||||
|
check_account(bob2, bob2.create_contact(alice), inviter_side=False)
|
||||||
|
|
||||||
|
# The QR code token is synced, so alice2 must be able to handle join requests.
|
||||||
|
logging.info("===================== Fiona joins the group via alice2 =====================")
|
||||||
|
alice.stop_io()
|
||||||
|
fiona.secure_join(qr_code)
|
||||||
|
alice2.wait_for_securejoin_inviter_success()
|
||||||
|
fiona.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
|
snapshot = fiona.wait_for_incoming_msg().get_snapshot()
|
||||||
|
assert snapshot.text == "You joined the channel."
|
||||||
|
|
||||||
|
get_broadcast(alice2).get_messages()[2].resend()
|
||||||
|
snapshot = fiona.wait_for_incoming_msg().get_snapshot()
|
||||||
|
assert snapshot.text == "Hello everyone!"
|
||||||
|
|
||||||
|
check_account(fiona, fiona.create_contact(alice), inviter_side=False, please_wait_info_msg=True)
|
||||||
|
|
||||||
|
# For Bob, the channel must not have changed:
|
||||||
|
check_account(bob, bob.create_contact(alice), inviter_side=False, please_wait_info_msg=True)
|
||||||
|
|
||||||
|
|
||||||
def test_qr_securejoin_contact_request(acfactory) -> None:
|
def test_qr_securejoin_contact_request(acfactory) -> None:
|
||||||
"""Alice invites Bob to a group when Bob's chat with Alice is in a contact request mode."""
|
"""Alice invites Bob to a group when Bob's chat with Alice is in a contact request mode."""
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_text("Hello!")
|
alice_chat_bob.send_text("Hello!")
|
||||||
|
|
||||||
@@ -127,8 +260,8 @@ def test_qr_securejoin_contact_request(acfactory) -> None:
|
|||||||
bob_chat_alice = snapshot.chat
|
bob_chat_alice = snapshot.chat
|
||||||
assert bob_chat_alice.get_basic_snapshot().is_contact_request
|
assert bob_chat_alice.get_basic_snapshot().is_contact_request
|
||||||
|
|
||||||
alice_chat = alice.create_group("Verified group", protect=True)
|
alice_chat = alice.create_group("Group")
|
||||||
logging.info("Bob joins verified group")
|
logging.info("Bob joins the group")
|
||||||
qr_code = alice_chat.get_qr_code()
|
qr_code = alice_chat.get_qr_code()
|
||||||
bob.secure_join(qr_code)
|
bob.secure_join(qr_code)
|
||||||
while True:
|
while True:
|
||||||
@@ -152,14 +285,11 @@ def test_qr_readreceipt(acfactory) -> None:
|
|||||||
for joiner in [bob, charlie]:
|
for joiner in [bob, charlie]:
|
||||||
joiner.wait_for_securejoin_joiner_success()
|
joiner.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
logging.info("Alice creates a verified group")
|
logging.info("Alice creates a group")
|
||||||
group = alice.create_group("Group", protect=True)
|
group = alice.create_group("Group")
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
charlie_addr = charlie.get_config("addr")
|
alice_contact_charlie = alice.create_contact(charlie, "Charlie")
|
||||||
|
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_contact_charlie = alice.create_contact(charlie_addr, "Charlie")
|
|
||||||
|
|
||||||
group.add_contact(alice_contact_bob)
|
group.add_contact(alice_contact_bob)
|
||||||
group.add_contact(alice_contact_charlie)
|
group.add_contact(alice_contact_charlie)
|
||||||
@@ -186,7 +316,7 @@ def test_qr_readreceipt(acfactory) -> None:
|
|||||||
charlie_snapshot = charlie_message.get_snapshot()
|
charlie_snapshot = charlie_message.get_snapshot()
|
||||||
assert charlie_snapshot.text == "Hi from Bob!"
|
assert charlie_snapshot.text == "Hi from Bob!"
|
||||||
|
|
||||||
bob_contact_charlie = bob.create_contact(charlie_addr, "Charlie")
|
bob_contact_charlie = bob.create_contact(charlie, "Charlie")
|
||||||
assert not bob.get_chat_by_contact(bob_contact_charlie)
|
assert not bob.get_chat_by_contact(bob_contact_charlie)
|
||||||
|
|
||||||
logging.info("Charlie reads Bob's message")
|
logging.info("Charlie reads Bob's message")
|
||||||
@@ -217,85 +347,20 @@ def test_setup_contact_resetup(acfactory) -> None:
|
|||||||
bob.wait_for_securejoin_joiner_success()
|
bob.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
|
|
||||||
def test_verified_group_recovery(acfactory) -> None:
|
|
||||||
"""Tests verified group recovery by reverifying a member and sending a message in a group."""
|
|
||||||
ac1, ac2, ac3 = acfactory.get_online_accounts(3)
|
|
||||||
|
|
||||||
logging.info("ac1 creates verified group")
|
|
||||||
chat = ac1.create_group("Verified group", protect=True)
|
|
||||||
assert chat.get_basic_snapshot().is_protected
|
|
||||||
|
|
||||||
logging.info("ac2 joins verified group")
|
|
||||||
qr_code = chat.get_qr_code()
|
|
||||||
ac2.secure_join(qr_code)
|
|
||||||
ac2.wait_for_securejoin_joiner_success()
|
|
||||||
|
|
||||||
# ac1 has ac2 directly verified.
|
|
||||||
ac1_contact_ac2 = ac1.get_contact_by_addr(ac2.get_config("addr"))
|
|
||||||
assert ac1_contact_ac2.get_snapshot().verifier_id == SpecialContactId.SELF
|
|
||||||
|
|
||||||
logging.info("ac3 joins verified group")
|
|
||||||
ac3_chat = ac3.secure_join(qr_code)
|
|
||||||
ac3.wait_for_securejoin_joiner_success()
|
|
||||||
ac3.wait_for_incoming_msg_event() # Member added
|
|
||||||
|
|
||||||
logging.info("ac2 logs in on a new device")
|
|
||||||
ac2 = acfactory.resetup_account(ac2)
|
|
||||||
|
|
||||||
logging.info("ac2 reverifies with ac3")
|
|
||||||
qr_code = ac3.get_qr_code()
|
|
||||||
ac2.secure_join(qr_code)
|
|
||||||
ac2.wait_for_securejoin_joiner_success()
|
|
||||||
|
|
||||||
logging.info("ac3 sends a message to the group")
|
|
||||||
assert len(ac3_chat.get_contacts()) == 3
|
|
||||||
ac3_chat.send_text("Hi!")
|
|
||||||
|
|
||||||
snapshot = ac1.get_message_by_id(ac1.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
|
||||||
assert snapshot.text == "Hi!"
|
|
||||||
|
|
||||||
msg_id = ac2.wait_for_incoming_msg_event().msg_id
|
|
||||||
message = ac2.get_message_by_id(msg_id)
|
|
||||||
snapshot = message.get_snapshot()
|
|
||||||
assert snapshot.text == "Hi!"
|
|
||||||
|
|
||||||
# ac1 contact is verified for ac2 because ac3 gossiped ac1 key in the "Hi!" message.
|
|
||||||
ac1_contact = ac2.get_contact_by_addr(ac1.get_config("addr"))
|
|
||||||
assert ac1_contact.get_snapshot().is_verified
|
|
||||||
|
|
||||||
# ac2 can write messages to the group.
|
|
||||||
snapshot.chat.send_text("Works again!")
|
|
||||||
|
|
||||||
snapshot = ac3.get_message_by_id(ac3.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
|
||||||
assert snapshot.text == "Works again!"
|
|
||||||
|
|
||||||
snapshot = ac1.get_message_by_id(ac1.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
|
||||||
assert snapshot.text == "Works again!"
|
|
||||||
|
|
||||||
ac1_chat_messages = snapshot.chat.get_messages()
|
|
||||||
ac2_addr = ac2.get_config("addr")
|
|
||||||
assert ac1_chat_messages[-2].get_snapshot().text == f"Changed setup for {ac2_addr}"
|
|
||||||
|
|
||||||
# ac2 is now verified by ac3 for ac1
|
|
||||||
ac1_contact_ac3 = ac1.get_contact_by_addr(ac3.get_config("addr"))
|
|
||||||
assert ac1_contact_ac2.get_snapshot().verifier_id == ac1_contact_ac3.id
|
|
||||||
|
|
||||||
|
|
||||||
def test_verified_group_member_added_recovery(acfactory) -> None:
|
def test_verified_group_member_added_recovery(acfactory) -> None:
|
||||||
"""Tests verified group recovery by reverifiying than removing and adding a member back."""
|
"""Tests verified group recovery by reverifying then removing and adding a member back."""
|
||||||
ac1, ac2, ac3 = acfactory.get_online_accounts(3)
|
ac1, ac2, ac3 = acfactory.get_online_accounts(3)
|
||||||
|
|
||||||
logging.info("ac1 creates verified group")
|
logging.info("ac1 creates a group")
|
||||||
chat = ac1.create_group("Verified group", protect=True)
|
chat = ac1.create_group("Group")
|
||||||
assert chat.get_basic_snapshot().is_protected
|
|
||||||
|
|
||||||
logging.info("ac2 joins verified group")
|
logging.info("ac2 joins the group")
|
||||||
qr_code = chat.get_qr_code()
|
qr_code = chat.get_qr_code()
|
||||||
ac2.secure_join(qr_code)
|
ac2.secure_join(qr_code)
|
||||||
ac2.wait_for_securejoin_joiner_success()
|
ac2.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
# ac1 has ac2 directly verified.
|
# ac1 has ac2 directly verified.
|
||||||
ac1_contact_ac2 = ac1.get_contact_by_addr(ac2.get_config("addr"))
|
ac1_contact_ac2 = ac1.create_contact(ac2)
|
||||||
assert ac1_contact_ac2.get_snapshot().verifier_id == SpecialContactId.SELF
|
assert ac1_contact_ac2.get_snapshot().verifier_id == SpecialContactId.SELF
|
||||||
|
|
||||||
logging.info("ac3 joins verified group")
|
logging.info("ac3 joins verified group")
|
||||||
@@ -303,6 +368,8 @@ def test_verified_group_member_added_recovery(acfactory) -> None:
|
|||||||
ac3.wait_for_securejoin_joiner_success()
|
ac3.wait_for_securejoin_joiner_success()
|
||||||
ac3.wait_for_incoming_msg_event() # Member added
|
ac3.wait_for_incoming_msg_event() # Member added
|
||||||
|
|
||||||
|
ac3_contact_ac2_old = ac3.create_contact(ac2)
|
||||||
|
|
||||||
logging.info("ac2 logs in on a new device")
|
logging.info("ac2 logs in on a new device")
|
||||||
ac2 = acfactory.resetup_account(ac2)
|
ac2 = acfactory.resetup_account(ac2)
|
||||||
|
|
||||||
@@ -315,21 +382,10 @@ def test_verified_group_member_added_recovery(acfactory) -> None:
|
|||||||
assert len(ac3_chat.get_contacts()) == 3
|
assert len(ac3_chat.get_contacts()) == 3
|
||||||
ac3_chat.send_text("Hi!")
|
ac3_chat.send_text("Hi!")
|
||||||
|
|
||||||
msg_id = ac2.wait_for_incoming_msg_event().msg_id
|
|
||||||
message = ac2.get_message_by_id(msg_id)
|
|
||||||
snapshot = message.get_snapshot()
|
|
||||||
logging.info("Received message %s", snapshot.text)
|
|
||||||
assert snapshot.text == "Hi!"
|
|
||||||
|
|
||||||
ac1.wait_for_incoming_msg_event() # Hi!
|
ac1.wait_for_incoming_msg_event() # Hi!
|
||||||
|
|
||||||
ac3_contact_ac2 = ac3.get_contact_by_addr(ac2.get_config("addr"))
|
ac3_contact_ac2 = ac3.create_contact(ac2)
|
||||||
ac3_chat.remove_contact(ac3_contact_ac2)
|
ac3_chat.remove_contact(ac3_contact_ac2_old)
|
||||||
|
|
||||||
msg_id = ac2.wait_for_incoming_msg_event().msg_id
|
|
||||||
message = ac2.get_message_by_id(msg_id)
|
|
||||||
snapshot = message.get_snapshot()
|
|
||||||
assert "removed" in snapshot.text
|
|
||||||
|
|
||||||
snapshot = ac1.get_message_by_id(ac1.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
snapshot = ac1.get_message_by_id(ac1.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
assert "removed" in snapshot.text
|
assert "removed" in snapshot.text
|
||||||
@@ -358,19 +414,18 @@ def test_verified_group_member_added_recovery(acfactory) -> None:
|
|||||||
snapshot = ac1.get_message_by_id(ac1.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
snapshot = ac1.get_message_by_id(ac1.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
assert snapshot.text == "Works again!"
|
assert snapshot.text == "Works again!"
|
||||||
|
|
||||||
ac1_contact_ac2 = ac1.get_contact_by_addr(ac2.get_config("addr"))
|
ac1_contact_ac2 = ac1.create_contact(ac2)
|
||||||
|
ac1_contact_ac3 = ac1.create_contact(ac3)
|
||||||
ac1_contact_ac2_snapshot = ac1_contact_ac2.get_snapshot()
|
ac1_contact_ac2_snapshot = ac1_contact_ac2.get_snapshot()
|
||||||
assert ac1_contact_ac2_snapshot.is_verified
|
# Until we reset verifications and then send the _verified header,
|
||||||
assert ac1_contact_ac2_snapshot.verifier_id == ac1.get_contact_by_addr(ac3.get_config("addr")).id
|
# verification is not gossiped here:
|
||||||
|
assert not ac1_contact_ac2_snapshot.is_verified
|
||||||
# ac2 is now verified by ac3 for ac1
|
assert ac1_contact_ac2_snapshot.verifier_id != ac1_contact_ac3.id
|
||||||
ac1_contact_ac3 = ac1.get_contact_by_addr(ac3.get_config("addr"))
|
|
||||||
assert ac1_contact_ac2.get_snapshot().verifier_id == ac1_contact_ac3.id
|
|
||||||
|
|
||||||
|
|
||||||
def test_qr_join_chat_with_pending_bobstate_issue4894(acfactory):
|
def test_qr_join_chat_with_pending_bobstate_issue4894(acfactory):
|
||||||
"""Regression test for
|
"""Regression test for
|
||||||
issue <https://github.com/deltachat/deltachat-core-rust/issues/4894>.
|
issue <https://github.com/chatmail/core/issues/4894>.
|
||||||
"""
|
"""
|
||||||
ac1, ac2, ac3, ac4 = acfactory.get_online_accounts(4)
|
ac1, ac2, ac3, ac4 = acfactory.get_online_accounts(4)
|
||||||
|
|
||||||
@@ -383,8 +438,8 @@ def test_qr_join_chat_with_pending_bobstate_issue4894(acfactory):
|
|||||||
# we first create a fully joined verified group, and then start
|
# we first create a fully joined verified group, and then start
|
||||||
# joining a second time but interrupt it, to create pending bob state
|
# joining a second time but interrupt it, to create pending bob state
|
||||||
|
|
||||||
logging.info("ac1: create verified group that ac2 fully joins")
|
logging.info("ac1: create a group that ac2 fully joins")
|
||||||
ch1 = ac1.create_group("Group", protect=True)
|
ch1 = ac1.create_group("Group")
|
||||||
qr_code = ch1.get_qr_code()
|
qr_code = ch1.get_qr_code()
|
||||||
ac2.secure_join(qr_code)
|
ac2.secure_join(qr_code)
|
||||||
ac1.wait_for_securejoin_inviter_success()
|
ac1.wait_for_securejoin_inviter_success()
|
||||||
@@ -394,7 +449,6 @@ def test_qr_join_chat_with_pending_bobstate_issue4894(acfactory):
|
|||||||
while 1:
|
while 1:
|
||||||
snapshot = ac2.get_message_by_id(ac2.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
snapshot = ac2.get_message_by_id(ac2.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
if snapshot.text == "ac1 says hello":
|
if snapshot.text == "ac1 says hello":
|
||||||
assert snapshot.chat.get_basic_snapshot().is_protected
|
|
||||||
break
|
break
|
||||||
|
|
||||||
logging.info("ac1: let ac2 join again but shutoff ac1 in the middle of securejoin")
|
logging.info("ac1: let ac2 join again but shutoff ac1 in the middle of securejoin")
|
||||||
@@ -404,19 +458,18 @@ def test_qr_join_chat_with_pending_bobstate_issue4894(acfactory):
|
|||||||
logging.info("ac2 now has pending bobstate but ac1 is shutoff")
|
logging.info("ac2 now has pending bobstate but ac1 is shutoff")
|
||||||
|
|
||||||
# we meanwhile expect ac3/ac2 verification started in the beginning to have completed
|
# we meanwhile expect ac3/ac2 verification started in the beginning to have completed
|
||||||
assert ac3.get_contact_by_addr(ac2.get_config("addr")).get_snapshot().is_verified
|
assert ac3.create_contact(ac2).get_snapshot().is_verified
|
||||||
assert ac2.get_contact_by_addr(ac3.get_config("addr")).get_snapshot().is_verified
|
assert ac2.create_contact(ac3).get_snapshot().is_verified
|
||||||
|
|
||||||
logging.info("ac3: create a verified group VG with ac2")
|
logging.info("ac3: create a verified group VG with ac2")
|
||||||
vg = ac3.create_group("ac3-created", protect=True)
|
vg = ac3.create_group("ac3-created")
|
||||||
vg.add_contact(ac3.get_contact_by_addr(ac2.get_config("addr")))
|
vg.add_contact(ac3.create_contact(ac2))
|
||||||
|
|
||||||
# ensure ac2 receives message in VG
|
# ensure ac2 receives message in VG
|
||||||
vg.send_text("hello")
|
vg.send_text("hello")
|
||||||
while 1:
|
while 1:
|
||||||
msg = ac2.get_message_by_id(ac2.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
msg = ac2.get_message_by_id(ac2.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
if msg.text == "hello":
|
if msg.text == "hello":
|
||||||
assert msg.chat.get_basic_snapshot().is_protected
|
|
||||||
break
|
break
|
||||||
|
|
||||||
logging.info("ac3: create a join-code for group VG and let ac4 join, check that ac2 got it")
|
logging.info("ac3: create a join-code for group VG and let ac4 join, check that ac2 got it")
|
||||||
@@ -440,14 +493,14 @@ def test_qr_new_group_unblocked(acfactory):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
ac1, ac2 = acfactory.get_online_accounts(2)
|
ac1, ac2 = acfactory.get_online_accounts(2)
|
||||||
ac1_chat = ac1.create_group("Group for joining", protect=True)
|
ac1_chat = ac1.create_group("Group for joining")
|
||||||
qr_code = ac1_chat.get_qr_code()
|
qr_code = ac1_chat.get_qr_code()
|
||||||
ac2.secure_join(qr_code)
|
ac2.secure_join(qr_code)
|
||||||
|
|
||||||
ac1.wait_for_securejoin_inviter_success()
|
ac1.wait_for_securejoin_inviter_success()
|
||||||
|
|
||||||
ac1_new_chat = ac1.create_group("Another group")
|
ac1_new_chat = ac1.create_group("Another group")
|
||||||
ac1_new_chat.add_contact(ac1.get_contact_by_addr(ac2.get_config("addr")))
|
ac1_new_chat.add_contact(ac1.create_contact(ac2))
|
||||||
# Receive "Member added" message.
|
# Receive "Member added" message.
|
||||||
ac2.wait_for_incoming_msg_event()
|
ac2.wait_for_incoming_msg_event()
|
||||||
|
|
||||||
@@ -462,12 +515,10 @@ def test_aeap_flow_verified(acfactory):
|
|||||||
"""Test that a new address is added to a contact when it changes its address."""
|
"""Test that a new address is added to a contact when it changes its address."""
|
||||||
ac1, ac2 = acfactory.get_online_accounts(2)
|
ac1, ac2 = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
# ac1new is only used to get a new address.
|
addr, password = acfactory.get_credentials()
|
||||||
ac1new = acfactory.new_preconfigured_account()
|
|
||||||
|
|
||||||
logging.info("ac1: create verified-group QR, ac2 scans and joins")
|
logging.info("ac1: create verified-group QR, ac2 scans and joins")
|
||||||
chat = ac1.create_group("hello", protect=True)
|
chat = ac1.create_group("hello")
|
||||||
assert chat.get_basic_snapshot().is_protected
|
|
||||||
qr_code = chat.get_qr_code()
|
qr_code = chat.get_qr_code()
|
||||||
logging.info("ac2: start QR-code based join-group protocol")
|
logging.info("ac2: start QR-code based join-group protocol")
|
||||||
ac2.secure_join(qr_code)
|
ac2.secure_join(qr_code)
|
||||||
@@ -483,8 +534,8 @@ def test_aeap_flow_verified(acfactory):
|
|||||||
assert msg_in_1.text == msg_out.text
|
assert msg_in_1.text == msg_out.text
|
||||||
|
|
||||||
logging.info("changing email account")
|
logging.info("changing email account")
|
||||||
ac1.set_config("addr", ac1new.get_config("addr"))
|
ac1.set_config("addr", addr)
|
||||||
ac1.set_config("mail_pw", ac1new.get_config("mail_pw"))
|
ac1.set_config("mail_pw", password)
|
||||||
ac1.stop_io()
|
ac1.stop_io()
|
||||||
ac1.configure()
|
ac1.configure()
|
||||||
ac1.start_io()
|
ac1.start_io()
|
||||||
@@ -497,11 +548,9 @@ def test_aeap_flow_verified(acfactory):
|
|||||||
msg_in_2_snapshot = msg_in_2.get_snapshot()
|
msg_in_2_snapshot = msg_in_2.get_snapshot()
|
||||||
assert msg_in_2_snapshot.text == msg_out.text
|
assert msg_in_2_snapshot.text == msg_out.text
|
||||||
assert msg_in_2_snapshot.chat.id == msg_in_1.chat.id
|
assert msg_in_2_snapshot.chat.id == msg_in_1.chat.id
|
||||||
assert msg_in_2.get_sender_contact().get_snapshot().address == ac1new.get_config("addr")
|
assert msg_in_2.get_sender_contact().get_snapshot().address == addr
|
||||||
assert len(msg_in_2_snapshot.chat.get_contacts()) == 2
|
assert len(msg_in_2_snapshot.chat.get_contacts()) == 2
|
||||||
assert ac1new.get_config("addr") in [
|
assert addr in [contact.get_snapshot().address for contact in msg_in_2_snapshot.chat.get_contacts()]
|
||||||
contact.get_snapshot().address for contact in msg_in_2_snapshot.chat.get_contacts()
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
def test_gossip_verification(acfactory) -> None:
|
def test_gossip_verification(acfactory) -> None:
|
||||||
@@ -517,13 +566,12 @@ def test_gossip_verification(acfactory) -> None:
|
|||||||
bob.secure_join(qr_code)
|
bob.secure_join(qr_code)
|
||||||
bob.wait_for_securejoin_joiner_success()
|
bob.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
bob_contact_alice = bob.create_contact(alice.get_config("addr"), "Alice")
|
bob_contact_alice = bob.create_contact(alice, "Alice")
|
||||||
bob_contact_carol = bob.create_contact(carol.get_config("addr"), "Carol")
|
bob_contact_carol = bob.create_contact(carol, "Carol")
|
||||||
carol_contact_alice = carol.create_contact(alice.get_config("addr"), "Alice")
|
carol_contact_alice = carol.create_contact(alice, "Alice")
|
||||||
|
|
||||||
logging.info("Bob creates an Autocrypt group")
|
logging.info("Bob creates an Autocrypt group")
|
||||||
bob_group_chat = bob.create_group("Autocrypt Group")
|
bob_group_chat = bob.create_group("Autocrypt Group")
|
||||||
assert not bob_group_chat.get_basic_snapshot().is_protected
|
|
||||||
bob_group_chat.add_contact(bob_contact_alice)
|
bob_group_chat.add_contact(bob_contact_alice)
|
||||||
bob_group_chat.add_contact(bob_contact_carol)
|
bob_group_chat.add_contact(bob_contact_carol)
|
||||||
bob_group_chat.send_message(text="Hello Autocrypt group")
|
bob_group_chat.send_message(text="Hello Autocrypt group")
|
||||||
@@ -532,13 +580,14 @@ def test_gossip_verification(acfactory) -> None:
|
|||||||
assert snapshot.text == "Hello Autocrypt group"
|
assert snapshot.text == "Hello Autocrypt group"
|
||||||
assert snapshot.show_padlock
|
assert snapshot.show_padlock
|
||||||
|
|
||||||
# Autocrypt group does not propagate verification.
|
# Group propagates verification using Autocrypt-Gossip header.
|
||||||
carol_contact_alice_snapshot = carol_contact_alice.get_snapshot()
|
carol_contact_alice_snapshot = carol_contact_alice.get_snapshot()
|
||||||
|
# Until we reset verifications and then send the _verified header,
|
||||||
|
# verification is not gossiped here:
|
||||||
assert not carol_contact_alice_snapshot.is_verified
|
assert not carol_contact_alice_snapshot.is_verified
|
||||||
|
|
||||||
logging.info("Bob creates a Securejoin group")
|
logging.info("Bob creates a Securejoin group")
|
||||||
bob_group_chat = bob.create_group("Securejoin Group", protect=True)
|
bob_group_chat = bob.create_group("Securejoin Group")
|
||||||
assert bob_group_chat.get_basic_snapshot().is_protected
|
|
||||||
bob_group_chat.add_contact(bob_contact_alice)
|
bob_group_chat.add_contact(bob_contact_alice)
|
||||||
bob_group_chat.add_contact(bob_contact_carol)
|
bob_group_chat.add_contact(bob_contact_carol)
|
||||||
bob_group_chat.send_message(text="Hello Securejoin group")
|
bob_group_chat.send_message(text="Hello Securejoin group")
|
||||||
@@ -549,7 +598,9 @@ def test_gossip_verification(acfactory) -> None:
|
|||||||
|
|
||||||
# Securejoin propagates verification.
|
# Securejoin propagates verification.
|
||||||
carol_contact_alice_snapshot = carol_contact_alice.get_snapshot()
|
carol_contact_alice_snapshot = carol_contact_alice.get_snapshot()
|
||||||
assert carol_contact_alice_snapshot.is_verified
|
# Until we reset verifications and then send the _verified header,
|
||||||
|
# verification is not gossiped here:
|
||||||
|
assert not carol_contact_alice_snapshot.is_verified
|
||||||
|
|
||||||
|
|
||||||
def test_securejoin_after_contact_resetup(acfactory) -> None:
|
def test_securejoin_after_contact_resetup(acfactory) -> None:
|
||||||
@@ -561,7 +612,7 @@ def test_securejoin_after_contact_resetup(acfactory) -> None:
|
|||||||
ac1, ac2, ac3 = acfactory.get_online_accounts(3)
|
ac1, ac2, ac3 = acfactory.get_online_accounts(3)
|
||||||
|
|
||||||
# ac3 creates protected group with ac1.
|
# ac3 creates protected group with ac1.
|
||||||
ac3_chat = ac3.create_group("Verified group", protect=True)
|
ac3_chat = ac3.create_group("Group")
|
||||||
|
|
||||||
# ac1 joins ac3 group.
|
# ac1 joins ac3 group.
|
||||||
ac3_qr_code = ac3_chat.get_qr_code()
|
ac3_qr_code = ac3_chat.get_qr_code()
|
||||||
@@ -570,7 +621,7 @@ def test_securejoin_after_contact_resetup(acfactory) -> None:
|
|||||||
|
|
||||||
# ac1 waits for member added message and creates a QR code.
|
# ac1 waits for member added message and creates a QR code.
|
||||||
snapshot = ac1.get_message_by_id(ac1.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
snapshot = ac1.get_message_by_id(ac1.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
assert snapshot.text == "Member Me ({}) added by {}.".format(ac1.get_config("addr"), ac3.get_config("addr"))
|
assert snapshot.text == "Member Me added by {}.".format(ac3.get_config("addr"))
|
||||||
ac1_qr_code = snapshot.chat.get_qr_code()
|
ac1_qr_code = snapshot.chat.get_qr_code()
|
||||||
|
|
||||||
# ac2 verifies ac1
|
# ac2 verifies ac1
|
||||||
@@ -579,35 +630,13 @@ def test_securejoin_after_contact_resetup(acfactory) -> None:
|
|||||||
ac2.wait_for_securejoin_joiner_success()
|
ac2.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
# ac1 is verified for ac2.
|
# ac1 is verified for ac2.
|
||||||
ac2_contact_ac1 = ac2.create_contact(ac1.get_config("addr"), "")
|
ac2_contact_ac1 = ac2.create_contact(ac1, "")
|
||||||
assert ac2_contact_ac1.get_snapshot().is_verified
|
assert ac2_contact_ac1.get_snapshot().is_verified
|
||||||
|
|
||||||
# ac1 resetups the account.
|
# ac1 resetups the account.
|
||||||
ac1 = acfactory.resetup_account(ac1)
|
ac1 = acfactory.resetup_account(ac1)
|
||||||
|
ac2_contact_ac1 = ac2.create_contact(ac1, "")
|
||||||
# Loop sending message from ac1 to ac2
|
assert not ac2_contact_ac1.get_snapshot().is_verified
|
||||||
# until ac2 accepts new ac1 key.
|
|
||||||
#
|
|
||||||
# This may not happen immediately because resetup of ac1
|
|
||||||
# rewinds "smeared timestamp" so Date: header for messages
|
|
||||||
# sent by new ac1 are in the past compared to the last Date:
|
|
||||||
# header sent by old ac1.
|
|
||||||
while True:
|
|
||||||
# ac1 sends a message to ac2.
|
|
||||||
ac1_contact_ac2 = ac1.create_contact(ac2.get_config("addr"), "")
|
|
||||||
ac1_chat_ac2 = ac1_contact_ac2.create_chat()
|
|
||||||
ac1_chat_ac2.send_text("Hello!")
|
|
||||||
|
|
||||||
# ac2 receives a message.
|
|
||||||
snapshot = ac2.get_message_by_id(ac2.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
|
||||||
assert snapshot.text == "Hello!"
|
|
||||||
logging.info("ac2 received Hello!")
|
|
||||||
|
|
||||||
# ac1 is no longer verified for ac2 as new Autocrypt key is not the same as old verified key.
|
|
||||||
logging.info("ac2 addr={}, ac1 addr={}".format(ac2.get_config("addr"), ac1.get_config("addr")))
|
|
||||||
if not ac2_contact_ac1.get_snapshot().is_verified:
|
|
||||||
break
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
# ac1 goes offline.
|
# ac1 goes offline.
|
||||||
ac1.remove()
|
ac1.remove()
|
||||||
@@ -631,7 +660,6 @@ def test_securejoin_after_contact_resetup(acfactory) -> None:
|
|||||||
snapshot = ac2.get_message_by_id(ac2.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
snapshot = ac2.get_message_by_id(ac2.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
assert snapshot.is_info
|
assert snapshot.is_info
|
||||||
ac2_chat = snapshot.chat
|
ac2_chat = snapshot.chat
|
||||||
assert ac2_chat.get_basic_snapshot().is_protected
|
|
||||||
assert len(ac2_chat.get_contacts()) == 3
|
assert len(ac2_chat.get_contacts()) == 3
|
||||||
|
|
||||||
# ac1 is still "not verified" for ac2 due to inconsistent state.
|
# ac1 is still "not verified" for ac2 due to inconsistent state.
|
||||||
@@ -641,9 +669,8 @@ def test_securejoin_after_contact_resetup(acfactory) -> None:
|
|||||||
def test_withdraw_securejoin_qr(acfactory):
|
def test_withdraw_securejoin_qr(acfactory):
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
logging.info("Alice creates a verified group")
|
logging.info("Alice creates a group")
|
||||||
alice_chat = alice.create_group("Verified group", protect=True)
|
alice_chat = alice.create_group("Group")
|
||||||
assert alice_chat.get_basic_snapshot().is_protected
|
|
||||||
logging.info("Bob joins verified group")
|
logging.info("Bob joins verified group")
|
||||||
|
|
||||||
qr_code = alice_chat.get_qr_code()
|
qr_code = alice_chat.get_qr_code()
|
||||||
@@ -653,8 +680,7 @@ def test_withdraw_securejoin_qr(acfactory):
|
|||||||
alice.clear_all_events()
|
alice.clear_all_events()
|
||||||
|
|
||||||
snapshot = bob.get_message_by_id(bob.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
snapshot = bob.get_message_by_id(bob.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
assert snapshot.text == "Member Me ({}) added by {}.".format(bob.get_config("addr"), alice.get_config("addr"))
|
assert snapshot.text == "Member Me added by {}.".format(alice.get_config("addr"))
|
||||||
assert snapshot.chat.get_basic_snapshot().is_protected
|
|
||||||
bob_chat.leave()
|
bob_chat.leave()
|
||||||
|
|
||||||
snapshot = alice.get_message_by_id(alice.wait_for_msgs_changed_event().msg_id).get_snapshot()
|
snapshot = alice.get_message_by_id(alice.wait_for_msgs_changed_event().msg_id).get_snapshot()
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import pytest
|
|||||||
|
|
||||||
from deltachat_rpc_client import Contact, EventType, Message, events
|
from deltachat_rpc_client import Contact, EventType, Message, events
|
||||||
from deltachat_rpc_client.const import DownloadState, MessageState
|
from deltachat_rpc_client.const import DownloadState, MessageState
|
||||||
|
from deltachat_rpc_client.pytestplugin import E2EE_INFO_MSGS
|
||||||
from deltachat_rpc_client.rpc import JsonRpcError
|
from deltachat_rpc_client.rpc import JsonRpcError
|
||||||
|
|
||||||
|
|
||||||
@@ -61,52 +62,96 @@ def test_acfactory(acfactory) -> None:
|
|||||||
|
|
||||||
|
|
||||||
def test_configure_starttls(acfactory) -> None:
|
def test_configure_starttls(acfactory) -> None:
|
||||||
account = acfactory.new_preconfigured_account()
|
addr, password = acfactory.get_credentials()
|
||||||
|
account = acfactory.get_unconfigured_account()
|
||||||
# Use STARTTLS
|
account.add_or_update_transport(
|
||||||
account.set_config("mail_security", "2")
|
{
|
||||||
account.set_config("send_security", "2")
|
"addr": addr,
|
||||||
account.configure()
|
"password": password,
|
||||||
|
"imapSecurity": "starttls",
|
||||||
|
"smtpSecurity": "starttls",
|
||||||
|
},
|
||||||
|
)
|
||||||
assert account.is_configured()
|
assert account.is_configured()
|
||||||
|
|
||||||
|
|
||||||
|
def test_lowercase_address(acfactory) -> None:
|
||||||
|
addr, password = acfactory.get_credentials()
|
||||||
|
addr_upper = addr.upper()
|
||||||
|
account = acfactory.get_unconfigured_account()
|
||||||
|
account.add_or_update_transport(
|
||||||
|
{
|
||||||
|
"addr": addr_upper,
|
||||||
|
"password": password,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
assert account.is_configured()
|
||||||
|
assert addr_upper != addr
|
||||||
|
assert account.get_config("configured_addr") == addr
|
||||||
|
assert account.list_transports()[0]["addr"] == addr
|
||||||
|
|
||||||
|
for param in [
|
||||||
|
account.get_info()["used_account_settings"],
|
||||||
|
account.get_info()["entered_account_settings"],
|
||||||
|
]:
|
||||||
|
assert addr in param
|
||||||
|
assert addr_upper not in param
|
||||||
|
|
||||||
|
|
||||||
def test_configure_ip(acfactory) -> None:
|
def test_configure_ip(acfactory) -> None:
|
||||||
account = acfactory.new_preconfigured_account()
|
addr, password = acfactory.get_credentials()
|
||||||
|
account = acfactory.get_unconfigured_account()
|
||||||
|
ip_address = socket.gethostbyname(addr.rsplit("@")[-1])
|
||||||
|
|
||||||
domain = account.get_config("addr").rsplit("@")[-1]
|
|
||||||
ip_address = socket.gethostbyname(domain)
|
|
||||||
|
|
||||||
# This should fail TLS check.
|
|
||||||
account.set_config("mail_server", ip_address)
|
|
||||||
with pytest.raises(JsonRpcError):
|
with pytest.raises(JsonRpcError):
|
||||||
account.configure()
|
account.add_or_update_transport(
|
||||||
|
{
|
||||||
|
"addr": addr,
|
||||||
|
"password": password,
|
||||||
|
# This should fail TLS check.
|
||||||
|
"imapServer": ip_address,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def test_configure_alternative_port(acfactory) -> None:
|
def test_configure_alternative_port(acfactory) -> None:
|
||||||
"""Test that configuration with alternative port 443 works."""
|
"""Test that configuration with alternative port 443 works."""
|
||||||
account = acfactory.new_preconfigured_account()
|
addr, password = acfactory.get_credentials()
|
||||||
|
account = acfactory.get_unconfigured_account()
|
||||||
account.set_config("mail_port", "443")
|
account.add_or_update_transport(
|
||||||
account.set_config("send_port", "443")
|
{
|
||||||
|
"addr": addr,
|
||||||
account.configure()
|
"password": password,
|
||||||
|
"imapPort": 443,
|
||||||
|
"smtpPort": 443,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
assert account.is_configured()
|
||||||
|
|
||||||
|
|
||||||
def test_configure_username(acfactory) -> None:
|
def test_list_transports(acfactory) -> None:
|
||||||
account = acfactory.new_preconfigured_account()
|
addr, password = acfactory.get_credentials()
|
||||||
|
account = acfactory.get_unconfigured_account()
|
||||||
addr = account.get_config("addr")
|
account.add_or_update_transport(
|
||||||
account.set_config("mail_user", addr)
|
{
|
||||||
account.configure()
|
"addr": addr,
|
||||||
|
"password": password,
|
||||||
assert account.get_config("configured_mail_user") == addr
|
"imapUser": addr,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
transports = account.list_transports()
|
||||||
|
assert len(transports) == 1
|
||||||
|
params = transports[0]
|
||||||
|
assert params["addr"] == addr
|
||||||
|
assert params["password"] == password
|
||||||
|
assert params["imapUser"] == addr
|
||||||
|
|
||||||
|
|
||||||
def test_account(acfactory) -> None:
|
def test_account(acfactory) -> None:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
bob_addr = bob.get_config("addr")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_text("Hello!")
|
alice_chat_bob.send_text("Hello!")
|
||||||
|
|
||||||
@@ -126,7 +171,10 @@ def test_account(acfactory) -> None:
|
|||||||
assert alice.get_size()
|
assert alice.get_size()
|
||||||
assert alice.is_configured()
|
assert alice.is_configured()
|
||||||
assert not alice.get_avatar()
|
assert not alice.get_avatar()
|
||||||
assert alice.get_contact_by_addr(bob_addr) == alice_contact_bob
|
# get_contact_by_addr() can lookup a key contact by address:
|
||||||
|
bob_contact = alice.get_contact_by_addr(bob_addr).get_snapshot()
|
||||||
|
assert bob_contact.display_name == "Bob"
|
||||||
|
assert bob_contact.is_key_contact
|
||||||
assert alice.get_contacts()
|
assert alice.get_contacts()
|
||||||
assert alice.get_contacts(snapshot=True)
|
assert alice.get_contacts(snapshot=True)
|
||||||
assert alice.self_contact
|
assert alice.self_contact
|
||||||
@@ -171,8 +219,7 @@ def test_account(acfactory) -> None:
|
|||||||
def test_chat(acfactory) -> None:
|
def test_chat(acfactory) -> None:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_text("Hello!")
|
alice_chat_bob.send_text("Hello!")
|
||||||
|
|
||||||
@@ -205,6 +252,7 @@ def test_chat(acfactory) -> None:
|
|||||||
bob_chat_alice.get_encryption_info()
|
bob_chat_alice.get_encryption_info()
|
||||||
|
|
||||||
group = alice.create_group("test group")
|
group = alice.create_group("test group")
|
||||||
|
to_resend = group.send_text("will be resent")
|
||||||
group.add_contact(alice_contact_bob)
|
group.add_contact(alice_contact_bob)
|
||||||
group.get_qr_code()
|
group.get_qr_code()
|
||||||
|
|
||||||
@@ -216,6 +264,7 @@ def test_chat(acfactory) -> None:
|
|||||||
|
|
||||||
msg = group.send_message(text="hi")
|
msg = group.send_message(text="hi")
|
||||||
assert (msg.get_snapshot()).text == "hi"
|
assert (msg.get_snapshot()).text == "hi"
|
||||||
|
group.resend_messages([to_resend])
|
||||||
group.forward_messages([msg])
|
group.forward_messages([msg])
|
||||||
|
|
||||||
group.set_draft(text="test draft")
|
group.set_draft(text="test draft")
|
||||||
@@ -238,13 +287,12 @@ def test_contact(acfactory) -> None:
|
|||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
bob_addr = bob.get_config("addr")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
|
|
||||||
assert alice_contact_bob == alice.get_contact_by_id(alice_contact_bob.id)
|
assert alice_contact_bob == alice.get_contact_by_id(alice_contact_bob.id)
|
||||||
assert repr(alice_contact_bob)
|
assert repr(alice_contact_bob)
|
||||||
alice_contact_bob.block()
|
alice_contact_bob.block()
|
||||||
alice_contact_bob.unblock()
|
alice_contact_bob.unblock()
|
||||||
alice_contact_bob.reset_encryption()
|
|
||||||
alice_contact_bob.set_name("new name")
|
alice_contact_bob.set_name("new name")
|
||||||
alice_contact_bob.get_encryption_info()
|
alice_contact_bob.get_encryption_info()
|
||||||
snapshot = alice_contact_bob.get_snapshot()
|
snapshot = alice_contact_bob.get_snapshot()
|
||||||
@@ -255,8 +303,7 @@ def test_contact(acfactory) -> None:
|
|||||||
def test_message(acfactory) -> None:
|
def test_message(acfactory) -> None:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_text("Hello!")
|
alice_chat_bob.send_text("Hello!")
|
||||||
|
|
||||||
@@ -284,6 +331,36 @@ def test_message(acfactory) -> None:
|
|||||||
assert reactions == snapshot.reactions
|
assert reactions == snapshot.reactions
|
||||||
|
|
||||||
|
|
||||||
|
def test_receive_imf_failure(acfactory) -> None:
|
||||||
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
|
|
||||||
|
bob.set_config("fail_on_receiving_full_msg", "1")
|
||||||
|
alice_chat_bob.send_text("Hello!")
|
||||||
|
event = bob.wait_for_event(EventType.MSGS_CHANGED)
|
||||||
|
assert event.chat_id == bob.get_device_chat().id
|
||||||
|
msg_id = event.msg_id
|
||||||
|
message = bob.get_message_by_id(msg_id)
|
||||||
|
snapshot = message.get_snapshot()
|
||||||
|
assert (
|
||||||
|
snapshot.text == "❌ Failed to receive a message:"
|
||||||
|
" Condition failed: `!context.get_config_bool(Config::FailOnReceivingFullMsg).await?`."
|
||||||
|
" Please report this bug to delta@merlinux.eu or https://support.delta.chat/."
|
||||||
|
)
|
||||||
|
|
||||||
|
# The failed message doesn't break the IMAP loop.
|
||||||
|
bob.set_config("fail_on_receiving_full_msg", "0")
|
||||||
|
alice_chat_bob.send_text("Hello again!")
|
||||||
|
event = bob.wait_for_incoming_msg_event()
|
||||||
|
msg_id = event.msg_id
|
||||||
|
message = bob.get_message_by_id(msg_id)
|
||||||
|
snapshot = message.get_snapshot()
|
||||||
|
assert snapshot.text == "Hello again!"
|
||||||
|
assert snapshot.download_state == DownloadState.DONE
|
||||||
|
assert snapshot.error is None
|
||||||
|
|
||||||
|
|
||||||
def test_selfavatar_sync(acfactory, data, log) -> None:
|
def test_selfavatar_sync(acfactory, data, log) -> None:
|
||||||
alice = acfactory.get_online_account()
|
alice = acfactory.get_online_account()
|
||||||
|
|
||||||
@@ -314,8 +391,7 @@ def test_reaction_seen_on_another_dev(acfactory) -> None:
|
|||||||
alice2 = alice.clone()
|
alice2 = alice.clone()
|
||||||
alice2.start_io()
|
alice2.start_io()
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_text("Hello!")
|
alice_chat_bob.send_text("Hello!")
|
||||||
|
|
||||||
@@ -332,8 +408,7 @@ def test_reaction_seen_on_another_dev(acfactory) -> None:
|
|||||||
alice2.clear_all_events()
|
alice2.clear_all_events()
|
||||||
alice_chat_bob.mark_noticed()
|
alice_chat_bob.mark_noticed()
|
||||||
chat_id = alice2.wait_for_event(EventType.MSGS_NOTICED).chat_id
|
chat_id = alice2.wait_for_event(EventType.MSGS_NOTICED).chat_id
|
||||||
alice2_contact_bob = alice2.get_contact_by_addr(bob_addr)
|
alice2_chat_bob = alice2.create_chat(bob)
|
||||||
alice2_chat_bob = alice2_contact_bob.create_chat()
|
|
||||||
assert chat_id == alice2_chat_bob.id
|
assert chat_id == alice2_chat_bob.id
|
||||||
|
|
||||||
|
|
||||||
@@ -341,8 +416,7 @@ def test_is_bot(acfactory) -> None:
|
|||||||
"""Test that we can recognize messages submitted by bots."""
|
"""Test that we can recognize messages submitted by bots."""
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
|
|
||||||
# Alice becomes a bot.
|
# Alice becomes a bot.
|
||||||
@@ -401,26 +475,30 @@ def test_wait_next_messages(acfactory) -> None:
|
|||||||
alice = acfactory.new_configured_account()
|
alice = acfactory.new_configured_account()
|
||||||
|
|
||||||
# Create a bot account so it does not receive device messages in the beginning.
|
# Create a bot account so it does not receive device messages in the beginning.
|
||||||
bot = acfactory.new_preconfigured_account()
|
addr, password = acfactory.get_credentials()
|
||||||
|
bot = acfactory.get_unconfigured_account()
|
||||||
bot.set_config("bot", "1")
|
bot.set_config("bot", "1")
|
||||||
bot.configure()
|
bot.add_or_update_transport({"addr": addr, "password": password})
|
||||||
|
assert bot.is_configured()
|
||||||
|
|
||||||
# There are no old messages and the call returns immediately.
|
# There are no old messages and the call returns immediately.
|
||||||
assert not bot.wait_next_messages()
|
assert not bot.wait_next_messages()
|
||||||
|
|
||||||
with concurrent.futures.ThreadPoolExecutor(max_workers=1) as executor:
|
# Bot starts waiting for messages.
|
||||||
# Bot starts waiting for messages.
|
next_messages_task = bot.wait_next_messages.future()
|
||||||
next_messages_task = executor.submit(bot.wait_next_messages)
|
|
||||||
|
|
||||||
bot_addr = bot.get_config("addr")
|
alice_contact_bot = alice.create_contact(bot, "Bot")
|
||||||
alice_contact_bot = alice.create_contact(bot_addr, "Bot")
|
alice_chat_bot = alice_contact_bot.create_chat()
|
||||||
alice_chat_bot = alice_contact_bot.create_chat()
|
alice_chat_bot.send_text("Hello!")
|
||||||
alice_chat_bot.send_text("Hello!")
|
|
||||||
|
|
||||||
next_messages = next_messages_task.result()
|
next_messages = next_messages_task()
|
||||||
assert len(next_messages) == 1
|
|
||||||
snapshot = next_messages[0].get_snapshot()
|
if len(next_messages) == E2EE_INFO_MSGS:
|
||||||
assert snapshot.text == "Hello!"
|
next_messages += bot.wait_next_messages()
|
||||||
|
|
||||||
|
assert len(next_messages) == 1 + E2EE_INFO_MSGS
|
||||||
|
snapshot = next_messages[0 + E2EE_INFO_MSGS].get_snapshot()
|
||||||
|
assert snapshot.text == "Hello!"
|
||||||
|
|
||||||
|
|
||||||
def test_import_export_backup(acfactory, tmp_path) -> None:
|
def test_import_export_backup(acfactory, tmp_path) -> None:
|
||||||
@@ -437,9 +515,7 @@ def test_import_export_backup(acfactory, tmp_path) -> None:
|
|||||||
def test_import_export_keys(acfactory, tmp_path) -> None:
|
def test_import_export_keys(acfactory, tmp_path) -> None:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_chat_bob = alice.create_chat(bob)
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
|
||||||
alice_chat_bob.send_text("Hello Bob!")
|
alice_chat_bob.send_text("Hello Bob!")
|
||||||
|
|
||||||
snapshot = bob.get_message_by_id(bob.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
snapshot = bob.get_message_by_id(bob.wait_for_incoming_msg_event().msg_id).get_snapshot()
|
||||||
@@ -477,8 +553,11 @@ def test_provider_info(rpc) -> None:
|
|||||||
assert provider_info is None
|
assert provider_info is None
|
||||||
|
|
||||||
# Test MX record resolution.
|
# Test MX record resolution.
|
||||||
|
# This previously resulted in Gmail provider
|
||||||
|
# because MX record pointed to google.com domain,
|
||||||
|
# but MX record resolution has been removed.
|
||||||
provider_info = rpc.get_provider_info(account_id, "github.com")
|
provider_info = rpc.get_provider_info(account_id, "github.com")
|
||||||
assert provider_info["id"] == "gmail"
|
assert provider_info is None
|
||||||
|
|
||||||
# Disable MX record resolution.
|
# Disable MX record resolution.
|
||||||
rpc.set_config(account_id, "proxy_enabled", "1")
|
rpc.set_config(account_id, "proxy_enabled", "1")
|
||||||
@@ -489,9 +568,7 @@ def test_provider_info(rpc) -> None:
|
|||||||
def test_mdn_doesnt_break_autocrypt(acfactory) -> None:
|
def test_mdn_doesnt_break_autocrypt(acfactory) -> None:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
|
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
|
|
||||||
# Bob creates chat manually so chat with Alice is accepted.
|
# Bob creates chat manually so chat with Alice is accepted.
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
@@ -587,9 +664,13 @@ def test_reactions_for_a_reordering_move(acfactory, direct_imap):
|
|||||||
messages they refer to and thus dropped.
|
messages they refer to and thus dropped.
|
||||||
"""
|
"""
|
||||||
(ac1,) = acfactory.get_online_accounts(1)
|
(ac1,) = acfactory.get_online_accounts(1)
|
||||||
ac2 = acfactory.new_preconfigured_account()
|
|
||||||
ac2.configure()
|
addr, password = acfactory.get_credentials()
|
||||||
|
ac2 = acfactory.get_unconfigured_account()
|
||||||
|
ac2.add_or_update_transport({"addr": addr, "password": password})
|
||||||
ac2.set_config("mvbox_move", "1")
|
ac2.set_config("mvbox_move", "1")
|
||||||
|
assert ac2.is_configured()
|
||||||
|
|
||||||
ac2.bring_online()
|
ac2.bring_online()
|
||||||
chat1 = acfactory.get_accepted_chat(ac1, ac2)
|
chat1 = acfactory.get_accepted_chat(ac1, ac2)
|
||||||
ac2.stop_io()
|
ac2.stop_io()
|
||||||
@@ -633,9 +714,7 @@ def test_download_limit_chat_assignment(acfactory, tmp_path, n_accounts):
|
|||||||
chat.send_text("Hello Alice!")
|
chat.send_text("Hello Alice!")
|
||||||
assert alice.get_message_by_id(alice.wait_for_incoming_msg_event().msg_id).get_snapshot().text == "Hello Alice!"
|
assert alice.get_message_by_id(alice.wait_for_incoming_msg_event().msg_id).get_snapshot().text == "Hello Alice!"
|
||||||
|
|
||||||
contact_addr = account.get_config("addr")
|
contact = alice.create_contact(account)
|
||||||
contact = alice.create_contact(contact_addr, "")
|
|
||||||
|
|
||||||
alice_group.add_contact(contact)
|
alice_group.add_contact(contact)
|
||||||
|
|
||||||
if n_accounts == 2:
|
if n_accounts == 2:
|
||||||
@@ -689,6 +768,26 @@ def test_markseen_contact_request(acfactory):
|
|||||||
assert message2.get_snapshot().state == MessageState.IN_SEEN
|
assert message2.get_snapshot().state == MessageState.IN_SEEN
|
||||||
|
|
||||||
|
|
||||||
|
def test_read_receipt(acfactory):
|
||||||
|
"""
|
||||||
|
Test sending a read receipt and ensure it is attributed to the correct contact.
|
||||||
|
"""
|
||||||
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
|
alice_chat_bob = alice.create_chat(bob)
|
||||||
|
alice_contact_bob = alice.create_contact(bob)
|
||||||
|
bob.create_chat(alice) # Accept the chat
|
||||||
|
|
||||||
|
alice_chat_bob.send_text("Hello Bob!")
|
||||||
|
msg = bob.wait_for_incoming_msg()
|
||||||
|
msg.mark_seen()
|
||||||
|
|
||||||
|
read_msg = alice.get_message_by_id(alice.wait_for_event(EventType.MSG_READ).msg_id)
|
||||||
|
read_receipts = read_msg.get_read_receipts()
|
||||||
|
assert len(read_receipts) == 1
|
||||||
|
assert read_receipts[0].contact_id == alice_contact_bob.id
|
||||||
|
|
||||||
|
|
||||||
def test_get_http_response(acfactory):
|
def test_get_http_response(acfactory):
|
||||||
alice = acfactory.new_configured_account()
|
alice = acfactory.new_configured_account()
|
||||||
http_response = alice._rpc.get_http_response(alice.id, "https://example.org")
|
http_response = alice._rpc.get_http_response(alice.id, "https://example.org")
|
||||||
@@ -698,12 +797,11 @@ def test_get_http_response(acfactory):
|
|||||||
|
|
||||||
def test_configured_imap_certificate_checks(acfactory):
|
def test_configured_imap_certificate_checks(acfactory):
|
||||||
alice = acfactory.new_configured_account()
|
alice = acfactory.new_configured_account()
|
||||||
configured_certificate_checks = alice.get_config("configured_imap_certificate_checks")
|
|
||||||
|
|
||||||
# Certificate checks should be configured (not None)
|
# Certificate checks should be configured (not None)
|
||||||
assert configured_certificate_checks
|
assert "cert_strict" in alice.get_info().used_account_settings
|
||||||
|
|
||||||
# 0 is the value old Delta Chat core versions used
|
# "cert_old_automatic" is the value old Delta Chat core versions used
|
||||||
# to mean user entered "imap_certificate_checks=0" (Automatic)
|
# to mean user entered "imap_certificate_checks=0" (Automatic)
|
||||||
# and configuration failed to use strict TLS checks
|
# and configuration failed to use strict TLS checks
|
||||||
# so it switched strict TLS checks off.
|
# so it switched strict TLS checks off.
|
||||||
@@ -714,7 +812,7 @@ def test_configured_imap_certificate_checks(acfactory):
|
|||||||
#
|
#
|
||||||
# Core 1.142.4, 1.142.5 and 1.142.6 saved this value due to bug.
|
# Core 1.142.4, 1.142.5 and 1.142.6 saved this value due to bug.
|
||||||
# This test is a regression test to prevent this happening again.
|
# This test is a regression test to prevent this happening again.
|
||||||
assert configured_certificate_checks != "0"
|
assert "cert_old_automatic" not in alice.get_info().used_account_settings
|
||||||
|
|
||||||
|
|
||||||
def test_no_old_msg_is_fresh(acfactory):
|
def test_no_old_msg_is_fresh(acfactory):
|
||||||
@@ -742,3 +840,175 @@ def test_no_old_msg_is_fresh(acfactory):
|
|||||||
assert ev.chat_id == first_msg.get_snapshot().chat_id
|
assert ev.chat_id == first_msg.get_snapshot().chat_id
|
||||||
assert ac1.create_chat(ac2).get_fresh_message_count() == 0
|
assert ac1.create_chat(ac2).get_fresh_message_count() == 0
|
||||||
assert len(list(ac1.get_fresh_messages())) == 0
|
assert len(list(ac1.get_fresh_messages())) == 0
|
||||||
|
|
||||||
|
|
||||||
|
def test_rename_synchronization(acfactory):
|
||||||
|
"""Test synchronization of contact renaming."""
|
||||||
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
alice2 = alice.clone()
|
||||||
|
alice2.bring_online()
|
||||||
|
|
||||||
|
bob.set_config("displayname", "Bob")
|
||||||
|
bob.create_chat(alice).send_text("Hello!")
|
||||||
|
alice_msg = alice.wait_for_incoming_msg().get_snapshot()
|
||||||
|
alice2_msg = alice2.wait_for_incoming_msg().get_snapshot()
|
||||||
|
|
||||||
|
assert alice2_msg.sender.get_snapshot().display_name == "Bob"
|
||||||
|
alice_msg.sender.set_name("Bobby")
|
||||||
|
alice2.wait_for_event(EventType.CONTACTS_CHANGED)
|
||||||
|
assert alice2_msg.sender.get_snapshot().display_name == "Bobby"
|
||||||
|
|
||||||
|
|
||||||
|
def test_rename_group(acfactory):
|
||||||
|
"""Test renaming the group."""
|
||||||
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
|
alice_group = alice.create_group("Test group")
|
||||||
|
alice_contact_bob = alice.create_contact(bob)
|
||||||
|
alice_group.add_contact(alice_contact_bob)
|
||||||
|
alice_group.send_text("Hello!")
|
||||||
|
|
||||||
|
bob_msg = bob.wait_for_incoming_msg()
|
||||||
|
bob_chat = bob_msg.get_snapshot().chat
|
||||||
|
assert bob_chat.get_basic_snapshot().name == "Test group"
|
||||||
|
bob.wait_for_event(EventType.CHATLIST_ITEM_CHANGED)
|
||||||
|
|
||||||
|
for name in ["Baz", "Foo bar", "Xyzzy"]:
|
||||||
|
alice_group.set_name(name)
|
||||||
|
bob.wait_for_event(EventType.CHATLIST_ITEM_CHANGED)
|
||||||
|
bob.wait_for_event(EventType.CHATLIST_ITEM_CHANGED)
|
||||||
|
assert bob_chat.get_basic_snapshot().name == name
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_all_accounts_deadlock(rpc):
|
||||||
|
"""Regression test for get_all_accounts deadlock."""
|
||||||
|
for _ in range(100):
|
||||||
|
all_accounts = rpc.get_all_accounts.future()
|
||||||
|
rpc.add_account()
|
||||||
|
all_accounts()
|
||||||
|
|
||||||
|
|
||||||
|
def test_delete_deltachat_folder(acfactory, direct_imap):
|
||||||
|
"""Test that DeltaChat folder is recreated if user deletes it manually."""
|
||||||
|
ac1 = acfactory.new_configured_account()
|
||||||
|
ac1.set_config("mvbox_move", "1")
|
||||||
|
ac1.bring_online()
|
||||||
|
|
||||||
|
ac1_direct_imap = direct_imap(ac1)
|
||||||
|
ac1_direct_imap.conn.folder.delete("DeltaChat")
|
||||||
|
assert "DeltaChat" not in ac1_direct_imap.list_folders()
|
||||||
|
|
||||||
|
# Wait until new folder is created and UIDVALIDITY is updated.
|
||||||
|
while True:
|
||||||
|
event = ac1.wait_for_event()
|
||||||
|
if event.kind == EventType.INFO and "uid/validity change folder DeltaChat" in event.msg:
|
||||||
|
break
|
||||||
|
|
||||||
|
ac2 = acfactory.get_online_account()
|
||||||
|
ac2.create_chat(ac1).send_text("hello")
|
||||||
|
msg = ac1.wait_for_incoming_msg().get_snapshot()
|
||||||
|
assert msg.text == "hello"
|
||||||
|
|
||||||
|
assert "DeltaChat" in ac1_direct_imap.list_folders()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("all_devices_online", [True, False])
|
||||||
|
def test_leave_broadcast(acfactory, all_devices_online):
|
||||||
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
|
bob2 = bob.clone()
|
||||||
|
|
||||||
|
if all_devices_online:
|
||||||
|
bob2.start_io()
|
||||||
|
|
||||||
|
logging.info("===================== Alice creates a broadcast =====================")
|
||||||
|
alice_chat = alice.create_broadcast("Broadcast channel!")
|
||||||
|
|
||||||
|
logging.info("===================== Bob joins the broadcast =====================")
|
||||||
|
qr_code = alice_chat.get_qr_code()
|
||||||
|
bob.secure_join(qr_code)
|
||||||
|
alice.wait_for_securejoin_inviter_success()
|
||||||
|
bob.wait_for_securejoin_joiner_success()
|
||||||
|
|
||||||
|
alice_bob_contact = alice.create_contact(bob)
|
||||||
|
alice_contacts = alice_chat.get_contacts()
|
||||||
|
assert len(alice_contacts) == 1 # 1 recipient
|
||||||
|
assert alice_contacts[0].id == alice_bob_contact.id
|
||||||
|
|
||||||
|
member_added_msg = bob.wait_for_incoming_msg()
|
||||||
|
assert member_added_msg.get_snapshot().text == "You joined the channel."
|
||||||
|
|
||||||
|
def get_broadcast(ac):
|
||||||
|
chat = ac.get_chatlist(query="Broadcast channel!")[0]
|
||||||
|
assert chat.get_basic_snapshot().name == "Broadcast channel!"
|
||||||
|
return chat
|
||||||
|
|
||||||
|
def check_account(ac, contact, inviter_side, please_wait_info_msg=False):
|
||||||
|
chat = get_broadcast(ac)
|
||||||
|
contact_snapshot = contact.get_snapshot()
|
||||||
|
chat_msgs = chat.get_messages()
|
||||||
|
|
||||||
|
if please_wait_info_msg:
|
||||||
|
first_msg = chat_msgs.pop(0).get_snapshot()
|
||||||
|
assert first_msg.text == "Establishing guaranteed end-to-end encryption, please wait…"
|
||||||
|
assert first_msg.is_info
|
||||||
|
|
||||||
|
encrypted_msg = chat_msgs.pop(0).get_snapshot()
|
||||||
|
assert encrypted_msg.text == "Messages are end-to-end encrypted."
|
||||||
|
assert encrypted_msg.is_info
|
||||||
|
|
||||||
|
member_added_msg = chat_msgs.pop(0).get_snapshot()
|
||||||
|
if inviter_side:
|
||||||
|
assert member_added_msg.text == f"Member {contact_snapshot.display_name} added."
|
||||||
|
else:
|
||||||
|
assert member_added_msg.text == "You joined the channel."
|
||||||
|
assert member_added_msg.is_info
|
||||||
|
|
||||||
|
if not inviter_side:
|
||||||
|
leave_msg = chat_msgs.pop(0).get_snapshot()
|
||||||
|
assert leave_msg.text == "You left the channel."
|
||||||
|
|
||||||
|
assert len(chat_msgs) == 0
|
||||||
|
|
||||||
|
chat_snapshot = chat.get_full_snapshot()
|
||||||
|
|
||||||
|
# On Alice's side, SELF is not in the list of contact ids
|
||||||
|
# because OutBroadcast chats never contain SELF in the list.
|
||||||
|
# On Bob's side, SELF is not in the list because he left.
|
||||||
|
if inviter_side:
|
||||||
|
assert len(chat_snapshot.contact_ids) == 0
|
||||||
|
else:
|
||||||
|
assert chat_snapshot.contact_ids == [contact.id]
|
||||||
|
|
||||||
|
logging.info("===================== Bob leaves the broadcast =====================")
|
||||||
|
bob_chat = get_broadcast(bob)
|
||||||
|
assert bob_chat.get_full_snapshot().self_in_group
|
||||||
|
assert len(bob_chat.get_contacts()) == 2 # Alice and Bob
|
||||||
|
|
||||||
|
bob_chat.leave()
|
||||||
|
assert not bob_chat.get_full_snapshot().self_in_group
|
||||||
|
# After Bob left, only Alice will be left in Bob's memberlist
|
||||||
|
assert len(bob_chat.get_contacts()) == 1
|
||||||
|
|
||||||
|
check_account(bob, bob.create_contact(alice), inviter_side=False, please_wait_info_msg=True)
|
||||||
|
|
||||||
|
logging.info("===================== Test Alice's device =====================")
|
||||||
|
while len(alice_chat.get_contacts()) != 0: # After Bob left, there will be 0 recipients
|
||||||
|
alice.wait_for_event(EventType.CHAT_MODIFIED)
|
||||||
|
|
||||||
|
check_account(alice, alice.create_contact(bob), inviter_side=True)
|
||||||
|
|
||||||
|
logging.info("===================== Test Bob's second device =====================")
|
||||||
|
# Start second Bob device, if it wasn't started already.
|
||||||
|
bob2.start_io()
|
||||||
|
|
||||||
|
member_added_msg = bob2.wait_for_incoming_msg()
|
||||||
|
assert member_added_msg.get_snapshot().text == "You joined the channel."
|
||||||
|
|
||||||
|
bob2_chat = get_broadcast(bob2)
|
||||||
|
|
||||||
|
# After Bob left, only Alice will be left in Bob's memberlist
|
||||||
|
while len(bob2_chat.get_contacts()) != 1:
|
||||||
|
bob2.wait_for_event(EventType.CHAT_MODIFIED)
|
||||||
|
|
||||||
|
check_account(bob2, bob2.create_contact(alice), inviter_side=False)
|
||||||
|
|||||||
@@ -1,9 +1,12 @@
|
|||||||
def test_vcard(acfactory) -> None:
|
def test_vcard(acfactory) -> None:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob, fiona = acfactory.get_online_accounts(3)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
bob.create_chat(alice)
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_charlie = alice.create_contact("charlie@example.org", "Charlie")
|
alice_contact_charlie = alice.create_contact("charlie@example.org", "Charlie")
|
||||||
|
alice_contact_charlie_snapshot = alice_contact_charlie.get_snapshot()
|
||||||
|
alice_contact_fiona = alice.create_contact(fiona, "Fiona")
|
||||||
|
alice_contact_fiona_snapshot = alice_contact_fiona.get_snapshot()
|
||||||
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_contact(alice_contact_charlie)
|
alice_chat_bob.send_contact(alice_contact_charlie)
|
||||||
@@ -13,3 +16,12 @@ def test_vcard(acfactory) -> None:
|
|||||||
snapshot = message.get_snapshot()
|
snapshot = message.get_snapshot()
|
||||||
assert snapshot.vcard_contact
|
assert snapshot.vcard_contact
|
||||||
assert snapshot.vcard_contact.addr == "charlie@example.org"
|
assert snapshot.vcard_contact.addr == "charlie@example.org"
|
||||||
|
assert snapshot.vcard_contact.color == alice_contact_charlie_snapshot.color
|
||||||
|
|
||||||
|
alice_chat_bob.send_contact(alice_contact_fiona)
|
||||||
|
event = bob.wait_for_incoming_msg_event()
|
||||||
|
message = bob.get_message_by_id(event.msg_id)
|
||||||
|
snapshot = message.get_snapshot()
|
||||||
|
assert snapshot.vcard_contact
|
||||||
|
assert snapshot.vcard_contact.key
|
||||||
|
assert snapshot.vcard_contact.color == alice_contact_fiona_snapshot.color
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
def test_webxdc(acfactory) -> None:
|
def test_webxdc(acfactory) -> None:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
alice_chat_bob.send_message(text="Let's play chess!", file="../test-data/webxdc/chess.xdc")
|
alice_chat_bob.send_message(text="Let's play chess!", file="../test-data/webxdc/chess.xdc")
|
||||||
|
|
||||||
@@ -45,8 +44,7 @@ def test_webxdc(acfactory) -> None:
|
|||||||
def test_webxdc_insert_lots_of_updates(acfactory) -> None:
|
def test_webxdc_insert_lots_of_updates(acfactory) -> None:
|
||||||
alice, bob = acfactory.get_online_accounts(2)
|
alice, bob = acfactory.get_online_accounts(2)
|
||||||
|
|
||||||
bob_addr = bob.get_config("addr")
|
alice_contact_bob = alice.create_contact(bob, "Bob")
|
||||||
alice_contact_bob = alice.create_contact(bob_addr, "Bob")
|
|
||||||
alice_chat_bob = alice_contact_bob.create_chat()
|
alice_chat_bob = alice_contact_bob.create_chat()
|
||||||
message = alice_chat_bob.send_message(text="Let's play chess!", file="../test-data/webxdc/chess.xdc")
|
message = alice_chat_bob.send_message(text="Let's play chess!", file="../test-data/webxdc/chess.xdc")
|
||||||
|
|
||||||
|
|||||||
@@ -12,11 +12,8 @@ setenv =
|
|||||||
RUST_MIN_STACK=8388608
|
RUST_MIN_STACK=8388608
|
||||||
passenv =
|
passenv =
|
||||||
CHATMAIL_DOMAIN
|
CHATMAIL_DOMAIN
|
||||||
deps =
|
dependency_groups =
|
||||||
pytest
|
dev
|
||||||
pytest-timeout
|
|
||||||
pytest-xdist
|
|
||||||
imap-tools
|
|
||||||
|
|
||||||
[testenv:lint]
|
[testenv:lint]
|
||||||
skipsdist = True
|
skipsdist = True
|
||||||
@@ -24,7 +21,7 @@ skip_install = True
|
|||||||
deps =
|
deps =
|
||||||
ruff
|
ruff
|
||||||
commands =
|
commands =
|
||||||
ruff format --quiet --diff src/ examples/ tests/
|
ruff format --diff src/ examples/ tests/
|
||||||
ruff check src/ examples/ tests/
|
ruff check src/ examples/ tests/
|
||||||
|
|
||||||
[pytest]
|
[pytest]
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "deltachat-rpc-server"
|
name = "deltachat-rpc-server"
|
||||||
version = "1.157.3"
|
version = "2.25.0"
|
||||||
description = "DeltaChat JSON-RPC server"
|
description = "DeltaChat JSON-RPC server"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
|
|||||||
@@ -15,5 +15,5 @@
|
|||||||
},
|
},
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"types": "index.d.ts",
|
"types": "index.d.ts",
|
||||||
"version": "1.157.3"
|
"version": "2.25.0"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ async fn main() {
|
|||||||
// thread, and it is impossible to cancel that read. This can make shutdown of the runtime hang
|
// thread, and it is impossible to cancel that read. This can make shutdown of the runtime hang
|
||||||
// until the user presses enter."
|
// until the user presses enter."
|
||||||
if let Err(error) = &r {
|
if let Err(error) = &r {
|
||||||
log::error!("Fatal error: {error:#}.")
|
log::error!("Error: {error:#}.")
|
||||||
}
|
}
|
||||||
std::process::exit(if r.is_ok() { 0 } else { 1 });
|
std::process::exit(if r.is_ok() { 0 } else { 1 });
|
||||||
}
|
}
|
||||||
@@ -41,22 +41,22 @@ async fn main_impl() -> Result<()> {
|
|||||||
if let Some(first_arg) = args.next() {
|
if let Some(first_arg) = args.next() {
|
||||||
if first_arg.to_str() == Some("--version") {
|
if first_arg.to_str() == Some("--version") {
|
||||||
if let Some(arg) = args.next() {
|
if let Some(arg) = args.next() {
|
||||||
return Err(anyhow!("Unrecognized argument {:?}", arg));
|
return Err(anyhow!("Unrecognized argument {arg:?}"));
|
||||||
}
|
}
|
||||||
eprintln!("{}", &*DC_VERSION_STR);
|
eprintln!("{}", &*DC_VERSION_STR);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
} else if first_arg.to_str() == Some("--openrpc") {
|
} else if first_arg.to_str() == Some("--openrpc") {
|
||||||
if let Some(arg) = args.next() {
|
if let Some(arg) = args.next() {
|
||||||
return Err(anyhow!("Unrecognized argument {:?}", arg));
|
return Err(anyhow!("Unrecognized argument {arg:?}"));
|
||||||
}
|
}
|
||||||
println!("{}", CommandApi::openrpc_specification()?);
|
println!("{}", CommandApi::openrpc_specification()?);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
} else {
|
} else {
|
||||||
return Err(anyhow!("Unrecognized option {:?}", first_arg));
|
return Err(anyhow!("Unrecognized option {first_arg:?}"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if let Some(arg) = args.next() {
|
if let Some(arg) = args.next() {
|
||||||
return Err(anyhow!("Unrecognized argument {:?}", arg));
|
return Err(anyhow!("Unrecognized argument {arg:?}"));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Install signal handlers early so that the shutdown is graceful starting from here.
|
// Install signal handlers early so that the shutdown is graceful starting from here.
|
||||||
@@ -73,7 +73,7 @@ async fn main_impl() -> Result<()> {
|
|||||||
.init();
|
.init();
|
||||||
|
|
||||||
let path = std::env::var("DC_ACCOUNTS_PATH").unwrap_or_else(|_| "accounts".to_string());
|
let path = std::env::var("DC_ACCOUNTS_PATH").unwrap_or_else(|_| "accounts".to_string());
|
||||||
log::info!("Starting with accounts directory `{}`.", path);
|
log::info!("Starting with accounts directory `{path}`.");
|
||||||
let writable = true;
|
let writable = true;
|
||||||
let accounts = Accounts::new(PathBuf::from(&path), writable).await?;
|
let accounts = Accounts::new(PathBuf::from(&path), writable).await?;
|
||||||
|
|
||||||
@@ -97,7 +97,7 @@ async fn main_impl() -> Result<()> {
|
|||||||
Some(message) => serde_json::to_string(&message)?,
|
Some(message) => serde_json::to_string(&message)?,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
log::trace!("RPC send {}", message);
|
log::trace!("RPC send {message}");
|
||||||
println!("{message}");
|
println!("{message}");
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -141,7 +141,7 @@ async fn main_impl() -> Result<()> {
|
|||||||
Some(message) => message,
|
Some(message) => message,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
log::trace!("RPC recv {}", message);
|
log::trace!("RPC recv {message}");
|
||||||
let session = session.clone();
|
let session = session.clone();
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
session.handle_incoming(&message).await;
|
session.handle_incoming(&message).await;
|
||||||
|
|||||||
@@ -20,6 +20,11 @@ impl SystemTimeTools {
|
|||||||
pub fn shift(duration: Duration) {
|
pub fn shift(duration: Duration) {
|
||||||
*SYSTEM_TIME_SHIFT.write().unwrap() += duration;
|
*SYSTEM_TIME_SHIFT.write().unwrap() += duration;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Simulates the system clock being rewound by `duration`.
|
||||||
|
pub fn shift_back(duration: Duration) {
|
||||||
|
*SYSTEM_TIME_SHIFT.write().unwrap() -= duration;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
|||||||
28
deny.toml
28
deny.toml
@@ -10,9 +10,6 @@ ignore = [
|
|||||||
# Unmaintained instant
|
# Unmaintained instant
|
||||||
"RUSTSEC-2024-0384",
|
"RUSTSEC-2024-0384",
|
||||||
|
|
||||||
# Unmaintained backoff
|
|
||||||
"RUSTSEC-2025-0012",
|
|
||||||
|
|
||||||
# Unmaintained paste
|
# Unmaintained paste
|
||||||
"RUSTSEC-2024-0436",
|
"RUSTSEC-2024-0436",
|
||||||
]
|
]
|
||||||
@@ -24,34 +21,30 @@ ignore = [
|
|||||||
# Please keep this list alphabetically sorted.
|
# Please keep this list alphabetically sorted.
|
||||||
skip = [
|
skip = [
|
||||||
{ name = "async-channel", version = "1.9.0" },
|
{ name = "async-channel", version = "1.9.0" },
|
||||||
{ name = "base64", version = "<0.21" },
|
|
||||||
{ name = "base64", version = "0.21.7" },
|
|
||||||
{ name = "bitflags", version = "1.3.2" },
|
{ name = "bitflags", version = "1.3.2" },
|
||||||
{ name = "core-foundation", version = "0.9.4" },
|
{ name = "derive_more-impl", version = "1.0.0" },
|
||||||
|
{ name = "derive_more", version = "1.0.0" },
|
||||||
{ name = "event-listener", version = "2.5.3" },
|
{ name = "event-listener", version = "2.5.3" },
|
||||||
{ name = "generator", version = "0.7.5" },
|
|
||||||
{ name = "getrandom", version = "0.2.12" },
|
{ name = "getrandom", version = "0.2.12" },
|
||||||
|
{ name = "hashbrown", version = "0.14.5" },
|
||||||
{ name = "heck", version = "0.4.1" },
|
{ name = "heck", version = "0.4.1" },
|
||||||
{ name = "http", version = "0.2.12" },
|
{ name = "http", version = "0.2.12" },
|
||||||
{ name = "loom", version = "0.5.6" },
|
{ name = "linux-raw-sys", version = "0.4.14" },
|
||||||
|
{ name = "lru", version = "0.12.3" },
|
||||||
{ name = "netlink-packet-route", version = "0.17.1" },
|
{ name = "netlink-packet-route", version = "0.17.1" },
|
||||||
{ name = "nix", version = "0.26.4" },
|
{ name = "nom", version = "7.1.3" },
|
||||||
{ name = "nix", version = "0.27.1" },
|
|
||||||
{ name = "quick-error", version = "<2.0" },
|
|
||||||
{ name = "rand_chacha", version = "0.3.1" },
|
{ name = "rand_chacha", version = "0.3.1" },
|
||||||
{ name = "rand_core", version = "0.6.4" },
|
{ name = "rand_core", version = "0.6.4" },
|
||||||
{ name = "rand", version = "0.8.5" },
|
{ name = "rand", version = "0.8.5" },
|
||||||
{ name = "redox_syscall", version = "0.3.5" },
|
{ name = "rustix", version = "0.38.44" },
|
||||||
{ name = "regex-automata", version = "0.1.10" },
|
{ name = "serdect", version = "0.2.0" },
|
||||||
{ name = "regex-syntax", version = "0.6.29" },
|
{ name = "spin", version = "0.9.8" },
|
||||||
{ name = "rtnetlink", version = "0.13.1" },
|
|
||||||
{ name = "security-framework", version = "2.11.1" },
|
|
||||||
{ name = "strum_macros", version = "0.26.2" },
|
{ name = "strum_macros", version = "0.26.2" },
|
||||||
{ name = "strum", version = "0.26.2" },
|
{ name = "strum", version = "0.26.2" },
|
||||||
{ name = "syn", version = "1.0.109" },
|
{ name = "syn", version = "1.0.109" },
|
||||||
{ name = "thiserror-impl", version = "1.0.69" },
|
{ name = "thiserror-impl", version = "1.0.69" },
|
||||||
{ name = "thiserror", version = "1.0.69" },
|
{ name = "thiserror", version = "1.0.69" },
|
||||||
{ name = "unicode-width", version = "0.1.11" },
|
{ name = "toml_datetime", version = "0.6.11" },
|
||||||
{ name = "wasi", version = "0.11.0+wasi-snapshot-preview1" },
|
{ name = "wasi", version = "0.11.0+wasi-snapshot-preview1" },
|
||||||
{ name = "windows" },
|
{ name = "windows" },
|
||||||
{ name = "windows_aarch64_gnullvm" },
|
{ name = "windows_aarch64_gnullvm" },
|
||||||
@@ -86,6 +79,7 @@ allow = [
|
|||||||
"MPL-2.0",
|
"MPL-2.0",
|
||||||
"Unicode-3.0",
|
"Unicode-3.0",
|
||||||
"Unicode-DFS-2016",
|
"Unicode-DFS-2016",
|
||||||
|
"Unlicense",
|
||||||
"Zlib",
|
"Zlib",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|||||||
24
flake.lock
generated
24
flake.lock
generated
@@ -47,11 +47,11 @@
|
|||||||
"rust-analyzer-src": "rust-analyzer-src"
|
"rust-analyzer-src": "rust-analyzer-src"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1737527504,
|
"lastModified": 1747291057,
|
||||||
"narHash": "sha256-Z8S5gLPdIYeKwBXDaSxlJ72ZmiilYhu3418h3RSQZA0=",
|
"narHash": "sha256-9Wir6aLJAeJKqdoQUiwfKdBn7SyNXTJGRSscRyVOo2Y=",
|
||||||
"owner": "nix-community",
|
"owner": "nix-community",
|
||||||
"repo": "fenix",
|
"repo": "fenix",
|
||||||
"rev": "aa13f23e3e91b95377a693ac655bbc6545ebec0d",
|
"rev": "76ffc1b7b3ec8078fe01794628b6abff35cbda8f",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -147,11 +147,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs_2": {
|
"nixpkgs_2": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1737469691,
|
"lastModified": 1747179050,
|
||||||
"narHash": "sha256-nmKOgAU48S41dTPIXAq0AHZSehWUn6ZPrUKijHAMmIk=",
|
"narHash": "sha256-qhFMmDkeJX9KJwr5H32f1r7Prs7XbQWtO0h3V0a0rFY=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "9e4d5190a9482a1fb9d18adf0bdb83c6e506eaab",
|
"rev": "adaa24fbf46737f3f1b5497bf64bae750f82942e",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -175,11 +175,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs_4": {
|
"nixpkgs_4": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1731139594,
|
"lastModified": 1747179050,
|
||||||
"narHash": "sha256-IigrKK3vYRpUu+HEjPL/phrfh7Ox881er1UEsZvw9Q4=",
|
"narHash": "sha256-qhFMmDkeJX9KJwr5H32f1r7Prs7XbQWtO0h3V0a0rFY=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "76612b17c0ce71689921ca12d9ffdc9c23ce40b2",
|
"rev": "adaa24fbf46737f3f1b5497bf64bae750f82942e",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -202,11 +202,11 @@
|
|||||||
"rust-analyzer-src": {
|
"rust-analyzer-src": {
|
||||||
"flake": false,
|
"flake": false,
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1737453499,
|
"lastModified": 1746889290,
|
||||||
"narHash": "sha256-fa5AJI9mjFU2oVXqdCq2oA2pripAXbHzkUkewJRQpxA=",
|
"narHash": "sha256-h3LQYZgyv2l3U7r+mcsrEOGRldaK0zJFwAAva4hV/6g=",
|
||||||
"owner": "rust-lang",
|
"owner": "rust-lang",
|
||||||
"repo": "rust-analyzer",
|
"repo": "rust-analyzer",
|
||||||
"rev": "0b68402d781955d526b80e5d479e9e47addb4075",
|
"rev": "2bafe9d96c6734aacfd49e115f6cf61e7adc68bc",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
|||||||
16
flake.nix
16
flake.nix
@@ -98,9 +98,6 @@
|
|||||||
nativeBuildInputs = [
|
nativeBuildInputs = [
|
||||||
pkgs.perl # Needed to build vendored OpenSSL.
|
pkgs.perl # Needed to build vendored OpenSSL.
|
||||||
];
|
];
|
||||||
buildInputs = pkgs.lib.optionals isDarwin [
|
|
||||||
pkgs.darwin.apple_sdk.frameworks.SystemConfiguration
|
|
||||||
];
|
|
||||||
auditable = false; # Avoid cargo-auditable failures.
|
auditable = false; # Avoid cargo-auditable failures.
|
||||||
doCheck = false; # Disable test as it requires network access.
|
doCheck = false; # Disable test as it requires network access.
|
||||||
};
|
};
|
||||||
@@ -240,6 +237,9 @@
|
|||||||
auditable = false; # Avoid cargo-auditable failures.
|
auditable = false; # Avoid cargo-auditable failures.
|
||||||
doCheck = false; # Disable test as it requires network access.
|
doCheck = false; # Disable test as it requires network access.
|
||||||
|
|
||||||
|
CARGO_TARGET_X86_64_APPLE_DARWIN_RUSTFLAGS = "-Clink-args=-L${pkgsCross.libiconv}/lib";
|
||||||
|
CARGO_TARGET_AARCH64_APPLE_DARWIN_RUSTFLAGS = "-Clink-args=-L${pkgsCross.libiconv}/lib";
|
||||||
|
|
||||||
CARGO_BUILD_TARGET = rustTarget;
|
CARGO_BUILD_TARGET = rustTarget;
|
||||||
TARGET_CC = "${pkgsCross.stdenv.cc}/bin/${pkgsCross.stdenv.cc.targetPrefix}cc";
|
TARGET_CC = "${pkgsCross.stdenv.cc}/bin/${pkgsCross.stdenv.cc.targetPrefix}cc";
|
||||||
CARGO_BUILD_RUSTFLAGS = [
|
CARGO_BUILD_RUSTFLAGS = [
|
||||||
@@ -483,12 +483,6 @@
|
|||||||
pkgs.rustPlatform.cargoSetupHook
|
pkgs.rustPlatform.cargoSetupHook
|
||||||
pkgs.cargo
|
pkgs.cargo
|
||||||
];
|
];
|
||||||
buildInputs = pkgs.lib.optionals isDarwin [
|
|
||||||
pkgs.darwin.apple_sdk.frameworks.CoreFoundation
|
|
||||||
pkgs.darwin.apple_sdk.frameworks.Security
|
|
||||||
pkgs.darwin.apple_sdk.frameworks.SystemConfiguration
|
|
||||||
pkgs.libiconv
|
|
||||||
];
|
|
||||||
|
|
||||||
postInstall = ''
|
postInstall = ''
|
||||||
substituteInPlace $out/include/deltachat.h \
|
substituteInPlace $out/include/deltachat.h \
|
||||||
@@ -584,6 +578,10 @@
|
|||||||
cargo-nextest
|
cargo-nextest
|
||||||
perl # needed to build vendored OpenSSL
|
perl # needed to build vendored OpenSSL
|
||||||
git-cliff
|
git-cliff
|
||||||
|
(python3.withPackages (pypkgs: with pypkgs; [
|
||||||
|
tox
|
||||||
|
]))
|
||||||
|
nodejs
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ edition = "2021"
|
|||||||
license = "MPL-2.0"
|
license = "MPL-2.0"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
bolero = "0.8"
|
bolero = "0.13.4"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
mailparse = { workspace = true }
|
mailparse = { workspace = true }
|
||||||
|
|||||||
@@ -1,49 +0,0 @@
|
|||||||
# content of group_tracking.py
|
|
||||||
|
|
||||||
from deltachat import account_hookimpl, run_cmdline
|
|
||||||
|
|
||||||
|
|
||||||
class GroupTrackingPlugin:
|
|
||||||
@account_hookimpl
|
|
||||||
def ac_incoming_message(self, message):
|
|
||||||
print("process_incoming message", message)
|
|
||||||
if message.text.strip() == "/quit":
|
|
||||||
message.account.shutdown()
|
|
||||||
else:
|
|
||||||
# unconditionally accept the chat
|
|
||||||
message.create_chat()
|
|
||||||
addr = message.get_sender_contact().addr
|
|
||||||
text = message.text
|
|
||||||
message.chat.send_text(f"echoing from {addr}:\n{text}")
|
|
||||||
|
|
||||||
@account_hookimpl
|
|
||||||
def ac_outgoing_message(self, message):
|
|
||||||
print("ac_outgoing_message:", message)
|
|
||||||
|
|
||||||
@account_hookimpl
|
|
||||||
def ac_configure_completed(self, success):
|
|
||||||
print("ac_configure_completed:", success)
|
|
||||||
|
|
||||||
@account_hookimpl
|
|
||||||
def ac_chat_modified(self, chat):
|
|
||||||
print("ac_chat_modified:", chat.id, chat.get_name())
|
|
||||||
for member in chat.get_contacts():
|
|
||||||
print(f"chat member: {member.addr}")
|
|
||||||
|
|
||||||
@account_hookimpl
|
|
||||||
def ac_member_added(self, chat, contact, actor, message):
|
|
||||||
print(f"ac_member_added {contact.addr} to chat {chat.id} from {actor or message.get_sender_contact().addr}")
|
|
||||||
for member in chat.get_contacts():
|
|
||||||
print(f"chat member: {member.addr}")
|
|
||||||
|
|
||||||
@account_hookimpl
|
|
||||||
def ac_member_removed(self, chat, contact, actor, message):
|
|
||||||
print(f"ac_member_removed {contact.addr} from chat {chat.id} by {actor or message.get_sender_contact().addr}")
|
|
||||||
|
|
||||||
|
|
||||||
def main(argv=None):
|
|
||||||
run_cmdline(argv=argv, account_plugins=[GroupTrackingPlugin()])
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
||||||
@@ -1,10 +1,7 @@
|
|||||||
import echo_and_quit
|
import echo_and_quit
|
||||||
import group_tracking
|
|
||||||
import py
|
import py
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from deltachat.events import FFIEventLogger
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="session")
|
@pytest.fixture(scope="session")
|
||||||
def datadir():
|
def datadir():
|
||||||
@@ -25,8 +22,8 @@ def test_echo_quit_plugin(acfactory, lp):
|
|||||||
(ac1,) = acfactory.get_online_accounts(1)
|
(ac1,) = acfactory.get_online_accounts(1)
|
||||||
|
|
||||||
lp.sec("sending a message to the bot")
|
lp.sec("sending a message to the bot")
|
||||||
bot_contact = ac1.create_contact(botproc.addr)
|
bot_chat = ac1.qr_setup_contact(botproc.qr)
|
||||||
bot_chat = bot_contact.create_chat()
|
ac1._evtracker.wait_securejoin_joiner_progress(1000)
|
||||||
bot_chat.send_text("hello")
|
bot_chat.send_text("hello")
|
||||||
|
|
||||||
lp.sec("waiting for the reply message from the bot to arrive")
|
lp.sec("waiting for the reply message from the bot to arrive")
|
||||||
@@ -36,53 +33,3 @@ def test_echo_quit_plugin(acfactory, lp):
|
|||||||
lp.sec("send quit sequence")
|
lp.sec("send quit sequence")
|
||||||
bot_chat.send_text("/quit")
|
bot_chat.send_text("/quit")
|
||||||
botproc.wait()
|
botproc.wait()
|
||||||
|
|
||||||
|
|
||||||
def test_group_tracking_plugin(acfactory, lp):
|
|
||||||
lp.sec("creating one group-tracking bot and two temp accounts")
|
|
||||||
botproc = acfactory.run_bot_process(group_tracking)
|
|
||||||
|
|
||||||
ac1, ac2 = acfactory.get_online_accounts(2)
|
|
||||||
|
|
||||||
ac1.add_account_plugin(FFIEventLogger(ac1))
|
|
||||||
ac2.add_account_plugin(FFIEventLogger(ac2))
|
|
||||||
|
|
||||||
lp.sec("creating bot test group with bot")
|
|
||||||
bot_contact = ac1.create_contact(botproc.addr)
|
|
||||||
ch = ac1.create_group_chat("bot test group")
|
|
||||||
ch.add_contact(bot_contact)
|
|
||||||
ch.send_text("hello")
|
|
||||||
|
|
||||||
botproc.fnmatch_lines(
|
|
||||||
"""
|
|
||||||
*ac_chat_modified*bot test group*
|
|
||||||
""",
|
|
||||||
)
|
|
||||||
|
|
||||||
lp.sec("adding third member {}".format(ac2.get_config("addr")))
|
|
||||||
contact3 = ac1.create_contact(ac2.get_config("addr"))
|
|
||||||
ch.add_contact(contact3)
|
|
||||||
|
|
||||||
reply = ac1._evtracker.wait_next_incoming_message()
|
|
||||||
assert "hello" in reply.text
|
|
||||||
|
|
||||||
lp.sec("now looking at what the bot received")
|
|
||||||
botproc.fnmatch_lines(
|
|
||||||
"""
|
|
||||||
*ac_member_added {}*from*{}*
|
|
||||||
""".format(
|
|
||||||
contact3.addr,
|
|
||||||
ac1.get_config("addr"),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
lp.sec("contact successfully added, now removing")
|
|
||||||
ch.remove_contact(contact3)
|
|
||||||
botproc.fnmatch_lines(
|
|
||||||
"""
|
|
||||||
*ac_member_removed {}*from*{}*
|
|
||||||
""".format(
|
|
||||||
contact3.addr,
|
|
||||||
ac1.get_config("addr"),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "deltachat"
|
name = "deltachat"
|
||||||
version = "1.157.3"
|
version = "2.25.0"
|
||||||
description = "Python bindings for the Delta Chat Core library using CFFI against the Rust-implemented libdeltachat"
|
description = "Python bindings for the Delta Chat Core library using CFFI against the Rust-implemented libdeltachat"
|
||||||
readme = "README.rst"
|
readme = "README.rst"
|
||||||
requires-python = ">=3.8"
|
requires-python = ">=3.8"
|
||||||
@@ -47,6 +47,10 @@ line-length = 120
|
|||||||
|
|
||||||
[tool.ruff]
|
[tool.ruff]
|
||||||
lint.select = ["E", "F", "W", "YTT", "C4", "ISC", "ICN", "TID", "DTZ", "PLC", "PLE", "PLW", "PIE", "COM", "UP004", "UP010", "UP031", "UP032", "ANN204"]
|
lint.select = ["E", "F", "W", "YTT", "C4", "ISC", "ICN", "TID", "DTZ", "PLC", "PLE", "PLW", "PIE", "COM", "UP004", "UP010", "UP031", "UP032", "ANN204"]
|
||||||
|
lint.ignore = [
|
||||||
|
"PLC0415", # `import` should be at the top-level of a file
|
||||||
|
"PLW1641" # Object does not implement `__hash__` method
|
||||||
|
]
|
||||||
line-length = 120
|
line-length = 120
|
||||||
|
|
||||||
[tool.isort]
|
[tool.isort]
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user