Compare commits
828 Commits
Author | SHA1 | Date | |
---|---|---|---|
858e16d34f | |||
a60e62efb1 | |||
97f9d4be67 | |||
fa4eec41f7 | |||
77516c97db | |||
cba01f0865 | |||
8b754017ca | |||
a27600046e | |||
fb2667631d | |||
b638f7037a | |||
74699a8262 | |||
eabf2a4582 | |||
325d62b41c | |||
e955a056e2 | |||
723f8c5fd5 | |||
a16137f53f | |||
d60b8b97f9 | |||
7b0bc51183 | |||
53aa076555 | |||
f57370f33a | |||
c557d51b6f | |||
df3fdc26a0 | |||
af00c34aac | |||
120bf39f55 | |||
26a7e35f27 | |||
d44d2a5f00 | |||
7f1d86b338 | |||
d8816280f0 | |||
b09a73040f | |||
740b5f2602 | |||
96841c70c7 | |||
f92735d35d | |||
516fd3c92d | |||
a775b57134 | |||
bf21604d42 | |||
1bb39eba87 | |||
3190703dc8 | |||
5095db8a43 | |||
1f1634ea59 | |||
a7dd033c3b | |||
95e78ffa05 | |||
42276ea7d0 | |||
dffd67eb31 | |||
55e79063d6 | |||
46f4bbb3b5 | |||
240559581a | |||
48ba829465 | |||
eef654de98 | |||
d76a04bd0a | |||
a8fe54a78d | |||
0bcb0b882f | |||
4525fa31aa | |||
aeaea0574f | |||
99d71c2177 | |||
3e60cfafd3 | |||
3123695869 | |||
777af73e2b | |||
716751cf76 | |||
6ebd5cbbd8 | |||
077b818d82 | |||
5af1d80055 | |||
f236d12166 | |||
127eb908f3 | |||
40d76b2296 | |||
8147815037 | |||
57f156be83 | |||
2cfd880cdb | |||
430b38e770 | |||
e7f463a082 | |||
1d39c771e4 | |||
c81c0dd22a | |||
f8a1ab4622 | |||
5d3309fdcd | |||
4ae028fe73 | |||
707db950c8 | |||
94812d8648 | |||
8548b69e6e | |||
e3cb665d92 | |||
fb713ed91b | |||
d99eacc2e1 | |||
62e55214fc | |||
464d27ad7e | |||
f88c5f6c08 | |||
da6ce791bc | |||
b33b50987b | |||
5193634a52 | |||
0d94746f4a | |||
85680935d4 | |||
46e2683995 | |||
492722af8b | |||
56749dfb20 | |||
04567c765e | |||
048158ad6d | |||
7326b9e10d | |||
8522d8f29c | |||
bab385c342 | |||
bb27ef7939 | |||
d2044c647b | |||
c585d00f16 | |||
015c076315 | |||
426aa33723 | |||
da8e415ae1 | |||
1b834c6858 | |||
d82726cd1b | |||
288f0a06bb | |||
0121d75032 | |||
53c86702a3 | |||
192fe89789 | |||
959ca3cef3 | |||
ccd55d2a28 | |||
bfa9a83d31 | |||
2f7b4d7f68 | |||
d887855e16 | |||
1a1e68ec98 | |||
a2754f15fc | |||
b6d81f34ba | |||
f9fb33e696 | |||
f72d5de2d7 | |||
0365c0786a | |||
af7a00d030 | |||
8a7efce941 | |||
ce73aa5a74 | |||
64d63a25cc | |||
2cef9c4fcf | |||
4265d43096 | |||
25857591a2 | |||
27f5a1a685 | |||
84da2d6a29 | |||
859ebad55d | |||
e538a4d304 | |||
f94c2b40a3 | |||
47d29ecf63 | |||
f2088a687e | |||
faeeee2948 | |||
7923cfe8f8 | |||
b51d0a9b05 | |||
09f22a801e | |||
3a824c5f9d | |||
df02f51c56 | |||
fc5e3a6728 | |||
57fbd3c723 | |||
25cd1e2cc1 | |||
f5659d455d | |||
5ed7abdbeb | |||
09875fe160 | |||
f716b8fc0f | |||
9f66f93641 | |||
f00d4d7d3f | |||
0929535b2e | |||
8869e253ca | |||
f3a5ea2956 | |||
f4d4dc91b1 | |||
c6fd65d1d7 | |||
0795906533 | |||
a2b45bc799 | |||
757657f29c | |||
219c7659e1 | |||
ae32bae791 | |||
57eba77561 | |||
d5bc7c4343 | |||
32f57b7c26 | |||
692bb8faa7 | |||
455a0fc239 | |||
b2cbd13251 | |||
ce21ba1545 | |||
c89085bf44 | |||
4254ed3c63 | |||
85564a35fd | |||
09713d40ba | |||
16d5aeac7c | |||
e19ba5a06a | |||
f7a5077d5d | |||
f8dc24bc09 | |||
e9419f10d3 | |||
cded603c27 | |||
d2ae3ebf9e | |||
730ccdd456 | |||
2f042ad915 | |||
ba70691877 | |||
ed11686a99 | |||
5c50d86908 | |||
fea31753b0 | |||
0d64cd8bab | |||
9be0f8f000 | |||
78401214b0 | |||
b2a07aba3a | |||
1e0bb3da95 | |||
59994da176 | |||
3d281b3316 | |||
ea86849a58 | |||
399789811e | |||
8d117cb0a4 | |||
588b8e0303 | |||
1794922263 | |||
0ededb8863 | |||
aa59bb1a41 | |||
f2703979a4 | |||
d2a1dc792f | |||
06d66a0b2b | |||
0e2522279e | |||
141a42a75b | |||
a1bf37e457 | |||
a20b7895a9 | |||
5666821e7b | |||
5132d8f097 | |||
b81ff9c008 | |||
7e62bc4819 | |||
d058be25ad | |||
1269be1d04 | |||
3b8837a16b | |||
32f478e4a0 | |||
e2b50d6194 | |||
74e33b0a51 | |||
107969c09a | |||
d379118772 | |||
291594b99c | |||
f2cdda7278 | |||
6911458d15 | |||
6238effdc2 | |||
498377a230 | |||
3dd4ec57ff | |||
e15b0e04b8 | |||
97b1fc813b | |||
917040b044 | |||
69646a160d | |||
54adb0509e | |||
bd3a3b6eaf | |||
296428d53e | |||
e0ca876de2 | |||
a431a4fa04 | |||
cc2bd03ec9 | |||
1fe81b7d1e | |||
0bd5a0d92d | |||
330ddb6a30 | |||
52dbd702ad | |||
d7c3570ba3 | |||
ab4d51b40b | |||
1665c93d3b | |||
b51fdbce9f | |||
351b423e15 | |||
7690be1647 | |||
68aeb93afa | |||
51062863a5 | |||
4fb4b7aa6c | |||
7f3cbcedc0 | |||
6ef09def81 | |||
c4c6aff9a5 | |||
d71850cef6 | |||
2597c9bfac | |||
93307b57aa | |||
618953c865 | |||
e04dd78624 | |||
fa0c4025f7 | |||
2d2d185200 | |||
cb7278eb50 | |||
89aa114192 | |||
ed062e0ce5 | |||
a69ef8402b | |||
8779f67d2d | |||
e4b72136b8 | |||
4ff5091bc2 | |||
6f131250f1 | |||
221a63d980 | |||
d02eda147c | |||
9b25716136 | |||
6628a47f23 | |||
ec0e6bc3f8 | |||
d2c02be3a0 | |||
594492fbdd | |||
bd9ea7a88d | |||
51327a4056 | |||
33bd60528b | |||
7e54474111 | |||
e307069d62 | |||
91db63294c | |||
fd04e08c9c | |||
6576409d60 | |||
045cb2058c | |||
d03afc12fd | |||
48799a3cff | |||
dba259e9f1 | |||
07885f5810 | |||
696c518550 | |||
411ef2691c | |||
fc6074ea9f | |||
ab1670e2ce | |||
9142a33bbf | |||
f6eefa4ecc | |||
f1db166ac4 | |||
887c2bc56d | |||
f0738a93c3 | |||
75381c2c6e | |||
bf0b9959d1 | |||
406a54b597 | |||
be04d1a862 | |||
85b2d5a124 | |||
521a7ed7b0 | |||
529b188164 | |||
8d307d8134 | |||
8c675b52bc | |||
aa51aa2aa0 | |||
86865c6da5 | |||
45296100df | |||
1605fbc012 | |||
c6c92e273d | |||
467b373c43 | |||
72ce7f06e9 | |||
346a7284f7 | |||
ee4ac67081 | |||
5a93d14d75 | |||
96a47a60ad | |||
b24a47ad7f | |||
cd1fd1bb7c | |||
d44df7b6e6 | |||
9d1ac0c84b | |||
76af9cba5a | |||
b69fc30902 | |||
c3174f4de9 | |||
99ce68e9ba | |||
0cf73673a9 | |||
08f442dc7b | |||
8a8b95228c | |||
31a752fa21 | |||
a83831e68d | |||
a12a8d4fe2 | |||
e57f3a7e6c | |||
68fbed9281 | |||
8bfaa007d5 | |||
76360f89c1 | |||
d525230abd | |||
b4aa637d41 | |||
7c4334d0de | |||
062be8d7c9 | |||
db25ee59c5 | |||
4b0bc6d0bf | |||
8c0b04b995 | |||
e5989adf92 | |||
9e5da2f9d7 | |||
a284a228a3 | |||
2133e0d1be | |||
a6f37f1d61 | |||
9de9151826 | |||
fdd5ada98c | |||
80fcf18e24 | |||
ab94b5ca7a | |||
8d2ce56c37 | |||
1ec324354b | |||
16be6601c8 | |||
98027446c8 | |||
f2f1d874e1 | |||
25a72113b1 | |||
79c4ad5015 | |||
e24f1c7c87 | |||
dbf8a326d5 | |||
0bc9c70c66 | |||
594d2155e3 | |||
20dbd71306 | |||
6a727b9723 | |||
02a5bc096f | |||
2110db6f0c | |||
2bac867382 | |||
5fbd8a3be0 | |||
ad6440b603 | |||
064b6a915f | |||
1578ebb0e2 | |||
73525a4bbc | |||
d62f49d1fc | |||
63b88e77f2 | |||
3d8f15c20b | |||
cac5d56d60 | |||
bd2a672c14 | |||
82396e73f5 | |||
ba928b169d | |||
4fed720f97 | |||
78238c85d4 | |||
4f2ae7b73f | |||
f82a9cc7ac | |||
cce7624ab8 | |||
c5ecd09172 | |||
7b21c1c2f4 | |||
f8714d81f5 | |||
8622656005 | |||
52237fadb6 | |||
222cccf388 | |||
bab308508e | |||
dedb83c867 | |||
723a90cdd6 | |||
67d2398fa8 | |||
5f3b6ec007 | |||
55ab0c12f1 | |||
d1227b5fc9 | |||
6ea368c383 | |||
e92b6de09f | |||
e622587db4 | |||
f2efc06d1f | |||
a2b94452db | |||
4c506f7cc3 | |||
7886f05e88 | |||
f58be0d1c1 | |||
1152394bc1 | |||
a082b5a590 | |||
bae9484df2 | |||
6f78485878 | |||
fd0fe3390b | |||
2522158127 | |||
8be107cecc | |||
5aab158c0b | |||
1d33e60e36 | |||
83c28cb857 | |||
df5bce27b0 | |||
2b15739b48 | |||
3480c88e90 | |||
432cd0f99d | |||
e8b3e9b22d | |||
d4a47671ea | |||
0bcd1e62f3 | |||
80822b7fff | |||
78f1011f52 | |||
67f6257617 | |||
169c614489 | |||
da908c438a | |||
9c9c4bf1f9 | |||
7764493298 | |||
64a20ee61b | |||
62d1af8c37 | |||
0f5274fdf6 | |||
2e2187ebf4 | |||
762c3350f4 | |||
e1a4d7f77e | |||
a7a4554a85 | |||
6bd808ce91 | |||
a5c143bc46 | |||
87c9cac756 | |||
6a047f8722 | |||
6523494e83 | |||
7c6ce8bb90 | |||
dafbfe4021 | |||
a4d5c94d9b | |||
7119e378a7 | |||
e1dc3032c1 | |||
5de03b8921 | |||
7631d43c48 | |||
d0b2ee5c85 | |||
8830a5a1df | |||
ee87626a93 | |||
9f15d38c1c | |||
4a96a977c0 | |||
9a95293bdf | |||
0b3a06d263 | |||
9a6249c4f5 | |||
50bd51e461 | |||
04f8013314 | |||
a0aaf0057a | |||
8e78b3e6be | |||
57a503818d | |||
25d2ff3e9b | |||
31902d3e57 | |||
16f3fa6bae | |||
1f706673cf | |||
fac5f69ad2 | |||
97c944bb63 | |||
d0c4fe78ee | |||
265457b451 | |||
4a4a29c9f6 | |||
0a91b9e1c9 | |||
f56163295c | |||
d1c87c068b | |||
fa20761110 | |||
e4a0e0a0e9 | |||
d30ae19e2a | |||
5c919e6bff | |||
434393d1c3 | |||
af9aa5d7cb | |||
05eb75442a | |||
3496ed0c7e | |||
1b89604c7a | |||
67a9d133e9 | |||
ed9118b346 | |||
59e55cfbd5 | |||
788d3b32ac | |||
1d414cf2fd | |||
cc3c168162 | |||
1ee6837f0e | |||
27dcea7c5b | |||
dcda7f7b8c | |||
e0cbb69a4f | |||
7ec95f786d | |||
1efe40add5 | |||
cbd73ee313 | |||
34227a7a39 | |||
71cb9b2d1d | |||
cd4c9b194f | |||
98762a0235 | |||
2fd1fd9573 | |||
aff3964078 | |||
2778580397 | |||
962062fe44 | |||
0578b21270 | |||
36a800c3f5 | |||
6d21f84187 | |||
f1e9833310 | |||
46f5acc4f9 | |||
95d4dcaeb3 | |||
64c542e614 | |||
13d081ea80 | |||
c0f9d86287 | |||
bcdecdaa73 | |||
daac3ebca2 | |||
639f9cf966 | |||
4fc48b5aa4 | |||
307ff77b42 | |||
9b500bc5f7 | |||
e313154134 | |||
27e94c438d | |||
58392876df | |||
115c4b1aa7 | |||
ba5649d259 | |||
1b30575510 | |||
7dbebd3ea7 | |||
6f18790352 | |||
d1e04a2ece | |||
bea0bbd0c2 | |||
0530503ef2 | |||
d1e8ff814b | |||
4f8ae761a2 | |||
b530e92834 | |||
b2a6777995 | |||
b461fc5e40 | |||
b7a8c6b60f | |||
41aa8ad799 | |||
7973baedd0 | |||
299b71d982 | |||
76aafe1fa8 | |||
95a0229aaf | |||
915a8fbad7 | |||
d4d7fef313 | |||
4e1dc9f885 | |||
155ae80d22 | |||
c7e336efd9 | |||
ac3c65a0cc | |||
df74df475b | |||
a61e2db7cb | |||
7aabe12acf | |||
c4b75e5754 | |||
6a7adb20a8 | |||
b49fb2b69c | |||
4bda29cb38 | |||
5f14141ec9 | |||
c088e45d85 | |||
d59c51a94b | |||
47b7fae61b | |||
1a40b0c1e9 | |||
27d886826c | |||
18981cb636 | |||
ffa8f65aa8 | |||
82588b00c5 | |||
603449e850 | |||
248d88c849 | |||
d19535fa21 | |||
49204cafcc | |||
812db2d267 | |||
14490bea9f | |||
0352970051 | |||
ed01820722 | |||
90a61f15cc | |||
86cd7f1ba6 | |||
d6ee55e35f | |||
aef64eec32 | |||
c4193d5ccd | |||
0c94186818 | |||
9039720013 | |||
a3470f8aec | |||
01badde21d | |||
a37b232dd9 | |||
579ee48385 | |||
dd985d1dad | |||
d2caea70a2 | |||
21143cf5ee | |||
dc2aed698d | |||
37c350f19f | |||
9e03fcf162 | |||
8d4521c1df | |||
9226252336 | |||
f4fb83e787 | |||
e7fcb25107 | |||
5a85258f74 | |||
2f7df2df43 | |||
ad3a753718 | |||
e45c551880 | |||
e59d338d4e | |||
7a86044f7a | |||
8b98f605bc | |||
7c773ebae0 | |||
e84417430d | |||
5a8d7b5f6d | |||
cfb8107138 | |||
43bd779fb7 | |||
7f9a400776 | |||
ce1c5873ac | |||
85ff1995fd | |||
b963f83c6a | |||
f6297ebbb0 | |||
a5259f56c5 | |||
3f75ed9c18 | |||
49ece51167 | |||
e77c3eb20a | |||
59b2a5f8d0 | |||
28710d0bc7 | |||
ad4d461606 | |||
67905089ba | |||
f2483af561 | |||
c28b87641e | |||
f8e6a69d6e | |||
54216cec4b | |||
12989bbd99 | |||
38d09dba2e | |||
fafd0c68e9 | |||
41195c8e48 | |||
a97804548e | |||
ba653c0841 | |||
5b191f78a0 | |||
83ef61287e | |||
3527e09bc5 | |||
ddc5b3268f | |||
22307b1934 | |||
bd97357f8d | |||
10dab1366e | |||
52fc94c1fe | |||
c1c7961dd6 | |||
d3eef051b1 | |||
57654df81e | |||
0f791d7a9a | |||
58779e0d65 | |||
4ac361b5fd | |||
1e2f27c061 | |||
0302e4da82 | |||
dc8743e0c0 | |||
cc5ce3d5ae | |||
caaf6f3012 | |||
c5de8fd1cc | |||
c9f23869e3 | |||
61208c0e35 | |||
dcffc74255 | |||
23e23be1a6 | |||
710427248a | |||
a868042de2 | |||
15296cd8b4 | |||
717023245f | |||
320be5bffa | |||
778abea2d9 | |||
835a1ac3a6 | |||
20a7ef33f1 | |||
e72612c7ff | |||
04e0f001b0 | |||
5db24aa901 | |||
aec5e3d77b | |||
335ddf8db5 | |||
4abaf2b236 | |||
183d212431 | |||
e99532fb89 | |||
4aa646f6b0 | |||
9dcd51fb80 | |||
6dee988b76 | |||
5af40db396 | |||
b3553bee7a | |||
ac19c94b9f | |||
845f7dc331 | |||
2adeae37e1 | |||
16eb12b2a0 | |||
8411f2aa32 | |||
e8acc49cbd | |||
4bed073c65 | |||
272735fb26 | |||
b75cf2c189 | |||
1aaa992250 | |||
6256c066f1 | |||
870b89a8f0 | |||
65ac96913c | |||
480945cb09 | |||
bfc7130ed8 | |||
a0938d9386 | |||
2338c69d40 | |||
c714501a0e | |||
a58a3e5000 | |||
ba35212b67 | |||
f3e0358de7 | |||
8064744d3a | |||
d261949db2 | |||
877f0fe2e8 | |||
003d85772c | |||
e7e10131de | |||
830361e48b | |||
1b1a9ce250 | |||
25ac4c708f | |||
c268e90f49 | |||
c17512b7ab | |||
1b837b3dc7 | |||
2ece724f75 | |||
276ac840aa | |||
1f91461853 | |||
1f9874102a | |||
822605c157 | |||
e49266ae43 | |||
62e9de1a3b | |||
2ddc4f7ae9 | |||
2dd402675d | |||
25b1af1e11 | |||
75fb2b8156 | |||
2a403f8b85 | |||
c3d45a9f06 | |||
c07b85b625 | |||
511f653e6e | |||
5636eaca6d | |||
4b839b9958 | |||
3f79da84d5 | |||
d540638223 | |||
4ec9b6dd4e | |||
3bc219167a | |||
8a55c97b4e | |||
9e34162a09 | |||
860a371eeb | |||
41a46526a1 | |||
46b798ac1b | |||
359d0f2910 | |||
ad3cb0386b | |||
3a183cb218 | |||
2eecaccd1c | |||
5f30a98bc1 | |||
b8a2fcbaff | |||
01496cd080 | |||
6a968ab82a | |||
c0c4890887 | |||
171a53592d | |||
7811c330db | |||
9bcd131e66 | |||
c791423dd5 | |||
80bdf38388 | |||
9d9cb32f4e | |||
87229bab13 | |||
f065e9e4d5 | |||
3812693111 | |||
dd3c572256 | |||
c5dfe40326 | |||
ef278301e3 | |||
2888fd64b0 | |||
27c0f37e49 | |||
0774f6a5e7 | |||
4036d4459b | |||
ee643de5b6 | |||
8c7549a09e | |||
7a16146304 | |||
3d3809a21b | |||
29465397dd | |||
d300bb1735 | |||
2e703472f1 | |||
8fede90b9e | |||
d128f157c4 | |||
4fcedabfd0 | |||
246c8e4f74 | |||
4d2207aba7 | |||
17b8b86d68 | |||
fdb57230a3 | |||
7469732bbc | |||
d1dd6c3440 | |||
02612c0061 | |||
a4db63a773 | |||
035c2b906a | |||
6ea8be5749 | |||
36024d5439 | |||
8d52c98373 | |||
b4a4eb0057 | |||
b469c8ddbd | |||
eee0036c7f | |||
89c66b9430 | |||
bd38319d83 | |||
33dffd5ea8 | |||
57176dadd4 | |||
dd449a8705 | |||
587ad9f41d | |||
a16ad8bf3b | |||
1e0490bd36 | |||
8afc641f0c | |||
2e4d58cb92 | |||
02d7e2db65 | |||
f935c573e9 | |||
4a25e66c00 | |||
95f4e3448e | |||
eacb1c1771 | |||
07fd825349 | |||
be15cc8a36 | |||
2f68519b3c | |||
efe641f202 | |||
9bd663046a | |||
11b07f01ba | |||
6c2f370e6b | |||
936bccccd2 | |||
c30ffeb81e | |||
e05a323afd | |||
80895deae2 | |||
eddc691fc9 | |||
deb2d7194d | |||
fd8cfb11fb | |||
9407aa4600 | |||
263b8da37d | |||
b95988b4e2 | |||
35025e164a | |||
32bbab8518 | |||
84c0b745af | |||
8b286fb009 | |||
386fa58b67 | |||
c5cfbc2297 | |||
cd0a2beb11 | |||
73f01ad8d8 | |||
930b639cc9 | |||
58483ea70c | |||
072cac0347 | |||
956d7cf3f3 | |||
7558a2162e | |||
62b165c0b4 | |||
fe258e1b67 | |||
dc37232100 | |||
163f55f9c2 | |||
2d16fd085e | |||
e1a5f5bca5 | |||
6e772ee189 | |||
2b0f178ba3 | |||
79e6c9fa6c | |||
1426ddec5f |
28
.github/ISSUE_TEMPLATE.md
vendored
@ -1,20 +1,36 @@
|
|||||||
Please answer the following questions.
|
<!-- This is a bug report template. By following the instructions below and
|
||||||
|
filling out the sections with your information, you will help the us to get all
|
||||||
|
the necessary data to fix your issue.
|
||||||
|
|
||||||
### Which version of matterbridge are you using?
|
You can also preview your report before submitting it.
|
||||||
run ```matterbridge -version```
|
|
||||||
|
|
||||||
### If you're having problems with mattermost please specify mattermost version.
|
Text between <!-- and --> marks will be invisible in the report.
|
||||||
|
-->
|
||||||
|
|
||||||
|
<!-- If you have a configuration problem, please first try to create a basic configuration following the instructions on [the wiki](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) before filing an issue. -->
|
||||||
|
|
||||||
|
|
||||||
|
### Environment
|
||||||
|
<!-- run `matterbridge -version` -->
|
||||||
|
<!-- If you're having problems with mattermost also specify the mattermost version. -->
|
||||||
|
Version:
|
||||||
|
|
||||||
|
<!-- What operating system are you using ? (be as specific as possible) -->
|
||||||
|
Operating system:
|
||||||
|
|
||||||
|
<!-- If you compiled matterbridge yourself:
|
||||||
|
* Specify the output of `go version`
|
||||||
|
* Specify the output of `git rev-parse HEAD` -->
|
||||||
|
|
||||||
### Please describe the expected behavior.
|
### Please describe the expected behavior.
|
||||||
|
|
||||||
|
|
||||||
### Please describe the actual behavior.
|
### Please describe the actual behavior.
|
||||||
#### Use logs from running ```matterbridge -debug``` if possible.
|
<!-- Use logs from running `matterbridge -debug` if possible. -->
|
||||||
|
|
||||||
|
|
||||||
### Any steps to reproduce the behavior?
|
### Any steps to reproduce the behavior?
|
||||||
|
|
||||||
|
|
||||||
### Please add your configuration file
|
### Please add your configuration file
|
||||||
#### (be sure to exclude or anonymize private data (tokens/passwords))
|
<!-- (be sure to exclude or anonymize private data (tokens/passwords)) -->
|
||||||
|
26
.github/ISSUE_TEMPLATE/Bug_report.md
vendored
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: Create a report to help us improve. (Check the FAQ on the wiki first)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Describe the bug**
|
||||||
|
A clear and concise description of what the bug is.
|
||||||
|
|
||||||
|
**To Reproduce**
|
||||||
|
Steps to reproduce the behavior:
|
||||||
|
|
||||||
|
**Expected behavior**
|
||||||
|
A clear and concise description of what you expected to happen.
|
||||||
|
|
||||||
|
**Screenshots/debug logs**
|
||||||
|
If applicable, add screenshots to help explain your problem.
|
||||||
|
Use logs from running `matterbridge -debug` if possible.
|
||||||
|
|
||||||
|
**Environment (please complete the following information):**
|
||||||
|
- OS: [e.g. linux]
|
||||||
|
- Matterbridge version: output of `matterbridge -version`
|
||||||
|
- If self compiled: output of `git rev-parse HEAD`
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Please add your configuration file (be sure to exclude or anonymize private data (tokens/passwords))
|
17
.github/ISSUE_TEMPLATE/Feature_request.md
vendored
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
---
|
||||||
|
name: Feature request
|
||||||
|
about: Suggest an idea for this project
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Is your feature request related to a problem? Please describe.**
|
||||||
|
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
||||||
|
|
||||||
|
**Describe the solution you'd like**
|
||||||
|
A clear and concise description of what you want to happen.
|
||||||
|
|
||||||
|
**Describe alternatives you've considered**
|
||||||
|
A clear and concise description of any alternative solutions or features you've considered.
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context or screenshots about the feature request here.
|
212
.golangci.yaml
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
# For full documentation of the configuration options please
|
||||||
|
# see: https://github.com/golangci/golangci-lint#config-file.
|
||||||
|
|
||||||
|
# options for analysis running
|
||||||
|
run:
|
||||||
|
# default concurrency is the available CPU number
|
||||||
|
# concurrency: 4
|
||||||
|
|
||||||
|
# timeout for analysis, e.g. 30s, 5m, default is 1m
|
||||||
|
deadline: 1m
|
||||||
|
|
||||||
|
# exit code when at least one issue was found, default is 1
|
||||||
|
issues-exit-code: 1
|
||||||
|
|
||||||
|
# include test files or not, default is true
|
||||||
|
tests: true
|
||||||
|
|
||||||
|
# list of build tags, all linters use it. Default is empty list.
|
||||||
|
build-tags:
|
||||||
|
|
||||||
|
# which dirs to skip: they won't be analyzed;
|
||||||
|
# can use regexp here: generated.*, regexp is applied on full path;
|
||||||
|
# default value is empty list, but next dirs are always skipped independently
|
||||||
|
# from this option's value:
|
||||||
|
# vendor$, third_party$, testdata$, examples$, Godeps$, builtin$
|
||||||
|
skip-dirs:
|
||||||
|
|
||||||
|
# which files to skip: they will be analyzed, but issues from them
|
||||||
|
# won't be reported. Default value is empty list, but there is
|
||||||
|
# no need to include all autogenerated files, we confidently recognize
|
||||||
|
# autogenerated files. If it's not please let us know.
|
||||||
|
skip-files:
|
||||||
|
|
||||||
|
|
||||||
|
# output configuration options
|
||||||
|
output:
|
||||||
|
# colored-line-number|line-number|json|tab|checkstyle, default is "colored-line-number"
|
||||||
|
format: colored-line-number
|
||||||
|
|
||||||
|
# print lines of code with issue, default is true
|
||||||
|
print-issued-lines: true
|
||||||
|
|
||||||
|
# print linter name in the end of issue text, default is true
|
||||||
|
print-linter-name: true
|
||||||
|
|
||||||
|
|
||||||
|
# all available settings of specific linters, we can set an option for
|
||||||
|
# a given linter even if we deactivate that same linter at runtime
|
||||||
|
linters-settings:
|
||||||
|
errcheck:
|
||||||
|
# report about not checking of errors in type assertions: `a := b.(MyStruct)`;
|
||||||
|
# default is false: such cases aren't reported by default.
|
||||||
|
check-type-assertions: false
|
||||||
|
|
||||||
|
# report about assignment of errors to blank identifier: `num, _ := strconv.Atoi(numStr)`;
|
||||||
|
# default is false: such cases aren't reported by default.
|
||||||
|
check-blank: false
|
||||||
|
govet:
|
||||||
|
# report about shadowed variables
|
||||||
|
check-shadowing: true
|
||||||
|
golint:
|
||||||
|
# minimal confidence for issues, default is 0.8
|
||||||
|
min-confidence: 0.8
|
||||||
|
gofmt:
|
||||||
|
# simplify code: gofmt with `-s` option, true by default
|
||||||
|
simplify: true
|
||||||
|
goimports:
|
||||||
|
# put imports beginning with prefix after 3rd-party packages;
|
||||||
|
# it's a comma-separated list of prefixes
|
||||||
|
local-prefixes: github.com
|
||||||
|
gocyclo:
|
||||||
|
# minimal code complexity to report, 30 by default (but we recommend 10-20)
|
||||||
|
min-complexity: 15
|
||||||
|
maligned:
|
||||||
|
# print struct with more effective memory layout or not, false by default
|
||||||
|
suggest-new: true
|
||||||
|
dupl:
|
||||||
|
# tokens count to trigger issue, 150 by default
|
||||||
|
threshold: 150
|
||||||
|
goconst:
|
||||||
|
# minimal length of string constant, 3 by default
|
||||||
|
min-len: 3
|
||||||
|
# minimal occurrences count to trigger, 3 by default
|
||||||
|
min-occurrences: 3
|
||||||
|
depguard:
|
||||||
|
list-type: blacklist
|
||||||
|
include-go-root: false
|
||||||
|
packages:
|
||||||
|
# List of packages that we would want to blacklist for... reasons.
|
||||||
|
misspell:
|
||||||
|
# Correct spellings using locale preferences for US or UK.
|
||||||
|
# Default is to use a neutral variety of English.
|
||||||
|
# Setting locale to US will correct the British spelling of 'colour' to 'color'.
|
||||||
|
locale: US
|
||||||
|
lll:
|
||||||
|
# max line length, lines longer will be reported. Default is 120.
|
||||||
|
# '\t' is counted as 1 character by default, and can be changed with the tab-width option
|
||||||
|
line-length: 150
|
||||||
|
# tab width in spaces. Default to 1.
|
||||||
|
tab-width: 1
|
||||||
|
unused:
|
||||||
|
# treat code as a program (not a library) and report unused exported identifiers; default is false.
|
||||||
|
# XXX: if you enable this setting, unused will report a lot of false-positives in text editors:
|
||||||
|
# if it's called for subdir of a project it can't find funcs usages. All text editor integrations
|
||||||
|
# with golangci-lint call it on a directory with the changed file.
|
||||||
|
check-exported: false
|
||||||
|
unparam:
|
||||||
|
# call graph construction algorithm (cha, rta). In general, use cha for libraries,
|
||||||
|
# and rta for programs with main packages. Default is cha.
|
||||||
|
algo: rta
|
||||||
|
|
||||||
|
# Inspect exported functions, default is false. Set to true if no external program/library imports your code.
|
||||||
|
# XXX: if you enable this setting, unparam will report a lot of false-positives in text editors:
|
||||||
|
# if it's called for subdir of a project it can't find external interfaces. All text editor integrations
|
||||||
|
# with golangci-lint call it on a directory with the changed file.
|
||||||
|
check-exported: false
|
||||||
|
nakedret:
|
||||||
|
# make an issue if func has more lines of code than this setting and it has naked returns; default is 30
|
||||||
|
max-func-lines: 0 # Warn on all naked returns.
|
||||||
|
prealloc:
|
||||||
|
# XXX: we don't recommend using this linter before doing performance profiling.
|
||||||
|
# For most programs usage of prealloc will be a premature optimization.
|
||||||
|
|
||||||
|
# Report preallocation suggestions only on simple loops that have no returns/breaks/continues/gotos in them.
|
||||||
|
# True by default.
|
||||||
|
simple: true
|
||||||
|
range-loops: true # Report preallocation suggestions on range loops, true by default
|
||||||
|
for-loops: false # Report preallocation suggestions on for loops, false by default
|
||||||
|
gocritic:
|
||||||
|
# which checks should be enabled; can't be combined with 'disabled-checks';
|
||||||
|
# default are: [appendAssign assignOp caseOrder dupArg dupBranchBody dupCase flagDeref
|
||||||
|
# ifElseChain regexpMust singleCaseSwitch sloppyLen switchTrue typeSwitchVar underef
|
||||||
|
# unlambda unslice rangeValCopy defaultCaseOrder];
|
||||||
|
# all checks list: https://github.com/go-critic/checkers
|
||||||
|
# disabled for now - hugeParam
|
||||||
|
enabled-checks:
|
||||||
|
- appendAssign
|
||||||
|
- assignOp
|
||||||
|
- boolExprSimplify
|
||||||
|
- builtinShadow
|
||||||
|
- captLocal
|
||||||
|
- caseOrder
|
||||||
|
- commentedOutImport
|
||||||
|
- defaultCaseOrder
|
||||||
|
- dupArg
|
||||||
|
- dupBranchBody
|
||||||
|
- dupCase
|
||||||
|
- dupSubExpr
|
||||||
|
- elseif
|
||||||
|
- emptyFallthrough
|
||||||
|
- ifElseChain
|
||||||
|
- importShadow
|
||||||
|
- indexAlloc
|
||||||
|
- methodExprCall
|
||||||
|
- nestingReduce
|
||||||
|
- offBy1
|
||||||
|
- ptrToRefParam
|
||||||
|
- regexpMust
|
||||||
|
- singleCaseSwitch
|
||||||
|
- sloppyLen
|
||||||
|
- switchTrue
|
||||||
|
- typeSwitchVar
|
||||||
|
- typeUnparen
|
||||||
|
- underef
|
||||||
|
- unlambda
|
||||||
|
- unnecessaryBlock
|
||||||
|
- unslice
|
||||||
|
- valSwap
|
||||||
|
- wrapperFunc
|
||||||
|
- yodaStyleExpr
|
||||||
|
|
||||||
|
|
||||||
|
# linters that we should / shouldn't run
|
||||||
|
linters:
|
||||||
|
enable-all: true
|
||||||
|
disable:
|
||||||
|
- gochecknoglobals
|
||||||
|
- lll
|
||||||
|
- maligned
|
||||||
|
- prealloc
|
||||||
|
|
||||||
|
|
||||||
|
# rules to deal with reported isues
|
||||||
|
issues:
|
||||||
|
# List of regexps of issue texts to exclude, empty list by default.
|
||||||
|
# But independently from this option we use default exclude patterns,
|
||||||
|
# it can be disabled by `exclude-use-default: false`. To list all
|
||||||
|
# excluded by default patterns execute `golangci-lint run --help`
|
||||||
|
exclude:
|
||||||
|
|
||||||
|
# Independently from option `exclude` we use default exclude patterns,
|
||||||
|
# it can be disabled by this option. To list all
|
||||||
|
# excluded by default patterns execute `golangci-lint run --help`.
|
||||||
|
# Default value for this option is true.
|
||||||
|
exclude-use-default: true
|
||||||
|
|
||||||
|
# Maximum issues count per one linter. Set to 0 to disable. Default is 50.
|
||||||
|
max-per-linter: 0
|
||||||
|
|
||||||
|
# Maximum count of issues with the same text. Set to 0 to disable. Default is 3.
|
||||||
|
max-same-issues: 0
|
||||||
|
|
||||||
|
# Show only new issues: if there are unstaged changes or untracked files,
|
||||||
|
# only those changes are analyzed, else only changes in HEAD~ are analyzed.
|
||||||
|
# It's a super-useful option for integration of golangci-lint into existing
|
||||||
|
# large codebase. It's not practical to fix all existing issues at the moment
|
||||||
|
# of integration: much better don't allow issues in new code.
|
||||||
|
# Default is false.
|
||||||
|
new: false
|
||||||
|
|
||||||
|
# Show only new issues created after git revision `REV`
|
||||||
|
new-from-rev: "HEAD~1"
|
34
.goreleaser.yml
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
release:
|
||||||
|
prerelease: auto
|
||||||
|
name_template: "{{.ProjectName}} v{{.Version}}"
|
||||||
|
|
||||||
|
builds:
|
||||||
|
- env:
|
||||||
|
- CGO_ENABLED=0
|
||||||
|
goos:
|
||||||
|
- freebsd
|
||||||
|
- windows
|
||||||
|
- darwin
|
||||||
|
- linux
|
||||||
|
- dragonfly
|
||||||
|
- netbsd
|
||||||
|
- openbsd
|
||||||
|
goarch:
|
||||||
|
- amd64
|
||||||
|
- arm
|
||||||
|
- arm64
|
||||||
|
- 386
|
||||||
|
ldflags:
|
||||||
|
- -s -w -X main.githash={{.ShortCommit}}
|
||||||
|
|
||||||
|
archive:
|
||||||
|
name_template: "{{ .Binary }}-{{ .Version }}-{{ .Os }}-{{ .Arch }}{{ if .Arm }}v{{ .Arm }}{{ end }}"
|
||||||
|
format: binary
|
||||||
|
files:
|
||||||
|
- none*
|
||||||
|
replacements:
|
||||||
|
386: 32bit
|
||||||
|
amd64: 64bit
|
||||||
|
|
||||||
|
checksum:
|
||||||
|
name_template: 'checksums.txt'
|
75
.travis.yml
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
language: go
|
||||||
|
go:
|
||||||
|
- 1.11.x
|
||||||
|
go_import_path: github.com/42wim/matterbridge
|
||||||
|
|
||||||
|
# we have everything vendored
|
||||||
|
install: true
|
||||||
|
|
||||||
|
git:
|
||||||
|
depth: 200
|
||||||
|
|
||||||
|
env:
|
||||||
|
global:
|
||||||
|
- GOOS=linux GOARCH=amd64
|
||||||
|
- GO111MODULE=on
|
||||||
|
- GOLANGCI_VERSION="v1.14.0"
|
||||||
|
|
||||||
|
matrix:
|
||||||
|
# It's ok if our code fails on unstable development versions of Go.
|
||||||
|
allow_failures:
|
||||||
|
- go: tip
|
||||||
|
# Don't wait for tip tests to finish. Mark the test run green if the
|
||||||
|
# tests pass on the stable versions of Go.
|
||||||
|
fast_finish: true
|
||||||
|
|
||||||
|
notifications:
|
||||||
|
email: false
|
||||||
|
|
||||||
|
before_script:
|
||||||
|
# Get version info from tags.
|
||||||
|
- MY_VERSION="$(git describe --tags)"
|
||||||
|
# Retrieve the golangci-lint linter binary.
|
||||||
|
- curl -sfL https://install.goreleaser.com/github.com/golangci/golangci-lint.sh | bash -s -- -b ${GOPATH}/bin ${GOLANGCI_VERSION}
|
||||||
|
# Retrieve and prepare CodeClimate's test coverage reporter.
|
||||||
|
- curl -L https://codeclimate.com/downloads/test-reporter/test-reporter-latest-linux-amd64 > ./cc-test-reporter
|
||||||
|
- chmod +x ./cc-test-reporter
|
||||||
|
- ./cc-test-reporter before-build
|
||||||
|
|
||||||
|
script:
|
||||||
|
# Ensure that the module files are being kept correctly and that vendored dependencies are up-to-date.
|
||||||
|
- go mod tidy
|
||||||
|
- go mod vendor
|
||||||
|
- git diff --exit-code --quiet || (echo "Please run 'go mod tidy' to clean up the 'go.mod' and 'go.sum' files."; false)
|
||||||
|
|
||||||
|
# Run the linter.
|
||||||
|
- golangci-lint run
|
||||||
|
|
||||||
|
# Run all the tests with the race detector and generate coverage.
|
||||||
|
- go test -v -race -coverprofile c.out ./...
|
||||||
|
|
||||||
|
# Run the build script to generate the necessary binaries and images.
|
||||||
|
- /bin/bash ci/bintray.sh
|
||||||
|
|
||||||
|
after_script:
|
||||||
|
# Upload test coverage to CodeClimate.
|
||||||
|
- ./cc-test-reporter after-build --exit-code ${TRAVIS_TEST_RESULT}
|
||||||
|
|
||||||
|
branches:
|
||||||
|
only:
|
||||||
|
- master
|
||||||
|
|
||||||
|
deploy:
|
||||||
|
on:
|
||||||
|
all_branches: true
|
||||||
|
provider: bintray
|
||||||
|
on:
|
||||||
|
all_branches: true
|
||||||
|
edge:
|
||||||
|
branch: v1.8.47
|
||||||
|
file: ci/deploy.json
|
||||||
|
user: 42wim
|
||||||
|
on:
|
||||||
|
all_branches: true
|
||||||
|
key:
|
||||||
|
secure: "CeXXe6JOmt7HYR81MdWLua0ltQHhDdkIeRGBFbgd7hkb1wi8eF9DgpAcQrTso8NIlHNZmSAP46uhFgsRvkuezzX0ygalZ7DCJyAyn3sAMEh+UQSHV1WGThRehTtidqRGjetzsIGSwdrJOWil+XTfbO1Z8DGzfakhSuAZka8CM4BAoe3YeP9rYK8h+84x0GHfczvsLtXZ3mWLvQuwe4pK6+ItBCUg0ae7O7ZUpWHy0xQQkkWztY/6RAzXfaG7DuGjIw+20fhx3WOXRNpHCtZ6Bc3qERCpk0s1HhlQWlrN9wDaFTBWYwlvSnNgvxxMbNXJ6RrRJ0l0bA7FUswYwyroxhzrGLdzWDg8dHaQkypocngdalfhpsnoO9j3ApJhomUFJ3UoEq5nOGRUrKn8MPi+dP0zE4kNQ3e4VNa1ufNrvfpWolMg3xh8OXuhQdD5wIM5zFAbRJLqWSCVAjPq4DDPecmvXBOlIial7oa312lN5qnBnUjvAcxszZ+FUyDHT1Grxzna4tMwxY9obPzZUzm7359AOCCwIQFVB8GLqD2nwIstcXS0zGRz+fhviPipHuBa02q5bGUZwmkvrSNab0s8Jo7pCrel2Rz3nWPKaiCfq2WjbW1CLheSMkOQrjsdUd1hhbqNWFPUjJPInTc77NAKCfm5runv5uyowRLh4NNd0sI="
|
@ -6,6 +6,6 @@ RUN apk update && apk add go git gcc musl-dev ca-certificates \
|
|||||||
&& cd /go/src/github.com/42wim/matterbridge \
|
&& cd /go/src/github.com/42wim/matterbridge \
|
||||||
&& export GOPATH=/go \
|
&& export GOPATH=/go \
|
||||||
&& go get \
|
&& go get \
|
||||||
&& go build -o /bin/matterbridge \
|
&& go build -x -ldflags "-X main.githash=$(git log --pretty=format:'%h' -n 1)" -o /bin/matterbridge \
|
||||||
&& rm -rf /go \
|
&& rm -rf /go \
|
||||||
&& apk del --purge git go gcc musl-dev
|
&& apk del --purge git go gcc musl-dev
|
||||||
|
115
README-0.6.md
@ -1,115 +0,0 @@
|
|||||||
# matterbridge
|
|
||||||
|
|
||||||
Simple bridge between mattermost, IRC, XMPP, Gitter and Slack
|
|
||||||
|
|
||||||
* Relays public channel messages between mattermost, IRC, XMPP, Gitter and Slack. Pick and mix.
|
|
||||||
* Supports multiple channels.
|
|
||||||
* Matterbridge can also work with private groups on your mattermost.
|
|
||||||
|
|
||||||
Look at [matterbridge.conf.sample] (https://github.com/42wim/matterbridge/blob/master/matterbridge.conf.sample) for documentation and an example.
|
|
||||||
|
|
||||||
## Changelog
|
|
||||||
Since v0.6.1 support for XMPP, Gitter and Slack is added. More details in [changelog.md] (https://github.com/42wim/matterbridge/blob/master/changelog.md)
|
|
||||||
|
|
||||||
## Requirements:
|
|
||||||
Accounts to one of the supported bridges
|
|
||||||
* [Mattermost] (https://github.com/mattermost/platform/)
|
|
||||||
* [IRC] (http://www.mirc.com/servers.html)
|
|
||||||
* [XMPP] (https://jabber.org)
|
|
||||||
* [Gitter] (https://gitter.im)
|
|
||||||
* [Slack] (https://www.slack.com)
|
|
||||||
|
|
||||||
## binaries
|
|
||||||
Binaries can be found [here] (https://github.com/42wim/matterbridge/releases/)
|
|
||||||
* For use with mattermost 3.3.0+ [v0.6.1](https://github.com/42wim/matterircd/releases/tag/v0.6.1)
|
|
||||||
* For use with mattermost 3.0.0-3.2.0 [v0.5.0](https://github.com/42wim/matterircd/releases/tag/v0.5.0)
|
|
||||||
|
|
||||||
|
|
||||||
## Docker
|
|
||||||
Create your matterbridge.conf file locally eg in ```/tmp/matterbridge.conf```
|
|
||||||
|
|
||||||
```
|
|
||||||
docker run -ti -v /tmp/matterbridge.conf:/matterbridge.conf 42wim/matterbridge:0.6.1
|
|
||||||
```
|
|
||||||
|
|
||||||
## Compatibility
|
|
||||||
### Mattermost
|
|
||||||
* Matterbridge v0.6.1 works with mattermost 3.3.0 and higher [3.3.0 release](https://github.com/mattermost/platform/releases/tag/v3.3.0)
|
|
||||||
* Matterbridge v0.5.0 works with mattermost 3.0.0 - 3.2.0 [3.2.0 release](https://github.com/mattermost/platform/releases/tag/v3.2.0)
|
|
||||||
|
|
||||||
|
|
||||||
#### Webhooks version
|
|
||||||
* Configured incoming/outgoing [webhooks](https://www.mattermost.org/webhooks/) on your mattermost instance.
|
|
||||||
|
|
||||||
#### Plus (API) version
|
|
||||||
* A dedicated user(bot) on your mattermost instance.
|
|
||||||
|
|
||||||
|
|
||||||
## building
|
|
||||||
Go 1.6+ is required. Make sure you have [Go](https://golang.org/doc/install) properly installed, including setting up your [GOPATH] (https://golang.org/doc/code.html#GOPATH)
|
|
||||||
|
|
||||||
```
|
|
||||||
cd $GOPATH
|
|
||||||
go get github.com/42wim/matterbridge
|
|
||||||
```
|
|
||||||
|
|
||||||
You should now have matterbridge binary in the bin directory:
|
|
||||||
|
|
||||||
```
|
|
||||||
$ ls bin/
|
|
||||||
matterbridge
|
|
||||||
```
|
|
||||||
|
|
||||||
## running
|
|
||||||
1) Copy the matterbridge.conf.sample to matterbridge.conf in the same directory as the matterbridge binary.
|
|
||||||
2) Edit matterbridge.conf with the settings for your environment. See below for more config information.
|
|
||||||
3) Now you can run matterbridge.
|
|
||||||
|
|
||||||
```
|
|
||||||
Usage of ./matterbridge:
|
|
||||||
-conf string
|
|
||||||
config file (default "matterbridge.conf")
|
|
||||||
-debug
|
|
||||||
enable debug
|
|
||||||
-plus
|
|
||||||
running using API instead of webhooks (deprecated, set Plus flag in [general] config)
|
|
||||||
-version
|
|
||||||
show version
|
|
||||||
```
|
|
||||||
|
|
||||||
## config
|
|
||||||
### matterbridge
|
|
||||||
matterbridge looks for matterbridge.conf in current directory. (use -conf to specify another file)
|
|
||||||
|
|
||||||
Look at [matterbridge.conf.sample] (https://github.com/42wim/matterbridge/blob/master/matterbridge.conf.sample) for an example.
|
|
||||||
|
|
||||||
### mattermost
|
|
||||||
#### webhooks version
|
|
||||||
You'll have to configure the incoming and outgoing webhooks.
|
|
||||||
|
|
||||||
* incoming webhooks
|
|
||||||
Go to "account settings" - integrations - "incoming webhooks".
|
|
||||||
Choose a channel at "Add a new incoming webhook", this will create a webhook URL right below.
|
|
||||||
This URL should be set in the matterbridge.conf in the [mattermost] section (see above)
|
|
||||||
|
|
||||||
* outgoing webhooks
|
|
||||||
Go to "account settings" - integrations - "outgoing webhooks".
|
|
||||||
Choose a channel (the same as the one from incoming webhooks) and fill in the address and port of the server matterbridge will run on.
|
|
||||||
|
|
||||||
e.g. http://192.168.1.1:9999 (192.168.1.1:9999 is the BindAddress specified in [mattermost] section of matterbridge.conf)
|
|
||||||
|
|
||||||
#### plus version
|
|
||||||
You'll have to create a new dedicated user on your mattermost instance.
|
|
||||||
Specify the login and password in [mattermost] section of matterbridge.conf
|
|
||||||
|
|
||||||
## FAQ
|
|
||||||
Please look at [matterbridge.conf.sample] (https://github.com/42wim/matterbridge/blob/master/matterbridge.conf.sample) for more information first.
|
|
||||||
### Mattermost doesn't show the IRC nicks
|
|
||||||
If you're running the webhooks version, this can be fixed by either:
|
|
||||||
* enabling "override usernames". See [mattermost documentation](http://docs.mattermost.com/developer/webhooks-incoming.html#enabling-incoming-webhooks)
|
|
||||||
* setting ```PrefixMessagesWithNick``` to ```true``` in ```mattermost``` section of your matterbridge.conf.
|
|
||||||
|
|
||||||
If you're running the plus version you'll need to:
|
|
||||||
* setting ```PrefixMessagesWithNick``` to ```true``` in ```mattermost``` section of your matterbridge.conf.
|
|
||||||
|
|
||||||
Also look at the ```RemoteNickFormat``` setting.
|
|
329
README.md
@ -1,58 +1,138 @@
|
|||||||
|
<div align="center">
|
||||||
|
|
||||||
# matterbridge
|
# matterbridge
|
||||||

|
|
||||||
|
|
||||||
Simple bridge between mattermost, IRC, XMPP, Gitter, Slack, Discord, Telegram, Rocket.Chat and Hipchat(via xmpp).
|
<br />
|
||||||
|
**A simple chat bridge**<br />
|
||||||
|
Letting people be where they want to be.<br />
|
||||||
|
<sub>Bridges between a growing number of protocols. Click below to demo or join the development chat.</sub>
|
||||||
|
|
||||||
* Relays public channel messages between multiple mattermost, IRC, XMPP, Gitter, Slack, Discord, Telegram, Rocket.Chat and Hipchat (via xmpp). Pick and mix.
|
<sup>
|
||||||
* Supports multiple channels.
|
|
||||||
* Matterbridge can also work with private groups on your mattermost.
|
|
||||||
* Allow for bridging the same bridges, which means you can eg bridge between multiple mattermosts.
|
|
||||||
* The bridge is now a gateway which has support multiple in and out bridges. (and supports multiple gateways).
|
|
||||||
|
|
||||||
Look at [matterbridge.toml.sample] (https://github.com/42wim/matterbridge/blob/master/matterbridge.toml.sample) for documentation and an example.
|
[Gitter][mb-gitter] |
|
||||||
Look at [matterbridge.toml.simple] (https://github.com/42wim/matterbridge/blob/master/matterbridge.toml.simple) for a simple example.
|
[IRC][mb-irc] |
|
||||||
|
[Discord][mb-discord] |
|
||||||
|
[Matrix][mb-matrix] |
|
||||||
|
[Slack][mb-slack] |
|
||||||
|
[Mattermost][mb-mattermost] |
|
||||||
|
[Rocket.Chat][mb-rocketchat] |
|
||||||
|
[XMPP][mb-xmpp] |
|
||||||
|
[Twitch][mb-twitch] |
|
||||||
|
[WhatsApp][mb-whatsapp] |
|
||||||
|
[Zulip][mb-zulip] |
|
||||||
|
[Telegram][mb-telegram] |
|
||||||
|
And more...
|
||||||
|
</sup>
|
||||||
|
|
||||||
|
----
|
||||||
|
[](https://github.com/42wim/matterbridge/releases/latest)
|
||||||
|
[](https://bintray.com/42wim/nightly/Matterbridge/_latestVersion)
|
||||||
|
[](https://codeclimate.com/github/42wim/matterbridge/maintainability)
|
||||||
|
[](https://codeclimate.com/github/42wim/matterbridge/test_coverage)<br />
|
||||||
|
<hr />
|
||||||
|
</div>
|
||||||
|
<div align="right"><sup>
|
||||||
|
|
||||||
## Changelog
|
**Note:** Matter<em>most</em> isn't required to run matter<em>bridge</em>.</sup></div>
|
||||||
Since v0.7.0 the configuration has changed. More details in [changelog.md] (https://github.com/42wim/matterbridge/blob/master/changelog.md)
|
|
||||||
|
|
||||||
## Requirements
|
### Table of Contents
|
||||||
Accounts to one of the supported bridges
|
* [Features](https://github.com/42wim/matterbridge/wiki/Features)
|
||||||
* [Mattermost] (https://github.com/mattermost/platform/)
|
* [Natively supported](#natively-supported)
|
||||||
* [IRC] (http://www.mirc.com/servers.html)
|
* [3rd party via matterbridge api](#3rd-party-via-matterbridge-api)
|
||||||
* [XMPP] (https://jabber.org)
|
* [API](#API)
|
||||||
* [Gitter] (https://gitter.im)
|
* [Chat with us](#chat-with-us)
|
||||||
* [Slack] (https://slack.com)
|
* [Screenshots](https://github.com/42wim/matterbridge/wiki/)
|
||||||
* [Discord] (https://discordapp.com)
|
* [Installing](#installing)
|
||||||
* [Telegram] (https://telegram.org)
|
* [Binaries](#binaries)
|
||||||
* [Hipchat] (https://www.hipchat.com)
|
* [Building](#building)
|
||||||
* [Rocket.chat] (https://rocket.chat)
|
* [Configuration](#configuration)
|
||||||
|
* [Howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config)
|
||||||
|
* [Examples](#examples)
|
||||||
|
* [Running](#running)
|
||||||
|
* [Docker](#docker)
|
||||||
|
* [Changelog](#changelog)
|
||||||
|
* [FAQ](#faq)
|
||||||
|
* [Related projects](#related-projects)
|
||||||
|
* [Articles](#articles)
|
||||||
|
* [Thanks](#thanks)
|
||||||
|
|
||||||
## Docker
|
## Features
|
||||||
Create your matterbridge.toml file locally eg in ```/tmp/matterbridge.toml```
|
* [Support bridging between any protocols](https://github.com/42wim/matterbridge/wiki/Features#support-bridging-between-any-protocols)
|
||||||
```
|
* [Support multiple gateways(bridges) for your protocols](https://github.com/42wim/matterbridge/wiki/Features#support-multiple-gatewaysbridges-for-your-protocols)
|
||||||
docker run -ti -v /tmp/matterbridge.toml:/matterbridge.toml 42wim/matterbridge
|
* [Message edits and deletes](https://github.com/42wim/matterbridge/wiki/Features#message-edits-and-deletes)
|
||||||
```
|
* Preserves threading when possible
|
||||||
|
* [Attachment / files handling](https://github.com/42wim/matterbridge/wiki/Features#attachment--files-handling)
|
||||||
|
* [Username and avatar spoofing](https://github.com/42wim/matterbridge/wiki/Features#username-and-avatar-spoofing)
|
||||||
|
* [Private groups](https://github.com/42wim/matterbridge/wiki/Features#private-groups)
|
||||||
|
* [API](https://github.com/42wim/matterbridge/wiki/Features#api)
|
||||||
|
|
||||||
## binaries
|
### Natively supported
|
||||||
Binaries can be found [here] (https://github.com/42wim/matterbridge/releases/)
|
|
||||||
* For use with mattermost 3.5.x - 3.6.0 [v0.9.2](https://github.com/42wim/matterircd/releases/tag/v0.9.2)
|
|
||||||
* For use with mattermost 3.3.0 - 3.4.0 [v0.7.1](https://github.com/42wim/matterircd/releases/tag/v0.7.1)
|
|
||||||
|
|
||||||
## Compatibility
|
* [Mattermost](https://github.com/mattermost/mattermost-server/) 4.x, 5.x
|
||||||
### Mattermost
|
* [IRC](http://www.mirc.com/servers.html)
|
||||||
* Matterbridge v0.9.2 works with mattermost 3.5.x - 3.6.0 [3.6.0 release](https://github.com/mattermost/platform/releases/tag/v3.6.0)
|
* [XMPP](https://xmpp.org)
|
||||||
* Matterbridge v0.7.1 works with mattermost 3.3.0 - 3.4.0 [3.4.0 release](https://github.com/mattermost/platform/releases/tag/v3.4.0)
|
* [Gitter](https://gitter.im)
|
||||||
|
* [Slack](https://slack.com)
|
||||||
|
* [Discord](https://discordapp.com)
|
||||||
|
* [Telegram](https://telegram.org)
|
||||||
|
* [Hipchat](https://www.hipchat.com)
|
||||||
|
* [Rocket.chat](https://rocket.chat)
|
||||||
|
* [Matrix](https://matrix.org)
|
||||||
|
* [Steam](https://store.steampowered.com/)
|
||||||
|
* [Twitch](https://twitch.tv)
|
||||||
|
* [Ssh-chat](https://github.com/shazow/ssh-chat)
|
||||||
|
* [WhatsApp](https://www.whatsapp.com/)
|
||||||
|
* [Zulip](https://zulipchat.com)
|
||||||
|
|
||||||
#### Webhooks version
|
### 3rd party via matterbridge api
|
||||||
* Configured incoming/outgoing [webhooks](https://www.mattermost.org/webhooks/) on your mattermost instance.
|
* [Minecraft](https://github.com/elytra/MatterLink)
|
||||||
|
* [Reddit](https://github.com/bonehurtingjuice/mattereddit)
|
||||||
|
* [Facebook messenger](https://github.com/VictorNine/fbridge)
|
||||||
|
* [Discourse](https://github.com/DeclanHoare/matterbabble)
|
||||||
|
|
||||||
#### API version
|
### API
|
||||||
* A dedicated user(bot) on your mattermost instance.
|
The API is very basic at the moment.
|
||||||
|
More info and examples on the [wiki](https://github.com/42wim/matterbridge/wiki/Api).
|
||||||
|
|
||||||
|
Used by the projects below. Feel free to make a PR to add your project to this list.
|
||||||
|
|
||||||
## building
|
* [MatterLink](https://github.com/elytra/MatterLink) (Matterbridge link for Minecraft Server chat)
|
||||||
Go 1.6+ is required. Make sure you have [Go](https://golang.org/doc/install) properly installed, including setting up your [GOPATH] (https://golang.org/doc/code.html#GOPATH)
|
* [pyCord](https://github.com/NikkyAI/pyCord) (crossplatform chatbot)
|
||||||
|
* [Mattereddit](https://github.com/bonehurtingjuice/mattereddit) (Reddit chat support)
|
||||||
|
* [fbridge](https://github.com/VictorNine/fbridge) (Facebook messenger support)
|
||||||
|
* [matterbabble](https://github.com/DeclanHoare/matterbabble) (Discourse support)
|
||||||
|
|
||||||
|
## Chat with us
|
||||||
|
|
||||||
|
Questions or want to test on your favorite platform? Join below:
|
||||||
|
|
||||||
|
* [Gitter][mb-gitter]
|
||||||
|
* [IRC][mb-irc]
|
||||||
|
* [Discord][mb-discord]
|
||||||
|
* [Matrix][mb-matrix]
|
||||||
|
* [Slack][mb-slack]
|
||||||
|
* [Mattermost][mb-mattermost]
|
||||||
|
* [Rocket.Chat][mb-rocketchat]
|
||||||
|
* [XMPP][mb-xmpp]
|
||||||
|
* [Twitch][mb-twitch]
|
||||||
|
* [Zulip][mb-zulip]
|
||||||
|
* [Telegram][mb-telegram]
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
See https://github.com/42wim/matterbridge/wiki
|
||||||
|
|
||||||
|
## Installing
|
||||||
|
### Binaries
|
||||||
|
* Latest stable release [v1.14.1](https://github.com/42wim/matterbridge/releases/latest)
|
||||||
|
* Development releases (follows master) can be downloaded [here](https://dl.bintray.com/42wim/nightly/)
|
||||||
|
|
||||||
|
### Packages
|
||||||
|
* [Overview](https://repology.org/metapackage/matterbridge/versions)
|
||||||
|
|
||||||
|
### Building
|
||||||
|
Go 1.9+ is required. Make sure you have [Go](https://golang.org/doc/install) properly installed, including setting up your [GOPATH](https://golang.org/doc/code.html#GOPATH).
|
||||||
|
|
||||||
|
After Go is setup, download matterbridge to your $GOPATH directory.
|
||||||
|
|
||||||
```
|
```
|
||||||
cd $GOPATH
|
cd $GOPATH
|
||||||
@ -66,10 +146,73 @@ $ ls bin/
|
|||||||
matterbridge
|
matterbridge
|
||||||
```
|
```
|
||||||
|
|
||||||
## running
|
## Configuration
|
||||||
1) Copy the matterbridge.conf.sample to matterbridge.conf in the same directory as the matterbridge binary.
|
### Basic configuration
|
||||||
2) Edit matterbridge.conf with the settings for your environment. See below for more config information.
|
See [howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) for a step by step walkthrough for creating your configuration.
|
||||||
3) Now you can run matterbridge.
|
|
||||||
|
### Advanced configuration
|
||||||
|
* [matterbridge.toml.sample](https://github.com/42wim/matterbridge/blob/master/matterbridge.toml.sample) for documentation and an example.
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
#### Bridge mattermost (off-topic) - irc (#testing)
|
||||||
|
```toml
|
||||||
|
[irc]
|
||||||
|
[irc.freenode]
|
||||||
|
Server="irc.freenode.net:6667"
|
||||||
|
Nick="yourbotname"
|
||||||
|
|
||||||
|
[mattermost]
|
||||||
|
[mattermost.work]
|
||||||
|
Server="yourmattermostserver.tld"
|
||||||
|
Team="yourteam"
|
||||||
|
Login="yourlogin"
|
||||||
|
Password="yourpass"
|
||||||
|
PrefixMessagesWithNick=true
|
||||||
|
RemoteNickFormat="[{PROTOCOL}] <{NICK}> "
|
||||||
|
|
||||||
|
[[gateway]]
|
||||||
|
name="mygateway"
|
||||||
|
enable=true
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="irc.freenode"
|
||||||
|
channel="#testing"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="mattermost.work"
|
||||||
|
channel="off-topic"
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Bridge slack (#general) - discord (general)
|
||||||
|
```toml
|
||||||
|
[slack]
|
||||||
|
[slack.test]
|
||||||
|
Token="yourslacktoken"
|
||||||
|
PrefixMessagesWithNick=true
|
||||||
|
|
||||||
|
[discord]
|
||||||
|
[discord.test]
|
||||||
|
Token="yourdiscordtoken"
|
||||||
|
Server="yourdiscordservername"
|
||||||
|
|
||||||
|
[general]
|
||||||
|
RemoteNickFormat="[{PROTOCOL}/{BRIDGE}] <{NICK}> "
|
||||||
|
|
||||||
|
[[gateway]]
|
||||||
|
name = "mygateway"
|
||||||
|
enable=true
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account = "discord.test"
|
||||||
|
channel="general"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account ="slack.test"
|
||||||
|
channel = "general"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Running
|
||||||
|
|
||||||
|
See [howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) for a step by step walkthrough for creating your configuration.
|
||||||
|
|
||||||
```
|
```
|
||||||
Usage of ./matterbridge:
|
Usage of ./matterbridge:
|
||||||
@ -77,39 +220,81 @@ Usage of ./matterbridge:
|
|||||||
config file (default "matterbridge.toml")
|
config file (default "matterbridge.toml")
|
||||||
-debug
|
-debug
|
||||||
enable debug
|
enable debug
|
||||||
|
-gops
|
||||||
|
enable gops agent
|
||||||
-version
|
-version
|
||||||
show version
|
show version
|
||||||
```
|
```
|
||||||
|
|
||||||
## config
|
### Docker
|
||||||
### matterbridge
|
Create your matterbridge.toml file locally eg in `/tmp/matterbridge.toml`
|
||||||
matterbridge looks for matterbridge.toml in current directory. (use -conf to specify another file)
|
```
|
||||||
|
docker run -ti -v /tmp/matterbridge.toml:/matterbridge.toml 42wim/matterbridge
|
||||||
|
```
|
||||||
|
|
||||||
Look at [matterbridge.toml.sample] (https://github.com/42wim/matterbridge/blob/master/matterbridge.toml.sample) for an example.
|
## Changelog
|
||||||
|
See [changelog.md](https://github.com/42wim/matterbridge/blob/master/changelog.md)
|
||||||
### mattermost
|
|
||||||
#### webhooks version
|
|
||||||
You'll have to configure the incoming and outgoing webhooks.
|
|
||||||
|
|
||||||
* incoming webhooks
|
|
||||||
Go to "account settings" - integrations - "incoming webhooks".
|
|
||||||
Choose a channel at "Add a new incoming webhook", this will create a webhook URL right below.
|
|
||||||
This URL should be set in the matterbridge.conf in the [mattermost] section (see above)
|
|
||||||
|
|
||||||
* outgoing webhooks
|
|
||||||
Go to "account settings" - integrations - "outgoing webhooks".
|
|
||||||
Choose a channel (the same as the one from incoming webhooks) and fill in the address and port of the server matterbridge will run on.
|
|
||||||
|
|
||||||
e.g. http://192.168.1.1:9999 (192.168.1.1:9999 is the BindAddress specified in [mattermost] section of matterbridge.conf)
|
|
||||||
|
|
||||||
## FAQ
|
## FAQ
|
||||||
Please look at [matterbridge.toml.sample] (https://github.com/42wim/matterbridge/blob/master/matterbridge.toml.sample) for more information first.
|
|
||||||
### Mattermost doesn't show the IRC nicks
|
|
||||||
If you're running the webhooks version, this can be fixed by either:
|
|
||||||
* enabling "override usernames". See [mattermost documentation](http://docs.mattermost.com/developer/webhooks-incoming.html#enabling-incoming-webhooks)
|
|
||||||
* setting ```PrefixMessagesWithNick``` to ```true``` in ```mattermost``` section of your matterbridge.toml.
|
|
||||||
|
|
||||||
If you're running the plus version you'll need to:
|
See [FAQ](https://github.com/42wim/matterbridge/wiki/FAQ)
|
||||||
* setting ```PrefixMessagesWithNick``` to ```true``` in ```mattermost``` section of your matterbridge.toml.
|
|
||||||
|
|
||||||
Also look at the ```RemoteNickFormat``` setting.
|
## Related projects
|
||||||
|
* [FOSSRIT/infrastructure - roles/matterbridge](https://github.com/FOSSRIT/infrastructure/tree/master/roles/matterbridge) (Ansible role used to automate deployments of Matterbridge)
|
||||||
|
* [matterbridge autoconfig](https://github.com/patcon/matterbridge-autoconfig)
|
||||||
|
* [matterbridge config viewer](https://github.com/patcon/matterbridge-heroku-viewer)
|
||||||
|
* [matterbridge-heroku](https://github.com/cadecairos/matterbridge-heroku)
|
||||||
|
* [mattereddit](https://github.com/bonehurtingjuice/mattereddit)
|
||||||
|
* [matterlink](https://github.com/elytra/MatterLink)
|
||||||
|
* [mattermost-plugin](https://github.com/matterbridge/mattermost-plugin) - Run matterbridge as a plugin in mattermost
|
||||||
|
* [pyCord](https://github.com/NikkyAI/pyCord) (crossplatform chatbot)
|
||||||
|
* [fbridge](https://github.com/VictorNine/fbridge) (Facebook messenger support)
|
||||||
|
* [isla](https://github.com/alphachung/isla) (Bot for Discord-Telegram groups used alongside matterbridge)
|
||||||
|
* [matterbabble](https://github.com/DeclanHoare/matterbabble) (Connect Discourse threads to Matterbridge)
|
||||||
|
|
||||||
|
## Articles
|
||||||
|
* [matterbridge on kubernetes](https://medium.freecodecamp.org/using-kubernetes-to-deploy-a-chat-gateway-or-when-technology-works-like-its-supposed-to-a169a8cd69a3)
|
||||||
|
* https://mattermost.com/blog/connect-irc-to-mattermost/
|
||||||
|
* https://blog.valvin.fr/2016/09/17/mattermost-et-un-channel-irc-cest-possible/
|
||||||
|
* https://blog.brightscout.com/top-10-mattermost-integrations/
|
||||||
|
* http://bencey.co.nz/2018/09/17/bridge/
|
||||||
|
* https://www.algoo.fr/blog/2018/01/19/recouvrez-votre-liberte-en-quittant-slack-pour-un-mattermost-auto-heberge/
|
||||||
|
* https://kopano.com/blog/matterbridge-bridging-mattermost-chat/
|
||||||
|
* https://www.stitcher.com/s/?eid=52382713
|
||||||
|
* https://daniele.tech/2019/02/how-to-use-matterbridge-to-connect-2-different-slack-workspaces/
|
||||||
|
|
||||||
|
## Thanks
|
||||||
|
[](https://www.digitalocean.com/) for sponsoring demo/testing droplets.
|
||||||
|
|
||||||
|
Matterbridge wouldn't exist without these libraries:
|
||||||
|
* discord - https://github.com/bwmarrin/discordgo
|
||||||
|
* echo - https://github.com/labstack/echo
|
||||||
|
* gitter - https://github.com/sromku/go-gitter
|
||||||
|
* gops - https://github.com/google/gops
|
||||||
|
* gozulipbot - https://github.com/ifo/gozulipbot
|
||||||
|
* irc - https://github.com/lrstanley/girc
|
||||||
|
* mattermost - https://github.com/mattermost/mattermost-server
|
||||||
|
* matrix - https://github.com/matrix-org/gomatrix
|
||||||
|
* sshchat - https://github.com/shazow/ssh-chat
|
||||||
|
* slack - https://github.com/nlopes/slack
|
||||||
|
* steam - https://github.com/Philipp15b/go-steam
|
||||||
|
* telegram - https://github.com/go-telegram-bot-api/telegram-bot-api
|
||||||
|
* xmpp - https://github.com/mattn/go-xmpp
|
||||||
|
* whatsapp - https://github.com/Rhymen/go-whatsapp/
|
||||||
|
* zulip - https://github.com/ifo/gozulipbot
|
||||||
|
* tengo - https://github.com/d5/tengo
|
||||||
|
|
||||||
|
<!-- Links -->
|
||||||
|
|
||||||
|
[mb-gitter]: https://gitter.im/42wim/matterbridge
|
||||||
|
[mb-irc]: https://webchat.freenode.net/?channels=matterbridgechat
|
||||||
|
[mb-discord]: https://discord.gg/AkKPtrQ
|
||||||
|
[mb-matrix]: https://riot.im/app/#/room/#matterbridge:matrix.org
|
||||||
|
[mb-slack]: https://join.slack.com/matterbridgechat/shared_invite/MjEwODMxNjU1NDMwLTE0OTk2MTU3NTMtMzZkZmRiNDZhOA
|
||||||
|
[mb-mattermost]: https://framateam.org/signup_user_complete/?id=tfqm33ggop8x3qgu4boeieta6e
|
||||||
|
[mb-rocketchat]: https://open.rocket.chat/channel/matterbridge
|
||||||
|
[mb-xmpp]: https://inverse.chat/
|
||||||
|
[mb-twitch]: https://www.twitch.tv/matterbridge
|
||||||
|
[mb-whatsapp]: https://www.whatsapp.com/
|
||||||
|
[mb-zulip]: https://matterbridge.zulipchat.com/register/
|
||||||
|
[mb-telegram]: https://t.me/Matterbridge
|
||||||
|
130
bridge/api/api.go
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"net/http"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/labstack/echo/v4"
|
||||||
|
"github.com/labstack/echo/v4/middleware"
|
||||||
|
"github.com/zfjagann/golang-ring"
|
||||||
|
)
|
||||||
|
|
||||||
|
type API struct {
|
||||||
|
Messages ring.Ring
|
||||||
|
sync.RWMutex
|
||||||
|
*bridge.Config
|
||||||
|
}
|
||||||
|
|
||||||
|
type Message struct {
|
||||||
|
Text string `json:"text"`
|
||||||
|
Username string `json:"username"`
|
||||||
|
UserID string `json:"userid"`
|
||||||
|
Avatar string `json:"avatar"`
|
||||||
|
Gateway string `json:"gateway"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
b := &API{Config: cfg}
|
||||||
|
e := echo.New()
|
||||||
|
e.HideBanner = true
|
||||||
|
e.HidePort = true
|
||||||
|
b.Messages = ring.Ring{}
|
||||||
|
if b.GetInt("Buffer") != 0 {
|
||||||
|
b.Messages.SetCapacity(b.GetInt("Buffer"))
|
||||||
|
}
|
||||||
|
if b.GetString("Token") != "" {
|
||||||
|
e.Use(middleware.KeyAuth(func(key string, c echo.Context) (bool, error) {
|
||||||
|
return key == b.GetString("Token"), nil
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
e.GET("/api/health", b.handleHealthcheck)
|
||||||
|
e.GET("/api/messages", b.handleMessages)
|
||||||
|
e.GET("/api/stream", b.handleStream)
|
||||||
|
e.POST("/api/message", b.handlePostMessage)
|
||||||
|
go func() {
|
||||||
|
if b.GetString("BindAddress") == "" {
|
||||||
|
b.Log.Fatalf("No BindAddress configured.")
|
||||||
|
}
|
||||||
|
b.Log.Infof("Listening on %s", b.GetString("BindAddress"))
|
||||||
|
b.Log.Fatal(e.Start(b.GetString("BindAddress")))
|
||||||
|
}()
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *API) Connect() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
func (b *API) Disconnect() error {
|
||||||
|
return nil
|
||||||
|
|
||||||
|
}
|
||||||
|
func (b *API) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
return nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *API) Send(msg config.Message) (string, error) {
|
||||||
|
b.Lock()
|
||||||
|
defer b.Unlock()
|
||||||
|
// ignore delete messages
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
b.Messages.Enqueue(&msg)
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *API) handleHealthcheck(c echo.Context) error {
|
||||||
|
return c.String(http.StatusOK, "OK")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *API) handlePostMessage(c echo.Context) error {
|
||||||
|
message := config.Message{}
|
||||||
|
if err := c.Bind(&message); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// these values are fixed
|
||||||
|
message.Channel = "api"
|
||||||
|
message.Protocol = "api"
|
||||||
|
message.Account = b.Account
|
||||||
|
message.ID = ""
|
||||||
|
message.Timestamp = time.Now()
|
||||||
|
b.Log.Debugf("Sending message from %s on %s to gateway", message.Username, "api")
|
||||||
|
b.Remote <- message
|
||||||
|
return c.JSON(http.StatusOK, message)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *API) handleMessages(c echo.Context) error {
|
||||||
|
b.Lock()
|
||||||
|
defer b.Unlock()
|
||||||
|
c.JSONPretty(http.StatusOK, b.Messages.Values(), " ")
|
||||||
|
b.Messages = ring.Ring{}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *API) handleStream(c echo.Context) error {
|
||||||
|
c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationJSON)
|
||||||
|
c.Response().WriteHeader(http.StatusOK)
|
||||||
|
greet := config.Message{
|
||||||
|
Event: config.EventAPIConnected,
|
||||||
|
Timestamp: time.Now(),
|
||||||
|
}
|
||||||
|
if err := json.NewEncoder(c.Response()).Encode(greet); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
c.Response().Flush()
|
||||||
|
for {
|
||||||
|
msg := b.Messages.Dequeue()
|
||||||
|
if msg != nil {
|
||||||
|
if err := json.NewEncoder(c.Response()).Encode(msg); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
c.Response().Flush()
|
||||||
|
}
|
||||||
|
time.Sleep(200 * time.Millisecond)
|
||||||
|
}
|
||||||
|
}
|
148
bridge/bridge.go
@ -1,68 +1,120 @@
|
|||||||
package bridge
|
package bridge
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
|
||||||
"github.com/42wim/matterbridge/bridge/discord"
|
|
||||||
"github.com/42wim/matterbridge/bridge/gitter"
|
|
||||||
"github.com/42wim/matterbridge/bridge/irc"
|
|
||||||
"github.com/42wim/matterbridge/bridge/mattermost"
|
|
||||||
"github.com/42wim/matterbridge/bridge/rocketchat"
|
|
||||||
"github.com/42wim/matterbridge/bridge/slack"
|
|
||||||
"github.com/42wim/matterbridge/bridge/telegram"
|
|
||||||
"github.com/42wim/matterbridge/bridge/xmpp"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Bridger interface {
|
type Bridger interface {
|
||||||
Send(msg config.Message) error
|
Send(msg config.Message) (string, error)
|
||||||
Connect() error
|
Connect() error
|
||||||
JoinChannel(channel string) error
|
JoinChannel(channel config.ChannelInfo) error
|
||||||
|
Disconnect() error
|
||||||
}
|
}
|
||||||
|
|
||||||
type Bridge struct {
|
type Bridge struct {
|
||||||
Config config.Protocol
|
|
||||||
Bridger
|
Bridger
|
||||||
Name string
|
*sync.RWMutex
|
||||||
Account string
|
|
||||||
Protocol string
|
Name string
|
||||||
|
Account string
|
||||||
|
Protocol string
|
||||||
|
Channels map[string]config.ChannelInfo
|
||||||
|
Joined map[string]bool
|
||||||
|
ChannelMembers *config.ChannelMembers
|
||||||
|
Log *logrus.Entry
|
||||||
|
Config config.Config
|
||||||
|
General *config.Protocol
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(cfg *config.Config, bridge *config.Bridge, c chan config.Message) *Bridge {
|
type Config struct {
|
||||||
b := new(Bridge)
|
*Bridge
|
||||||
|
|
||||||
|
Remote chan config.Message
|
||||||
|
}
|
||||||
|
|
||||||
|
// Factory is the factory function to create a bridge
|
||||||
|
type Factory func(*Config) Bridger
|
||||||
|
|
||||||
|
func New(bridge *config.Bridge) *Bridge {
|
||||||
accInfo := strings.Split(bridge.Account, ".")
|
accInfo := strings.Split(bridge.Account, ".")
|
||||||
protocol := accInfo[0]
|
protocol := accInfo[0]
|
||||||
name := accInfo[1]
|
name := accInfo[1]
|
||||||
b.Name = name
|
|
||||||
b.Protocol = protocol
|
|
||||||
b.Account = bridge.Account
|
|
||||||
|
|
||||||
// override config from environment
|
return &Bridge{
|
||||||
config.OverrideCfgFromEnv(cfg, protocol, name)
|
RWMutex: new(sync.RWMutex),
|
||||||
switch protocol {
|
Channels: make(map[string]config.ChannelInfo),
|
||||||
case "mattermost":
|
Name: name,
|
||||||
b.Config = cfg.Mattermost[name]
|
Protocol: protocol,
|
||||||
b.Bridger = bmattermost.New(cfg.Mattermost[name], bridge.Account, c)
|
Account: bridge.Account,
|
||||||
case "irc":
|
Joined: make(map[string]bool),
|
||||||
b.Config = cfg.IRC[name]
|
|
||||||
b.Bridger = birc.New(cfg.IRC[name], bridge.Account, c)
|
|
||||||
case "gitter":
|
|
||||||
b.Config = cfg.Gitter[name]
|
|
||||||
b.Bridger = bgitter.New(cfg.Gitter[name], bridge.Account, c)
|
|
||||||
case "slack":
|
|
||||||
b.Config = cfg.Slack[name]
|
|
||||||
b.Bridger = bslack.New(cfg.Slack[name], bridge.Account, c)
|
|
||||||
case "xmpp":
|
|
||||||
b.Config = cfg.Xmpp[name]
|
|
||||||
b.Bridger = bxmpp.New(cfg.Xmpp[name], bridge.Account, c)
|
|
||||||
case "discord":
|
|
||||||
b.Config = cfg.Discord[name]
|
|
||||||
b.Bridger = bdiscord.New(cfg.Discord[name], bridge.Account, c)
|
|
||||||
case "telegram":
|
|
||||||
b.Config = cfg.Telegram[name]
|
|
||||||
b.Bridger = btelegram.New(cfg.Telegram[name], bridge.Account, c)
|
|
||||||
case "rocketchat":
|
|
||||||
b.Config = cfg.Rocketchat[name]
|
|
||||||
b.Bridger = brocketchat.New(cfg.Rocketchat[name], bridge.Account, c)
|
|
||||||
}
|
}
|
||||||
return b
|
}
|
||||||
|
|
||||||
|
func (b *Bridge) JoinChannels() error {
|
||||||
|
return b.joinChannels(b.Channels, b.Joined)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetChannelMembers sets the newMembers to the bridge ChannelMembers
|
||||||
|
func (b *Bridge) SetChannelMembers(newMembers *config.ChannelMembers) {
|
||||||
|
b.Lock()
|
||||||
|
b.ChannelMembers = newMembers
|
||||||
|
b.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bridge) joinChannels(channels map[string]config.ChannelInfo, exists map[string]bool) error {
|
||||||
|
for ID, channel := range channels {
|
||||||
|
if !exists[ID] {
|
||||||
|
b.Log.Infof("%s: joining %s (ID: %s)", b.Account, channel.Name, ID)
|
||||||
|
err := b.JoinChannel(channel)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
exists[ID] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bridge) GetBool(key string) bool {
|
||||||
|
val, ok := b.Config.GetBool(b.Account + "." + key)
|
||||||
|
if !ok {
|
||||||
|
val, _ = b.Config.GetBool("general." + key)
|
||||||
|
}
|
||||||
|
return val
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bridge) GetInt(key string) int {
|
||||||
|
val, ok := b.Config.GetInt(b.Account + "." + key)
|
||||||
|
if !ok {
|
||||||
|
val, _ = b.Config.GetInt("general." + key)
|
||||||
|
}
|
||||||
|
return val
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bridge) GetString(key string) string {
|
||||||
|
val, ok := b.Config.GetString(b.Account + "." + key)
|
||||||
|
if !ok {
|
||||||
|
val, _ = b.Config.GetString("general." + key)
|
||||||
|
}
|
||||||
|
return val
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bridge) GetStringSlice(key string) []string {
|
||||||
|
val, ok := b.Config.GetStringSlice(b.Account + "." + key)
|
||||||
|
if !ok {
|
||||||
|
val, _ = b.Config.GetStringSlice("general." + key)
|
||||||
|
}
|
||||||
|
return val
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bridge) GetStringSlice2D(key string) [][]string {
|
||||||
|
val, ok := b.Config.GetStringSlice2D(b.Account + "." + key)
|
||||||
|
if !ok {
|
||||||
|
val, _ = b.Config.GetStringSlice2D("general." + key)
|
||||||
|
}
|
||||||
|
return val
|
||||||
}
|
}
|
||||||
|
@ -1,65 +1,161 @@
|
|||||||
package config
|
package config
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/BurntSushi/toml"
|
"bytes"
|
||||||
"log"
|
"io/ioutil"
|
||||||
"os"
|
|
||||||
"reflect"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/fsnotify/fsnotify"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"github.com/spf13/viper"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
EVENT_JOIN_LEAVE = "join_leave"
|
EventJoinLeave = "join_leave"
|
||||||
|
EventTopicChange = "topic_change"
|
||||||
|
EventFailure = "failure"
|
||||||
|
EventFileFailureSize = "file_failure_size"
|
||||||
|
EventAvatarDownload = "avatar_download"
|
||||||
|
EventRejoinChannels = "rejoin_channels"
|
||||||
|
EventUserAction = "user_action"
|
||||||
|
EventMsgDelete = "msg_delete"
|
||||||
|
EventAPIConnected = "api_connected"
|
||||||
|
EventUserTyping = "user_typing"
|
||||||
|
EventGetChannelMembers = "get_channel_members"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Message struct {
|
type Message struct {
|
||||||
Text string
|
Text string `json:"text"`
|
||||||
Channel string
|
Channel string `json:"channel"`
|
||||||
Username string
|
Username string `json:"username"`
|
||||||
Avatar string
|
UserID string `json:"userid"` // userid on the bridge
|
||||||
Account string
|
Avatar string `json:"avatar"`
|
||||||
Event string
|
Account string `json:"account"`
|
||||||
|
Event string `json:"event"`
|
||||||
|
Protocol string `json:"protocol"`
|
||||||
|
Gateway string `json:"gateway"`
|
||||||
|
ParentID string `json:"parent_id"`
|
||||||
|
Timestamp time.Time `json:"timestamp"`
|
||||||
|
ID string `json:"id"`
|
||||||
|
Extra map[string][]interface{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type FileInfo struct {
|
||||||
|
Name string
|
||||||
|
Data *[]byte
|
||||||
|
Comment string
|
||||||
|
URL string
|
||||||
|
Size int64
|
||||||
|
Avatar bool
|
||||||
|
SHA string
|
||||||
|
}
|
||||||
|
|
||||||
|
type ChannelInfo struct {
|
||||||
|
Name string
|
||||||
|
Account string
|
||||||
|
Direction string
|
||||||
|
ID string
|
||||||
|
SameChannel map[string]bool
|
||||||
|
Options ChannelOptions
|
||||||
|
}
|
||||||
|
|
||||||
|
type ChannelMember struct {
|
||||||
|
Username string
|
||||||
|
Nick string
|
||||||
|
UserID string
|
||||||
|
ChannelID string
|
||||||
|
ChannelName string
|
||||||
|
}
|
||||||
|
|
||||||
|
type ChannelMembers []ChannelMember
|
||||||
|
|
||||||
type Protocol struct {
|
type Protocol struct {
|
||||||
BindAddress string // mattermost, slack
|
AuthCode string // steam
|
||||||
|
BindAddress string // mattermost, slack // DEPRECATED
|
||||||
|
Buffer int // api
|
||||||
|
Charset string // irc
|
||||||
|
ColorNicks bool // only irc for now
|
||||||
|
Debug bool // general
|
||||||
|
DebugLevel int // only for irc now
|
||||||
|
EditSuffix string // mattermost, slack, discord, telegram, gitter
|
||||||
|
EditDisable bool // mattermost, slack, discord, telegram, gitter
|
||||||
IconURL string // mattermost, slack
|
IconURL string // mattermost, slack
|
||||||
|
IgnoreFailureOnStart bool // general
|
||||||
IgnoreNicks string // all protocols
|
IgnoreNicks string // all protocols
|
||||||
|
IgnoreMessages string // all protocols
|
||||||
Jid string // xmpp
|
Jid string // xmpp
|
||||||
Login string // mattermost
|
Label string // all protocols
|
||||||
Muc string // xmpp
|
Login string // mattermost, matrix
|
||||||
Name string // all protocols
|
MediaDownloadBlackList []string
|
||||||
Nick string // all protocols
|
MediaDownloadPath string // Basically MediaServerUpload, but instead of uploading it, just write it to a file on the same server.
|
||||||
NickFormatter string // mattermost, slack
|
MediaDownloadSize int // all protocols
|
||||||
NickServNick string // IRC
|
MediaServerDownload string
|
||||||
NickServPassword string // IRC
|
MediaServerUpload string
|
||||||
NicksPerRow int // mattermost, slack
|
MediaConvertWebPToPNG bool // telegram
|
||||||
NoTLS bool // mattermost
|
MessageDelay int // IRC, time in millisecond to wait between messages
|
||||||
Password string // IRC,mattermost,XMPP
|
MessageFormat string // telegram
|
||||||
PrefixMessagesWithNick bool // mattemost, slack
|
MessageLength int // IRC, max length of a message allowed
|
||||||
Protocol string //all protocols
|
MessageQueue int // IRC, size of message queue for flood control
|
||||||
MessageQueue int // IRC, size of message queue for flood control
|
MessageSplit bool // IRC, split long messages with newlines on MessageLength instead of clipping
|
||||||
MessageDelay int // IRC, time in millisecond to wait between messages
|
Muc string // xmpp
|
||||||
RemoteNickFormat string // all protocols
|
Name string // all protocols
|
||||||
Server string // IRC,mattermost,XMPP,discord
|
Nick string // all protocols
|
||||||
ShowJoinPart bool // all protocols
|
NickFormatter string // mattermost, slack
|
||||||
SkipTLSVerify bool // IRC, mattermost
|
NickServNick string // IRC
|
||||||
Team string // mattermost
|
NickServUsername string // IRC
|
||||||
Token string // gitter, slack, discord
|
NickServPassword string // IRC
|
||||||
URL string // mattermost, slack
|
NicksPerRow int // mattermost, slack
|
||||||
UseAPI bool // mattermost, slack
|
NoHomeServerSuffix bool // matrix
|
||||||
UseSASL bool // IRC
|
NoSendJoinPart bool // all protocols
|
||||||
UseTLS bool // IRC
|
NoTLS bool // mattermost
|
||||||
|
Password string // IRC,mattermost,XMPP,matrix
|
||||||
|
PrefixMessagesWithNick bool // mattemost, slack
|
||||||
|
PreserveThreading bool // slack
|
||||||
|
Protocol string // all protocols
|
||||||
|
QuoteDisable bool // telegram
|
||||||
|
QuoteFormat string // telegram
|
||||||
|
RejoinDelay int // IRC
|
||||||
|
ReplaceMessages [][]string // all protocols
|
||||||
|
ReplaceNicks [][]string // all protocols
|
||||||
|
RemoteNickFormat string // all protocols
|
||||||
|
RunCommands []string // irc
|
||||||
|
Server string // IRC,mattermost,XMPP,discord
|
||||||
|
ShowJoinPart bool // all protocols
|
||||||
|
ShowTopicChange bool // slack
|
||||||
|
ShowUserTyping bool // slack
|
||||||
|
ShowEmbeds bool // discord
|
||||||
|
SkipTLSVerify bool // IRC, mattermost
|
||||||
|
StripNick bool // all protocols
|
||||||
|
SyncTopic bool // slack
|
||||||
|
TengoModifyMessage string // general
|
||||||
|
Team string // mattermost
|
||||||
|
Token string // gitter, slack, discord, api
|
||||||
|
Topic string // zulip
|
||||||
|
URL string // mattermost, slack // DEPRECATED
|
||||||
|
UseAPI bool // mattermost, slack
|
||||||
|
UseSASL bool // IRC
|
||||||
|
UseTLS bool // IRC
|
||||||
|
UseDiscriminator bool // discord
|
||||||
|
UseFirstName bool // telegram
|
||||||
|
UseUserName bool // discord
|
||||||
|
UseInsecureURL bool // telegram
|
||||||
|
WebhookBindAddress string // mattermost, slack
|
||||||
|
WebhookURL string // mattermost, slack
|
||||||
}
|
}
|
||||||
|
|
||||||
type ChannelOptions struct {
|
type ChannelOptions struct {
|
||||||
Key string // irc
|
Key string // irc, xmpp
|
||||||
|
WebhookURL string // discord
|
||||||
|
Topic string // zulip
|
||||||
}
|
}
|
||||||
|
|
||||||
type Bridge struct {
|
type Bridge struct {
|
||||||
Account string
|
Account string
|
||||||
Channel string
|
Channel string
|
||||||
Options ChannelOptions
|
Options ChannelOptions
|
||||||
|
SameChannel bool
|
||||||
}
|
}
|
||||||
|
|
||||||
type Gateway struct {
|
type Gateway struct {
|
||||||
@ -77,67 +173,140 @@ type SameChannelGateway struct {
|
|||||||
Accounts []string
|
Accounts []string
|
||||||
}
|
}
|
||||||
|
|
||||||
type Config struct {
|
type BridgeValues struct {
|
||||||
|
API map[string]Protocol
|
||||||
IRC map[string]Protocol
|
IRC map[string]Protocol
|
||||||
Mattermost map[string]Protocol
|
Mattermost map[string]Protocol
|
||||||
|
Matrix map[string]Protocol
|
||||||
Slack map[string]Protocol
|
Slack map[string]Protocol
|
||||||
|
SlackLegacy map[string]Protocol
|
||||||
|
Steam map[string]Protocol
|
||||||
Gitter map[string]Protocol
|
Gitter map[string]Protocol
|
||||||
Xmpp map[string]Protocol
|
XMPP map[string]Protocol
|
||||||
Discord map[string]Protocol
|
Discord map[string]Protocol
|
||||||
Telegram map[string]Protocol
|
Telegram map[string]Protocol
|
||||||
Rocketchat map[string]Protocol
|
Rocketchat map[string]Protocol
|
||||||
|
SSHChat map[string]Protocol
|
||||||
|
WhatsApp map[string]Protocol // TODO is this struct used? Search for "SlackLegacy" for example didn't return any results
|
||||||
|
Zulip map[string]Protocol
|
||||||
General Protocol
|
General Protocol
|
||||||
Gateway []Gateway
|
Gateway []Gateway
|
||||||
SameChannelGateway []SameChannelGateway
|
SameChannelGateway []SameChannelGateway
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewConfig(cfgfile string) *Config {
|
type Config interface {
|
||||||
var cfg Config
|
BridgeValues() *BridgeValues
|
||||||
if _, err := toml.DecodeFile(cfgfile, &cfg); err != nil {
|
GetBool(key string) (bool, bool)
|
||||||
log.Fatal(err)
|
GetInt(key string) (int, bool)
|
||||||
}
|
GetString(key string) (string, bool)
|
||||||
return &cfg
|
GetStringSlice(key string) ([]string, bool)
|
||||||
|
GetStringSlice2D(key string) ([][]string, bool)
|
||||||
}
|
}
|
||||||
|
|
||||||
func OverrideCfgFromEnv(cfg *Config, protocol string, account string) {
|
type config struct {
|
||||||
var protoCfg Protocol
|
sync.RWMutex
|
||||||
val := reflect.ValueOf(cfg).Elem()
|
|
||||||
// loop over the Config struct
|
logger *logrus.Entry
|
||||||
for i := 0; i < val.NumField(); i++ {
|
v *viper.Viper
|
||||||
typeField := val.Type().Field(i)
|
cv *BridgeValues
|
||||||
// look for the protocol map (both lowercase)
|
}
|
||||||
if strings.ToLower(typeField.Name) == protocol {
|
|
||||||
// get the Protocol struct from the map
|
// NewConfig instantiates a new configuration based on the specified configuration file path.
|
||||||
data := val.Field(i).MapIndex(reflect.ValueOf(account))
|
func NewConfig(rootLogger *logrus.Logger, cfgfile string) Config {
|
||||||
protoCfg = data.Interface().(Protocol)
|
logger := rootLogger.WithFields(logrus.Fields{"prefix": "config"})
|
||||||
protoStruct := reflect.ValueOf(&protoCfg).Elem()
|
|
||||||
// loop over the found protocol struct
|
viper.SetConfigFile(cfgfile)
|
||||||
for i := 0; i < protoStruct.NumField(); i++ {
|
input, err := ioutil.ReadFile(cfgfile)
|
||||||
typeField := protoStruct.Type().Field(i)
|
if err != nil {
|
||||||
// build our environment key (eg MATTERBRIDGE_MATTERMOST_WORK_LOGIN)
|
logger.Fatalf("Failed to read configuration file: %#v", err)
|
||||||
key := "matterbridge_" + protocol + "_" + account + "_" + typeField.Name
|
}
|
||||||
key = strings.ToUpper(key)
|
|
||||||
// search the environment
|
mycfg := newConfigFromString(logger, input)
|
||||||
res := os.Getenv(key)
|
if mycfg.cv.General.MediaDownloadSize == 0 {
|
||||||
// if it exists and the current field is a string
|
mycfg.cv.General.MediaDownloadSize = 1000000
|
||||||
// then update the current field
|
}
|
||||||
if res != "" {
|
viper.WatchConfig()
|
||||||
fieldVal := protoStruct.Field(i)
|
viper.OnConfigChange(func(e fsnotify.Event) {
|
||||||
if fieldVal.Kind() == reflect.String {
|
logger.Println("Config file changed:", e.Name)
|
||||||
log.Printf("config: overriding %s from env with %s\n", key, res)
|
})
|
||||||
fieldVal.Set(reflect.ValueOf(res))
|
return mycfg
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
// NewConfigFromString instantiates a new configuration based on the specified string.
|
||||||
// update the map with the modified Protocol (cfg.Protocol[account] = Protocol)
|
func NewConfigFromString(rootLogger *logrus.Logger, input []byte) Config {
|
||||||
val.Field(i).SetMapIndex(reflect.ValueOf(account), reflect.ValueOf(protoCfg))
|
logger := rootLogger.WithFields(logrus.Fields{"prefix": "config"})
|
||||||
break
|
return newConfigFromString(logger, input)
|
||||||
|
}
|
||||||
|
|
||||||
|
func newConfigFromString(logger *logrus.Entry, input []byte) *config {
|
||||||
|
viper.SetConfigType("toml")
|
||||||
|
viper.SetEnvPrefix("matterbridge")
|
||||||
|
viper.SetEnvKeyReplacer(strings.NewReplacer(".", "_", "-", "_"))
|
||||||
|
viper.AutomaticEnv()
|
||||||
|
|
||||||
|
if err := viper.ReadConfig(bytes.NewBuffer(input)); err != nil {
|
||||||
|
logger.Fatalf("Failed to parse the configuration: %#v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := &BridgeValues{}
|
||||||
|
if err := viper.Unmarshal(cfg); err != nil {
|
||||||
|
logger.Fatalf("Failed to load the configuration: %#v", err)
|
||||||
|
}
|
||||||
|
return &config{
|
||||||
|
logger: logger,
|
||||||
|
v: viper.GetViper(),
|
||||||
|
cv: cfg,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *config) BridgeValues() *BridgeValues {
|
||||||
|
return c.cv
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *config) GetBool(key string) (bool, bool) {
|
||||||
|
c.RLock()
|
||||||
|
defer c.RUnlock()
|
||||||
|
return c.v.GetBool(key), c.v.IsSet(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *config) GetInt(key string) (int, bool) {
|
||||||
|
c.RLock()
|
||||||
|
defer c.RUnlock()
|
||||||
|
return c.v.GetInt(key), c.v.IsSet(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *config) GetString(key string) (string, bool) {
|
||||||
|
c.RLock()
|
||||||
|
defer c.RUnlock()
|
||||||
|
return c.v.GetString(key), c.v.IsSet(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *config) GetStringSlice(key string) ([]string, bool) {
|
||||||
|
c.RLock()
|
||||||
|
defer c.RUnlock()
|
||||||
|
return c.v.GetStringSlice(key), c.v.IsSet(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *config) GetStringSlice2D(key string) ([][]string, bool) {
|
||||||
|
c.RLock()
|
||||||
|
defer c.RUnlock()
|
||||||
|
|
||||||
|
res, ok := c.v.Get(key).([]interface{})
|
||||||
|
if !ok {
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
var result [][]string
|
||||||
|
for _, entry := range res {
|
||||||
|
result2 := []string{}
|
||||||
|
for _, entry2 := range entry.([]interface{}) {
|
||||||
|
result2 = append(result2, entry2.(string))
|
||||||
}
|
}
|
||||||
|
result = append(result, result2)
|
||||||
}
|
}
|
||||||
|
return result, true
|
||||||
}
|
}
|
||||||
|
|
||||||
func GetIconURL(msg *Message, cfg *Protocol) string {
|
func GetIconURL(msg *Message, iconURL string) string {
|
||||||
iconURL := cfg.IconURL
|
|
||||||
info := strings.Split(msg.Account, ".")
|
info := strings.Split(msg.Account, ".")
|
||||||
protocol := info[0]
|
protocol := info[0]
|
||||||
name := info[1]
|
name := info[1]
|
||||||
@ -146,3 +315,45 @@ func GetIconURL(msg *Message, cfg *Protocol) string {
|
|||||||
iconURL = strings.Replace(iconURL, "{PROTOCOL}", protocol, -1)
|
iconURL = strings.Replace(iconURL, "{PROTOCOL}", protocol, -1)
|
||||||
return iconURL
|
return iconURL
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type TestConfig struct {
|
||||||
|
Config
|
||||||
|
|
||||||
|
Overrides map[string]interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *TestConfig) GetBool(key string) (bool, bool) {
|
||||||
|
val, ok := c.Overrides[key]
|
||||||
|
if ok {
|
||||||
|
return val.(bool), true
|
||||||
|
}
|
||||||
|
return c.Config.GetBool(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *TestConfig) GetInt(key string) (int, bool) {
|
||||||
|
if val, ok := c.Overrides[key]; ok {
|
||||||
|
return val.(int), true
|
||||||
|
}
|
||||||
|
return c.Config.GetInt(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *TestConfig) GetString(key string) (string, bool) {
|
||||||
|
if val, ok := c.Overrides[key]; ok {
|
||||||
|
return val.(string), true
|
||||||
|
}
|
||||||
|
return c.Config.GetString(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *TestConfig) GetStringSlice(key string) ([]string, bool) {
|
||||||
|
if val, ok := c.Overrides[key]; ok {
|
||||||
|
return val.([]string), true
|
||||||
|
}
|
||||||
|
return c.Config.GetStringSlice(key)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *TestConfig) GetStringSlice2D(key string) ([][]string, bool) {
|
||||||
|
if val, ok := c.Overrides[key]; ok {
|
||||||
|
return val.([][]string), true
|
||||||
|
}
|
||||||
|
return c.Config.GetStringSlice2D(key)
|
||||||
|
}
|
||||||
|
@ -1,137 +1,367 @@
|
|||||||
package bdiscord
|
package bdiscord
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
"bytes"
|
||||||
log "github.com/Sirupsen/logrus"
|
"errors"
|
||||||
"github.com/bwmarrin/discordgo"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/bwmarrin/discordgo"
|
||||||
)
|
)
|
||||||
|
|
||||||
type bdiscord struct {
|
const MessageLength = 1950
|
||||||
c *discordgo.Session
|
|
||||||
Config *config.Protocol
|
type Bdiscord struct {
|
||||||
Remote chan config.Message
|
*bridge.Config
|
||||||
Account string
|
|
||||||
Channels []*discordgo.Channel
|
c *discordgo.Session
|
||||||
Nick string
|
|
||||||
UseChannelID bool
|
nick string
|
||||||
|
useChannelID bool
|
||||||
|
guildID string
|
||||||
|
webhookID string
|
||||||
|
webhookToken string
|
||||||
|
canEditWebhooks bool
|
||||||
|
|
||||||
|
channelsMutex sync.RWMutex
|
||||||
|
channels []*discordgo.Channel
|
||||||
|
channelInfoMap map[string]*config.ChannelInfo
|
||||||
|
|
||||||
|
membersMutex sync.RWMutex
|
||||||
|
userMemberMap map[string]*discordgo.Member
|
||||||
|
nickMemberMap map[string]*discordgo.Member
|
||||||
}
|
}
|
||||||
|
|
||||||
var flog *log.Entry
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
var protocol = "discord"
|
b := &Bdiscord{Config: cfg}
|
||||||
|
b.userMemberMap = make(map[string]*discordgo.Member)
|
||||||
func init() {
|
b.nickMemberMap = make(map[string]*discordgo.Member)
|
||||||
flog = log.WithFields(log.Fields{"module": protocol})
|
b.channelInfoMap = make(map[string]*config.ChannelInfo)
|
||||||
}
|
if b.GetString("WebhookURL") != "" {
|
||||||
|
b.Log.Debug("Configuring Discord Incoming Webhook")
|
||||||
func New(cfg config.Protocol, account string, c chan config.Message) *bdiscord {
|
b.webhookID, b.webhookToken = b.splitURL(b.GetString("WebhookURL"))
|
||||||
b := &bdiscord{}
|
}
|
||||||
b.Config = &cfg
|
|
||||||
b.Remote = c
|
|
||||||
b.Account = account
|
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *bdiscord) Connect() error {
|
func (b *Bdiscord) Connect() error {
|
||||||
var err error
|
var err error
|
||||||
flog.Info("Connecting")
|
var guildFound bool
|
||||||
if !strings.HasPrefix(b.Config.Token, "Bot ") {
|
token := b.GetString("Token")
|
||||||
b.Config.Token = "Bot " + b.Config.Token
|
b.Log.Info("Connecting")
|
||||||
|
if b.GetString("WebhookURL") == "" {
|
||||||
|
b.Log.Info("Connecting using token")
|
||||||
|
} else {
|
||||||
|
b.Log.Info("Connecting using webhookurl (for posting) and token")
|
||||||
}
|
}
|
||||||
b.c, err = discordgo.New(b.Config.Token)
|
if !strings.HasPrefix(b.GetString("Token"), "Bot ") {
|
||||||
|
token = "Bot " + b.GetString("Token")
|
||||||
|
}
|
||||||
|
// if we have a User token, remove the `Bot` prefix
|
||||||
|
if strings.HasPrefix(b.GetString("Token"), "User ") {
|
||||||
|
token = strings.Replace(b.GetString("Token"), "User ", "", -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
b.c, err = discordgo.New(token)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
flog.Info("Connection succeeded")
|
b.Log.Info("Connection succeeded")
|
||||||
b.c.AddHandler(b.messageCreate)
|
b.c.AddHandler(b.messageCreate)
|
||||||
|
b.c.AddHandler(b.memberUpdate)
|
||||||
|
b.c.AddHandler(b.messageUpdate)
|
||||||
|
b.c.AddHandler(b.messageDelete)
|
||||||
|
b.c.AddHandler(b.memberAdd)
|
||||||
|
b.c.AddHandler(b.memberRemove)
|
||||||
err = b.c.Open()
|
err = b.c.Open()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
guilds, err := b.c.UserGuilds()
|
guilds, err := b.c.UserGuilds(100, "", "")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
userinfo, err := b.c.User("@me")
|
userinfo, err := b.c.User("@me")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
b.Nick = userinfo.Username
|
serverName := strings.Replace(b.GetString("Server"), "ID:", "", -1)
|
||||||
|
b.nick = userinfo.Username
|
||||||
|
b.channelsMutex.Lock()
|
||||||
for _, guild := range guilds {
|
for _, guild := range guilds {
|
||||||
if guild.Name == b.Config.Server {
|
if guild.Name == serverName || guild.ID == serverName {
|
||||||
b.Channels, err = b.c.GuildChannels(guild.ID)
|
b.channels, err = b.c.GuildChannels(guild.ID)
|
||||||
|
b.guildID = guild.ID
|
||||||
|
guildFound = true
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
break
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
b.channelsMutex.Unlock()
|
||||||
}
|
if !guildFound {
|
||||||
|
msg := fmt.Sprintf("Server \"%s\" not found", b.GetString("Server"))
|
||||||
|
err = errors.New(msg)
|
||||||
|
b.Log.Error(msg)
|
||||||
|
b.Log.Info("Possible values:")
|
||||||
|
for _, guild := range guilds {
|
||||||
|
b.Log.Infof("Server=\"%s\" # Server name", guild.Name)
|
||||||
|
b.Log.Infof("Server=\"%s\" # Server ID", guild.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (b *bdiscord) JoinChannel(channel string) error {
|
if err != nil {
|
||||||
idcheck := strings.Split(channel, "ID:")
|
return err
|
||||||
if len(idcheck) > 1 {
|
}
|
||||||
b.UseChannelID = true
|
b.channelsMutex.RLock()
|
||||||
|
if b.GetString("WebhookURL") == "" {
|
||||||
|
for _, channel := range b.channels {
|
||||||
|
b.Log.Debugf("found channel %#v", channel)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
b.canEditWebhooks = true
|
||||||
|
for _, channel := range b.channels {
|
||||||
|
b.Log.Debugf("found channel %#v; verifying PermissionManageWebhooks", channel)
|
||||||
|
perms, permsErr := b.c.State.UserChannelPermissions(userinfo.ID, channel.ID)
|
||||||
|
manageWebhooks := discordgo.PermissionManageWebhooks
|
||||||
|
if permsErr != nil || perms&manageWebhooks != manageWebhooks {
|
||||||
|
b.Log.Warnf("Can't manage webhooks in channel \"%s\"", channel.Name)
|
||||||
|
b.canEditWebhooks = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if b.canEditWebhooks {
|
||||||
|
b.Log.Info("Can manage webhooks; will edit channel for global webhook on send")
|
||||||
|
} else {
|
||||||
|
b.Log.Warn("Can't manage webhooks; won't edit channel for global webhook on send")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.channelsMutex.RUnlock()
|
||||||
|
|
||||||
|
// Obtaining guild members and initializing nickname mapping.
|
||||||
|
b.membersMutex.Lock()
|
||||||
|
defer b.membersMutex.Unlock()
|
||||||
|
members, err := b.c.GuildMembers(b.guildID, "", 1000)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Error("Error obtaining server members: ", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, member := range members {
|
||||||
|
if member == nil {
|
||||||
|
b.Log.Warnf("Skipping missing information for a user.")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
b.userMemberMap[member.User.ID] = member
|
||||||
|
b.nickMemberMap[member.User.Username] = member
|
||||||
|
if member.Nick != "" {
|
||||||
|
b.nickMemberMap[member.Nick] = member
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *bdiscord) Send(msg config.Message) error {
|
func (b *Bdiscord) Disconnect() error {
|
||||||
flog.Debugf("Receiving %#v", msg)
|
return b.c.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
b.channelsMutex.Lock()
|
||||||
|
defer b.channelsMutex.Unlock()
|
||||||
|
|
||||||
|
b.channelInfoMap[channel.ID] = &channel
|
||||||
|
idcheck := strings.Split(channel.Name, "ID:")
|
||||||
|
if len(idcheck) > 1 {
|
||||||
|
b.useChannelID = true
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) Send(msg config.Message) (string, error) {
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
|
||||||
channelID := b.getChannelID(msg.Channel)
|
channelID := b.getChannelID(msg.Channel)
|
||||||
if channelID == "" {
|
if channelID == "" {
|
||||||
flog.Errorf("Could not find channelID for %v", msg.Channel)
|
return "", fmt.Errorf("Could not find channelID for %v", msg.Channel)
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
b.c.ChannelMessageSend(channelID, msg.Username+msg.Text)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *bdiscord) messageCreate(s *discordgo.Session, m *discordgo.MessageCreate) {
|
// Make a action /me of the message
|
||||||
// not relay our own messages
|
if msg.Event == config.EventUserAction {
|
||||||
if m.Author.Username == b.Nick {
|
msg.Text = "_" + msg.Text + "_"
|
||||||
return
|
|
||||||
}
|
}
|
||||||
if len(m.Attachments) > 0 {
|
|
||||||
for _, attach := range m.Attachments {
|
// use initial webhook configured for the entire Discord account
|
||||||
m.Content = m.Content + "\n" + attach.URL
|
isGlobalWebhook := true
|
||||||
|
wID := b.webhookID
|
||||||
|
wToken := b.webhookToken
|
||||||
|
|
||||||
|
// check if have a channel specific webhook
|
||||||
|
b.channelsMutex.RLock()
|
||||||
|
if ci, ok := b.channelInfoMap[msg.Channel+b.Account]; ok {
|
||||||
|
if ci.Options.WebhookURL != "" {
|
||||||
|
wID, wToken = b.splitURL(ci.Options.WebhookURL)
|
||||||
|
isGlobalWebhook = false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if m.Content == "" {
|
b.channelsMutex.RUnlock()
|
||||||
return
|
|
||||||
}
|
|
||||||
flog.Debugf("Sending message from %s on %s to gateway", m.Author.Username, b.Account)
|
|
||||||
channelName := b.getChannelName(m.ChannelID)
|
|
||||||
if b.UseChannelID {
|
|
||||||
channelName = "ID:" + m.ChannelID
|
|
||||||
}
|
|
||||||
b.Remote <- config.Message{Username: m.Author.Username, Text: m.ContentWithMentionsReplaced(), Channel: channelName,
|
|
||||||
Account: b.Account, Avatar: "https://cdn.discordapp.com/avatars/" + m.Author.ID + "/" + m.Author.Avatar + ".jpg"}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *bdiscord) getChannelID(name string) string {
|
// Use webhook to send the message
|
||||||
idcheck := strings.Split(name, "ID:")
|
if wID != "" {
|
||||||
if len(idcheck) > 1 {
|
// skip events
|
||||||
return idcheck[1]
|
if msg.Event != "" && msg.Event != config.EventJoinLeave && msg.Event != config.EventTopicChange {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
b.Log.Debugf("Broadcasting using Webhook")
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text += fi.Comment + ": "
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text = fi.URL
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text = fi.Comment + ": " + fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// skip empty messages
|
||||||
|
if msg.Text == "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
msg.Text = helper.ClipMessage(msg.Text, MessageLength)
|
||||||
|
msg.Text = b.replaceUserMentions(msg.Text)
|
||||||
|
// discord username must be [0..32] max
|
||||||
|
if len(msg.Username) > 32 {
|
||||||
|
msg.Username = msg.Username[0:32]
|
||||||
|
}
|
||||||
|
// if we have a global webhook for this Discord account, and permission
|
||||||
|
// to modify webhooks (previously verified), then set its channel to
|
||||||
|
// the message channel before using it
|
||||||
|
// TODO: this isn't necessary if the last message from this webhook was
|
||||||
|
// sent to the current channel
|
||||||
|
if isGlobalWebhook && b.canEditWebhooks {
|
||||||
|
b.Log.Debugf("Setting webhook channel to \"%s\"", msg.Channel)
|
||||||
|
_, err := b.c.WebhookEdit(wID, "", "", channelID)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("Could not set webhook channel: %s", err)
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
err := b.c.WebhookExecute(
|
||||||
|
wID,
|
||||||
|
wToken,
|
||||||
|
true,
|
||||||
|
&discordgo.WebhookParams{
|
||||||
|
Content: msg.Text,
|
||||||
|
Username: msg.Username,
|
||||||
|
AvatarURL: msg.Avatar,
|
||||||
|
})
|
||||||
|
return "", err
|
||||||
}
|
}
|
||||||
for _, channel := range b.Channels {
|
|
||||||
if channel.Name == name {
|
b.Log.Debugf("Broadcasting using token (API)")
|
||||||
return channel.ID
|
|
||||||
|
// Delete message
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
if msg.ID == "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
err := b.c.ChannelMessageDelete(channelID, msg.ID)
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload a file if it exists
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
rmsg.Text = helper.ClipMessage(rmsg.Text, MessageLength)
|
||||||
|
if _, err := b.c.ChannelMessageSend(channelID, rmsg.Username+rmsg.Text); err != nil {
|
||||||
|
b.Log.Errorf("Could not send message %#v: %s", rmsg, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// check if we have files to upload (from slack, telegram or mattermost)
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
return b.handleUploadFile(&msg, channelID)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return ""
|
|
||||||
|
msg.Text = helper.ClipMessage(msg.Text, MessageLength)
|
||||||
|
msg.Text = b.replaceUserMentions(msg.Text)
|
||||||
|
|
||||||
|
// Edit message
|
||||||
|
if msg.ID != "" {
|
||||||
|
_, err := b.c.ChannelMessageEdit(channelID, msg.ID, msg.Username+msg.Text)
|
||||||
|
return msg.ID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Post normal message
|
||||||
|
res, err := b.c.ChannelMessageSend(channelID, msg.Username+msg.Text)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return res.ID, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *bdiscord) getChannelName(id string) string {
|
// useWebhook returns true if we have a webhook defined somewhere
|
||||||
for _, channel := range b.Channels {
|
func (b *Bdiscord) useWebhook() bool {
|
||||||
if channel.ID == id {
|
if b.GetString("WebhookURL") != "" {
|
||||||
return channel.Name
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
b.channelsMutex.RLock()
|
||||||
|
defer b.channelsMutex.RUnlock()
|
||||||
|
|
||||||
|
for _, channel := range b.channelInfoMap {
|
||||||
|
if channel.Options.WebhookURL != "" {
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return ""
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// isWebhookID returns true if the specified id is used in a defined webhook
|
||||||
|
func (b *Bdiscord) isWebhookID(id string) bool {
|
||||||
|
if b.GetString("WebhookURL") != "" {
|
||||||
|
wID, _ := b.splitURL(b.GetString("WebhookURL"))
|
||||||
|
if wID == id {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
b.channelsMutex.RLock()
|
||||||
|
defer b.channelsMutex.RUnlock()
|
||||||
|
|
||||||
|
for _, channel := range b.channelInfoMap {
|
||||||
|
if channel.Options.WebhookURL != "" {
|
||||||
|
wID, _ := b.splitURL(channel.Options.WebhookURL)
|
||||||
|
if wID == id {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleUploadFile handles native upload of files
|
||||||
|
func (b *Bdiscord) handleUploadFile(msg *config.Message, channelID string) (string, error) {
|
||||||
|
var err error
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
file := discordgo.File{
|
||||||
|
Name: fi.Name,
|
||||||
|
ContentType: "",
|
||||||
|
Reader: bytes.NewReader(*fi.Data),
|
||||||
|
}
|
||||||
|
m := discordgo.MessageSend{
|
||||||
|
Content: msg.Username + fi.Comment,
|
||||||
|
Files: []*discordgo.File{&file},
|
||||||
|
}
|
||||||
|
_, err = b.c.ChannelMessageSendComplex(channelID, &m)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("file upload failed: %s", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
}
|
}
|
||||||
|
170
bridge/discord/handlers.go
Normal file
@ -0,0 +1,170 @@
|
|||||||
|
package bdiscord
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/bwmarrin/discordgo"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Bdiscord) messageDelete(s *discordgo.Session, m *discordgo.MessageDelete) { //nolint:unparam
|
||||||
|
rmsg := config.Message{Account: b.Account, ID: m.ID, Event: config.EventMsgDelete, Text: config.EventMsgDelete}
|
||||||
|
rmsg.Channel = b.getChannelName(m.ChannelID)
|
||||||
|
if b.useChannelID {
|
||||||
|
rmsg.Channel = "ID:" + m.ChannelID
|
||||||
|
}
|
||||||
|
b.Log.Debugf("<= Sending message from %s to gateway", b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) messageUpdate(s *discordgo.Session, m *discordgo.MessageUpdate) { //nolint:unparam
|
||||||
|
if b.GetBool("EditDisable") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// only when message is actually edited
|
||||||
|
if m.Message.EditedTimestamp != "" {
|
||||||
|
b.Log.Debugf("Sending edit message")
|
||||||
|
m.Content += b.GetString("EditSuffix")
|
||||||
|
b.messageCreate(s, (*discordgo.MessageCreate)(m))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) messageCreate(s *discordgo.Session, m *discordgo.MessageCreate) { //nolint:unparam
|
||||||
|
var err error
|
||||||
|
|
||||||
|
// not relay our own messages
|
||||||
|
if m.Author.Username == b.nick {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// if using webhooks, do not relay if it's ours
|
||||||
|
if b.useWebhook() && m.Author.Bot { // && b.isWebhookID(m.Author.ID) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// add the url of the attachments to content
|
||||||
|
if len(m.Attachments) > 0 {
|
||||||
|
for _, attach := range m.Attachments {
|
||||||
|
m.Content = m.Content + "\n" + attach.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rmsg := config.Message{Account: b.Account, Avatar: "https://cdn.discordapp.com/avatars/" + m.Author.ID + "/" + m.Author.Avatar + ".jpg", UserID: m.Author.ID, ID: m.ID}
|
||||||
|
|
||||||
|
if m.Content != "" {
|
||||||
|
b.Log.Debugf("== Receiving event %#v", m.Message)
|
||||||
|
m.Message.Content = b.stripCustomoji(m.Message.Content)
|
||||||
|
m.Message.Content = b.replaceChannelMentions(m.Message.Content)
|
||||||
|
rmsg.Text, err = m.ContentWithMoreMentionsReplaced(b.c)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("ContentWithMoreMentionsReplaced failed: %s", err)
|
||||||
|
rmsg.Text = m.ContentWithMentionsReplaced()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// set channel name
|
||||||
|
rmsg.Channel = b.getChannelName(m.ChannelID)
|
||||||
|
if b.useChannelID {
|
||||||
|
rmsg.Channel = "ID:" + m.ChannelID
|
||||||
|
}
|
||||||
|
|
||||||
|
// set username
|
||||||
|
if !b.GetBool("UseUserName") {
|
||||||
|
rmsg.Username = b.getNick(m.Author)
|
||||||
|
} else {
|
||||||
|
rmsg.Username = m.Author.Username
|
||||||
|
if b.GetBool("UseDiscriminator") {
|
||||||
|
rmsg.Username += "#" + m.Author.Discriminator
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// if we have embedded content add it to text
|
||||||
|
if b.GetBool("ShowEmbeds") && m.Message.Embeds != nil {
|
||||||
|
for _, embed := range m.Message.Embeds {
|
||||||
|
rmsg.Text = rmsg.Text + "embed: " + embed.Title + " - " + embed.Description + " - " + embed.URL + "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// no empty messages
|
||||||
|
if rmsg.Text == "" {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// do we have a /me action
|
||||||
|
var ok bool
|
||||||
|
rmsg.Text, ok = b.replaceAction(rmsg.Text)
|
||||||
|
if ok {
|
||||||
|
rmsg.Event = config.EventUserAction
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", m.Author.Username, b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) memberUpdate(s *discordgo.Session, m *discordgo.GuildMemberUpdate) {
|
||||||
|
if m.Member == nil {
|
||||||
|
b.Log.Warnf("Received member update with no member information: %#v", m)
|
||||||
|
}
|
||||||
|
|
||||||
|
b.membersMutex.Lock()
|
||||||
|
defer b.membersMutex.Unlock()
|
||||||
|
|
||||||
|
if currMember, ok := b.userMemberMap[m.Member.User.ID]; ok {
|
||||||
|
b.Log.Debugf(
|
||||||
|
"%s: memberupdate: user %s (nick %s) changes nick to %s",
|
||||||
|
b.Account,
|
||||||
|
m.Member.User.Username,
|
||||||
|
b.userMemberMap[m.Member.User.ID].Nick,
|
||||||
|
m.Member.Nick,
|
||||||
|
)
|
||||||
|
delete(b.nickMemberMap, currMember.User.Username)
|
||||||
|
delete(b.nickMemberMap, currMember.Nick)
|
||||||
|
delete(b.userMemberMap, m.Member.User.ID)
|
||||||
|
}
|
||||||
|
b.userMemberMap[m.Member.User.ID] = m.Member
|
||||||
|
b.nickMemberMap[m.Member.User.Username] = m.Member
|
||||||
|
if m.Member.Nick != "" {
|
||||||
|
b.nickMemberMap[m.Member.Nick] = m.Member
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) memberAdd(s *discordgo.Session, m *discordgo.GuildMemberAdd) {
|
||||||
|
if m.Member == nil {
|
||||||
|
b.Log.Warnf("Received member update with no member information: %#v", m)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
username := m.Member.User.Username
|
||||||
|
if m.Member.Nick != "" {
|
||||||
|
username = m.Member.Nick
|
||||||
|
}
|
||||||
|
|
||||||
|
rmsg := config.Message{
|
||||||
|
Account: b.Account,
|
||||||
|
Event: config.EventJoinLeave,
|
||||||
|
Username: "system",
|
||||||
|
Text: username + " joins",
|
||||||
|
}
|
||||||
|
b.Log.Debugf("<= Sending message from %s to gateway", b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) memberRemove(s *discordgo.Session, m *discordgo.GuildMemberRemove) {
|
||||||
|
if m.Member == nil {
|
||||||
|
b.Log.Warnf("Received member update with no member information: %#v", m)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
username := m.Member.User.Username
|
||||||
|
if m.Member.Nick != "" {
|
||||||
|
username = m.Member.Nick
|
||||||
|
}
|
||||||
|
|
||||||
|
rmsg := config.Message{
|
||||||
|
Account: b.Account,
|
||||||
|
Event: config.EventJoinLeave,
|
||||||
|
Username: "system",
|
||||||
|
Text: username + " leaves",
|
||||||
|
}
|
||||||
|
b.Log.Debugf("<= Sending message from %s to gateway", b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
189
bridge/discord/helpers.go
Normal file
@ -0,0 +1,189 @@
|
|||||||
|
package bdiscord
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
"unicode"
|
||||||
|
|
||||||
|
"github.com/bwmarrin/discordgo"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Bdiscord) getNick(user *discordgo.User) string {
|
||||||
|
b.membersMutex.RLock()
|
||||||
|
defer b.membersMutex.RUnlock()
|
||||||
|
|
||||||
|
if member, ok := b.userMemberMap[user.ID]; ok {
|
||||||
|
if member.Nick != "" {
|
||||||
|
// Only return if nick is set.
|
||||||
|
return member.Nick
|
||||||
|
}
|
||||||
|
// Otherwise return username.
|
||||||
|
return user.Username
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we didn't find nick, search for it.
|
||||||
|
member, err := b.c.GuildMember(b.guildID, user.ID)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Warnf("Failed to fetch information for member %#v: %s", user, err)
|
||||||
|
return user.Username
|
||||||
|
} else if member == nil {
|
||||||
|
b.Log.Warnf("Got no information for member %#v", user)
|
||||||
|
return user.Username
|
||||||
|
}
|
||||||
|
b.userMemberMap[user.ID] = member
|
||||||
|
b.nickMemberMap[member.User.Username] = member
|
||||||
|
if member.Nick != "" {
|
||||||
|
b.nickMemberMap[member.Nick] = member
|
||||||
|
return member.Nick
|
||||||
|
}
|
||||||
|
return user.Username
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) getGuildMemberByNick(nick string) (*discordgo.Member, error) {
|
||||||
|
b.membersMutex.RLock()
|
||||||
|
defer b.membersMutex.RUnlock()
|
||||||
|
|
||||||
|
if member, ok := b.nickMemberMap[nick]; ok {
|
||||||
|
return member, nil
|
||||||
|
}
|
||||||
|
return nil, errors.New("Couldn't find guild member with nick " + nick) // This will most likely get ignored by the caller
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) getChannelID(name string) string {
|
||||||
|
b.channelsMutex.RLock()
|
||||||
|
defer b.channelsMutex.RUnlock()
|
||||||
|
|
||||||
|
idcheck := strings.Split(name, "ID:")
|
||||||
|
if len(idcheck) > 1 {
|
||||||
|
return idcheck[1]
|
||||||
|
}
|
||||||
|
for _, channel := range b.channels {
|
||||||
|
if channel.Name == name {
|
||||||
|
return channel.ID
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) getChannelName(id string) string {
|
||||||
|
b.channelsMutex.RLock()
|
||||||
|
defer b.channelsMutex.RUnlock()
|
||||||
|
|
||||||
|
for _, channel := range b.channels {
|
||||||
|
if channel.ID == id {
|
||||||
|
return channel.Name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
// See https://discordapp.com/developers/docs/reference#message-formatting.
|
||||||
|
channelMentionRE = regexp.MustCompile("<#[0-9]+>")
|
||||||
|
emojiRE = regexp.MustCompile("<(:.*?:)[0-9]+>")
|
||||||
|
userMentionRE = regexp.MustCompile("@[^@\n]{1,32}")
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Bdiscord) replaceChannelMentions(text string) string {
|
||||||
|
replaceChannelMentionFunc := func(match string) string {
|
||||||
|
var err error
|
||||||
|
channelID := match[2 : len(match)-1]
|
||||||
|
|
||||||
|
channelName := b.getChannelName(channelID)
|
||||||
|
// If we don't have the channel refresh our list.
|
||||||
|
if channelName == "" {
|
||||||
|
b.channels, err = b.c.GuildChannels(b.guildID)
|
||||||
|
if err != nil {
|
||||||
|
return "#unknownchannel"
|
||||||
|
}
|
||||||
|
channelName = b.getChannelName(channelID)
|
||||||
|
}
|
||||||
|
return "#" + channelName
|
||||||
|
}
|
||||||
|
return channelMentionRE.ReplaceAllStringFunc(text, replaceChannelMentionFunc)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) replaceUserMentions(text string) string {
|
||||||
|
replaceUserMentionFunc := func(match string) string {
|
||||||
|
var (
|
||||||
|
err error
|
||||||
|
member *discordgo.Member
|
||||||
|
username string
|
||||||
|
)
|
||||||
|
|
||||||
|
usernames := enumerateUsernames(match[1:])
|
||||||
|
for _, username = range usernames {
|
||||||
|
b.Log.Debugf("Testing mention: '%s'", username)
|
||||||
|
member, err = b.getGuildMemberByNick(username)
|
||||||
|
if err == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if member == nil {
|
||||||
|
return match
|
||||||
|
}
|
||||||
|
return strings.Replace(match, "@"+username, member.User.Mention(), 1)
|
||||||
|
}
|
||||||
|
return userMentionRE.ReplaceAllStringFunc(text, replaceUserMentionFunc)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) stripCustomoji(text string) string {
|
||||||
|
return emojiRE.ReplaceAllString(text, `$1`)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bdiscord) replaceAction(text string) (string, bool) {
|
||||||
|
if strings.HasPrefix(text, "_") && strings.HasSuffix(text, "_") {
|
||||||
|
return text[1:], true
|
||||||
|
}
|
||||||
|
return text, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// splitURL splits a webhookURL and returns the ID and token.
|
||||||
|
func (b *Bdiscord) splitURL(url string) (string, string) {
|
||||||
|
const (
|
||||||
|
expectedWebhookSplitCount = 7
|
||||||
|
webhookIdxID = 5
|
||||||
|
webhookIdxToken = 6
|
||||||
|
)
|
||||||
|
webhookURLSplit := strings.Split(url, "/")
|
||||||
|
if len(webhookURLSplit) != expectedWebhookSplitCount {
|
||||||
|
b.Log.Fatalf("%s is no correct discord WebhookURL", url)
|
||||||
|
}
|
||||||
|
return webhookURLSplit[webhookIdxID], webhookURLSplit[webhookIdxToken]
|
||||||
|
}
|
||||||
|
|
||||||
|
func enumerateUsernames(s string) []string {
|
||||||
|
onlySpace := true
|
||||||
|
for _, r := range s {
|
||||||
|
if !unicode.IsSpace(r) {
|
||||||
|
onlySpace = false
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if onlySpace {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var username, endSpace string
|
||||||
|
var usernames []string
|
||||||
|
skippingSpace := true
|
||||||
|
for _, r := range s {
|
||||||
|
if unicode.IsSpace(r) {
|
||||||
|
if !skippingSpace {
|
||||||
|
usernames = append(usernames, username)
|
||||||
|
skippingSpace = true
|
||||||
|
}
|
||||||
|
endSpace += string(r)
|
||||||
|
username += string(r)
|
||||||
|
} else {
|
||||||
|
endSpace = ""
|
||||||
|
username += string(r)
|
||||||
|
skippingSpace = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if endSpace == "" {
|
||||||
|
usernames = append(usernames, username)
|
||||||
|
}
|
||||||
|
return usernames
|
||||||
|
}
|
46
bridge/discord/helpers_test.go
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
package bdiscord
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestEnumerateUsernames(t *testing.T) {
|
||||||
|
testcases := map[string]struct {
|
||||||
|
match string
|
||||||
|
expectedUsernames []string
|
||||||
|
}{
|
||||||
|
"only space": {
|
||||||
|
match: " \t\n \t",
|
||||||
|
expectedUsernames: nil,
|
||||||
|
},
|
||||||
|
"single word": {
|
||||||
|
match: "veni",
|
||||||
|
expectedUsernames: []string{"veni"},
|
||||||
|
},
|
||||||
|
"single word with preceeding space": {
|
||||||
|
match: " vidi",
|
||||||
|
expectedUsernames: []string{" vidi"},
|
||||||
|
},
|
||||||
|
"single word with suffixed space": {
|
||||||
|
match: "vici ",
|
||||||
|
expectedUsernames: []string{"vici"},
|
||||||
|
},
|
||||||
|
"multi-word with varying whitespace": {
|
||||||
|
match: "just me and\tmy friends \t",
|
||||||
|
expectedUsernames: []string{
|
||||||
|
"just",
|
||||||
|
"just me",
|
||||||
|
"just me and",
|
||||||
|
"just me and\tmy",
|
||||||
|
"just me and\tmy friends",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for testname, testcase := range testcases {
|
||||||
|
foundUsernames := enumerateUsernames(testcase.match)
|
||||||
|
assert.Equalf(t, testcase.expectedUsernames, foundUsernames, "Should have found the expected usernames for testcase %s", testname)
|
||||||
|
}
|
||||||
|
}
|
@ -1,56 +1,58 @@
|
|||||||
package bgitter
|
package bgitter
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/42wim/go-gitter"
|
"fmt"
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
|
||||||
log "github.com/Sirupsen/logrus"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/42wim/go-gitter"
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Bgitter struct {
|
type Bgitter struct {
|
||||||
c *gitter.Gitter
|
c *gitter.Gitter
|
||||||
Config *config.Protocol
|
User *gitter.User
|
||||||
Remote chan config.Message
|
Users []gitter.User
|
||||||
Account string
|
Rooms []gitter.Room
|
||||||
Users []gitter.User
|
*bridge.Config
|
||||||
Rooms []gitter.Room
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var flog *log.Entry
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
var protocol = "gitter"
|
return &Bgitter{Config: cfg}
|
||||||
|
|
||||||
func init() {
|
|
||||||
flog = log.WithFields(log.Fields{"module": protocol})
|
|
||||||
}
|
|
||||||
|
|
||||||
func New(cfg config.Protocol, account string, c chan config.Message) *Bgitter {
|
|
||||||
b := &Bgitter{}
|
|
||||||
b.Config = &cfg
|
|
||||||
b.Remote = c
|
|
||||||
b.Account = account
|
|
||||||
return b
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bgitter) Connect() error {
|
func (b *Bgitter) Connect() error {
|
||||||
var err error
|
var err error
|
||||||
flog.Info("Connecting")
|
b.Log.Info("Connecting")
|
||||||
b.c = gitter.New(b.Config.Token)
|
b.c = gitter.New(b.GetString("Token"))
|
||||||
_, err = b.c.GetUser()
|
b.User, err = b.c.GetUser()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
flog.Info("Connection succeeded")
|
b.Rooms, err = b.c.GetRooms()
|
||||||
b.Rooms, _ = b.c.GetRooms()
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.Log.Info("Connection succeeded")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bgitter) JoinChannel(channel string) error {
|
func (b *Bgitter) Disconnect() error {
|
||||||
room := channel
|
return nil
|
||||||
roomID := b.getRoomID(room)
|
|
||||||
if roomID == "" {
|
}
|
||||||
return nil
|
|
||||||
|
func (b *Bgitter) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
roomID, err := b.c.GetRoomId(channel.Name)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Could not find roomID for %v. Please create the room on gitter.im", channel.Name)
|
||||||
}
|
}
|
||||||
|
room, err := b.c.GetRoom(roomID)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.Rooms = append(b.Rooms, *room)
|
||||||
user, err := b.c.GetUser()
|
user, err := b.c.GetUser()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@ -68,29 +70,74 @@ func (b *Bgitter) JoinChannel(channel string) error {
|
|||||||
for event := range stream.Event {
|
for event := range stream.Event {
|
||||||
switch ev := event.Data.(type) {
|
switch ev := event.Data.(type) {
|
||||||
case *gitter.MessageReceived:
|
case *gitter.MessageReceived:
|
||||||
// check for ZWSP to see if it's not an echo
|
// ignore message sent from ourselves
|
||||||
if !strings.HasSuffix(ev.Message.Text, "") {
|
if ev.Message.From.ID != b.User.ID {
|
||||||
flog.Debugf("Sending message from %s on %s to gateway", ev.Message.From.Username, b.Account)
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", ev.Message.From.Username, b.Account)
|
||||||
b.Remote <- config.Message{Username: ev.Message.From.Username, Text: ev.Message.Text, Channel: room,
|
rmsg := config.Message{Username: ev.Message.From.Username, Text: ev.Message.Text, Channel: room,
|
||||||
Account: b.Account, Avatar: b.getAvatar(ev.Message.From.Username)}
|
Account: b.Account, Avatar: b.getAvatar(ev.Message.From.Username), UserID: ev.Message.From.ID,
|
||||||
|
ID: ev.Message.ID}
|
||||||
|
if strings.HasPrefix(ev.Message.Text, "@"+ev.Message.From.Username) {
|
||||||
|
rmsg.Event = config.EventUserAction
|
||||||
|
rmsg.Text = strings.Replace(rmsg.Text, "@"+ev.Message.From.Username+" ", "", -1)
|
||||||
|
}
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
}
|
}
|
||||||
case *gitter.GitterConnectionClosed:
|
case *gitter.GitterConnectionClosed:
|
||||||
flog.Errorf("connection with gitter closed for room %s", room)
|
b.Log.Errorf("connection with gitter closed for room %s", room)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}(stream, room)
|
}(stream, room.URI)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bgitter) Send(msg config.Message) error {
|
func (b *Bgitter) Send(msg config.Message) (string, error) {
|
||||||
flog.Debugf("Receiving %#v", msg)
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
roomID := b.getRoomID(msg.Channel)
|
roomID := b.getRoomID(msg.Channel)
|
||||||
if roomID == "" {
|
if roomID == "" {
|
||||||
flog.Errorf("Could not find roomID for %v", msg.Channel)
|
b.Log.Errorf("Could not find roomID for %v", msg.Channel)
|
||||||
return nil
|
return "", nil
|
||||||
}
|
}
|
||||||
// add ZWSP because gitter echoes our own messages
|
|
||||||
return b.c.SendMessage(roomID, msg.Username+msg.Text+" ")
|
// Delete message
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
if msg.ID == "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
// gitter has no delete message api so we edit message to ""
|
||||||
|
_, err := b.c.UpdateMessage(roomID, msg.ID, "")
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload a file (in gitter case send the upload URL because gitter has no native upload support)
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
b.c.SendMessage(roomID, rmsg.Username+rmsg.Text)
|
||||||
|
}
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
return b.handleUploadFile(&msg, roomID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Edit message
|
||||||
|
if msg.ID != "" {
|
||||||
|
b.Log.Debugf("updating message with id %s", msg.ID)
|
||||||
|
_, err := b.c.UpdateMessage(roomID, msg.ID, msg.Username+msg.Text)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Post normal message
|
||||||
|
resp, err := b.c.SendMessage(roomID, msg.Username+msg.Text)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return resp.ID, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bgitter) getRoomID(channel string) string {
|
func (b *Bgitter) getRoomID(channel string) string {
|
||||||
@ -113,3 +160,23 @@ func (b *Bgitter) getAvatar(user string) string {
|
|||||||
}
|
}
|
||||||
return avatar
|
return avatar
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Bgitter) handleUploadFile(msg *config.Message, roomID string) (string, error) {
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text += fi.Comment + ": "
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text = fi.URL
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text = fi.Comment + ": " + fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_, err := b.c.SendMessage(roomID, msg.Username+msg.Text)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
201
bridge/helper/helper.go
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
package helper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"image/png"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
"unicode/utf8"
|
||||||
|
|
||||||
|
"golang.org/x/image/webp"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"gitlab.com/golang-commonmark/markdown"
|
||||||
|
)
|
||||||
|
|
||||||
|
// DownloadFile downloads the given non-authenticated URL.
|
||||||
|
func DownloadFile(url string) (*[]byte, error) {
|
||||||
|
return DownloadFileAuth(url, "")
|
||||||
|
}
|
||||||
|
|
||||||
|
// DownloadFileAuth downloads the given URL using the specified authentication token.
|
||||||
|
func DownloadFileAuth(url string, auth string) (*[]byte, error) {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
client := &http.Client{
|
||||||
|
Timeout: time.Second * 5,
|
||||||
|
}
|
||||||
|
req, err := http.NewRequest("GET", url, nil)
|
||||||
|
if auth != "" {
|
||||||
|
req.Header.Add("Authorization", auth)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
io.Copy(&buf, resp.Body)
|
||||||
|
data := buf.Bytes()
|
||||||
|
return &data, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetSubLines splits messages in newline-delimited lines. If maxLineLength is
|
||||||
|
// specified as non-zero GetSubLines will also clip long lines to the maximum
|
||||||
|
// length and insert a warning marker that the line was clipped.
|
||||||
|
//
|
||||||
|
// TODO: The current implementation has the inconvenient that it disregards
|
||||||
|
// word boundaries when splitting but this is hard to solve without potentially
|
||||||
|
// breaking formatting and other stylistic effects.
|
||||||
|
func GetSubLines(message string, maxLineLength int) []string {
|
||||||
|
const clippingMessage = " <clipped message>"
|
||||||
|
|
||||||
|
var lines []string
|
||||||
|
for _, line := range strings.Split(strings.TrimSpace(message), "\n") {
|
||||||
|
if maxLineLength == 0 || len([]byte(line)) <= maxLineLength {
|
||||||
|
lines = append(lines, line)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// !!! WARNING !!!
|
||||||
|
// Before touching the splitting logic below please ensure that you PROPERLY
|
||||||
|
// understand how strings, runes and range loops over strings work in Go.
|
||||||
|
// A good place to start is to read https://blog.golang.org/strings. :-)
|
||||||
|
var splitStart int
|
||||||
|
var startOfPreviousRune int
|
||||||
|
for i := range line {
|
||||||
|
if i-splitStart > maxLineLength-len([]byte(clippingMessage)) {
|
||||||
|
lines = append(lines, line[splitStart:startOfPreviousRune]+clippingMessage)
|
||||||
|
splitStart = startOfPreviousRune
|
||||||
|
}
|
||||||
|
startOfPreviousRune = i
|
||||||
|
}
|
||||||
|
// This last append is safe to do without looking at the remaining byte-length
|
||||||
|
// as we assume that the byte-length of the last rune will never exceed that of
|
||||||
|
// the byte-length of the clipping message.
|
||||||
|
lines = append(lines, line[splitStart:])
|
||||||
|
}
|
||||||
|
return lines
|
||||||
|
}
|
||||||
|
|
||||||
|
// HandleExtra manages the supplementary details stored inside a message's 'Extra' field map.
|
||||||
|
func HandleExtra(msg *config.Message, general *config.Protocol) []config.Message {
|
||||||
|
extra := msg.Extra
|
||||||
|
rmsg := []config.Message{}
|
||||||
|
for _, f := range extra[config.EventFileFailureSize] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
text := fmt.Sprintf("file %s too big to download (%#v > allowed size: %#v)", fi.Name, fi.Size, general.MediaDownloadSize)
|
||||||
|
rmsg = append(rmsg, config.Message{
|
||||||
|
Text: text,
|
||||||
|
Username: "<system> ",
|
||||||
|
Channel: msg.Channel,
|
||||||
|
Account: msg.Account,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return rmsg
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAvatar constructs a URL for a given user-avatar if it is available in the cache.
|
||||||
|
func GetAvatar(av map[string]string, userid string, general *config.Protocol) string {
|
||||||
|
if sha, ok := av[userid]; ok {
|
||||||
|
return general.MediaServerDownload + "/" + sha + "/" + userid + ".png"
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// HandleDownloadSize checks a specified filename against the configured download blacklist
|
||||||
|
// and checks a specified file-size against the configure limit.
|
||||||
|
func HandleDownloadSize(logger *logrus.Entry, msg *config.Message, name string, size int64, general *config.Protocol) error {
|
||||||
|
// check blacklist here
|
||||||
|
for _, entry := range general.MediaDownloadBlackList {
|
||||||
|
if entry != "" {
|
||||||
|
re, err := regexp.Compile(entry)
|
||||||
|
if err != nil {
|
||||||
|
logger.Errorf("incorrect regexp %s for %s", entry, msg.Account)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if re.MatchString(name) {
|
||||||
|
return fmt.Errorf("Matching blacklist %s. Not downloading %s", entry, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
logger.Debugf("Trying to download %#v with size %#v", name, size)
|
||||||
|
if int(size) > general.MediaDownloadSize {
|
||||||
|
msg.Event = config.EventFileFailureSize
|
||||||
|
msg.Extra[msg.Event] = append(msg.Extra[msg.Event], config.FileInfo{
|
||||||
|
Name: name,
|
||||||
|
Comment: msg.Text,
|
||||||
|
Size: size,
|
||||||
|
})
|
||||||
|
return fmt.Errorf("File %#v to large to download (%#v). MediaDownloadSize is %#v", name, size, general.MediaDownloadSize)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// HandleDownloadData adds the data for a remote file into a Matterbridge gateway message.
|
||||||
|
func HandleDownloadData(logger *logrus.Entry, msg *config.Message, name, comment, url string, data *[]byte, general *config.Protocol) {
|
||||||
|
var avatar bool
|
||||||
|
logger.Debugf("Download OK %#v %#v", name, len(*data))
|
||||||
|
if msg.Event == config.EventAvatarDownload {
|
||||||
|
avatar = true
|
||||||
|
}
|
||||||
|
msg.Extra["file"] = append(msg.Extra["file"], config.FileInfo{
|
||||||
|
Name: name,
|
||||||
|
Data: data,
|
||||||
|
URL: url,
|
||||||
|
Comment: comment,
|
||||||
|
Avatar: avatar,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
var emptyLineMatcher = regexp.MustCompile("\n+")
|
||||||
|
|
||||||
|
// RemoveEmptyNewLines collapses consecutive newline characters into a single one and
|
||||||
|
// trims any preceding or trailing newline characters as well.
|
||||||
|
func RemoveEmptyNewLines(msg string) string {
|
||||||
|
return emptyLineMatcher.ReplaceAllString(strings.Trim(msg, "\n"), "\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ClipMessage trims a message to the specified length if it exceeds it and adds a warning
|
||||||
|
// to the message in case it does so.
|
||||||
|
func ClipMessage(text string, length int) string {
|
||||||
|
const clippingMessage = " <clipped message>"
|
||||||
|
if len(text) > length {
|
||||||
|
text = text[:length-len(clippingMessage)]
|
||||||
|
if r, size := utf8.DecodeLastRuneInString(text); r == utf8.RuneError {
|
||||||
|
text = text[:len(text)-size]
|
||||||
|
}
|
||||||
|
text += clippingMessage
|
||||||
|
}
|
||||||
|
return text
|
||||||
|
}
|
||||||
|
|
||||||
|
func ParseMarkdown(input string) string {
|
||||||
|
md := markdown.New(markdown.XHTMLOutput(true), markdown.Breaks(true))
|
||||||
|
res := md.RenderToString([]byte(input))
|
||||||
|
res = strings.TrimPrefix(res, "<p>")
|
||||||
|
res = strings.TrimSuffix(res, "</p>\n")
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
// ConvertWebPToPNG convert input data (which should be WebP format to PNG format)
|
||||||
|
func ConvertWebPToPNG(data *[]byte) error {
|
||||||
|
r := bytes.NewReader(*data)
|
||||||
|
m, err := webp.Decode(r)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
var output []byte
|
||||||
|
w := bytes.NewBuffer(output)
|
||||||
|
if err := png.Encode(w, m); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*data = w.Bytes()
|
||||||
|
return nil
|
||||||
|
}
|
126
bridge/helper/helper_test.go
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
package helper
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
const testLineLength = 64
|
||||||
|
|
||||||
|
var (
|
||||||
|
lineSplittingTestCases = map[string]struct {
|
||||||
|
input string
|
||||||
|
splitOutput []string
|
||||||
|
nonSplitOutput []string
|
||||||
|
}{
|
||||||
|
"Short single-line message": {
|
||||||
|
input: "short",
|
||||||
|
splitOutput: []string{"short"},
|
||||||
|
nonSplitOutput: []string{"short"},
|
||||||
|
},
|
||||||
|
"Long single-line message": {
|
||||||
|
input: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.",
|
||||||
|
splitOutput: []string{
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipis <clipped message>",
|
||||||
|
"cing elit, sed do eiusmod tempor incididunt ut <clipped message>",
|
||||||
|
" labore et dolore magna aliqua.",
|
||||||
|
},
|
||||||
|
nonSplitOutput: []string{"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua."},
|
||||||
|
},
|
||||||
|
"Short multi-line message": {
|
||||||
|
input: "I\ncan't\nget\nno\nsatisfaction!",
|
||||||
|
splitOutput: []string{
|
||||||
|
"I",
|
||||||
|
"can't",
|
||||||
|
"get",
|
||||||
|
"no",
|
||||||
|
"satisfaction!",
|
||||||
|
},
|
||||||
|
nonSplitOutput: []string{
|
||||||
|
"I",
|
||||||
|
"can't",
|
||||||
|
"get",
|
||||||
|
"no",
|
||||||
|
"satisfaction!",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"Long multi-line message": {
|
||||||
|
input: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.\n" +
|
||||||
|
"Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.\n" +
|
||||||
|
"Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.\n" +
|
||||||
|
"Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.",
|
||||||
|
splitOutput: []string{
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipis <clipped message>",
|
||||||
|
"cing elit, sed do eiusmod tempor incididunt ut <clipped message>",
|
||||||
|
" labore et dolore magna aliqua.",
|
||||||
|
"Ut enim ad minim veniam, quis nostrud exercita <clipped message>",
|
||||||
|
"tion ullamco laboris nisi ut aliquip ex ea com <clipped message>",
|
||||||
|
"modo consequat.",
|
||||||
|
"Duis aute irure dolor in reprehenderit in volu <clipped message>",
|
||||||
|
"ptate velit esse cillum dolore eu fugiat nulla <clipped message>",
|
||||||
|
" pariatur.",
|
||||||
|
"Excepteur sint occaecat cupidatat non proident <clipped message>",
|
||||||
|
", sunt in culpa qui officia deserunt mollit an <clipped message>",
|
||||||
|
"im id est laborum.",
|
||||||
|
},
|
||||||
|
nonSplitOutput: []string{
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.",
|
||||||
|
"Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.",
|
||||||
|
"Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.",
|
||||||
|
"Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"Message ending with new-line.": {
|
||||||
|
input: "Newline ending\n",
|
||||||
|
splitOutput: []string{"Newline ending"},
|
||||||
|
nonSplitOutput: []string{"Newline ending"},
|
||||||
|
},
|
||||||
|
"Long message containing UTF-8 multi-byte runes": {
|
||||||
|
input: "不布人個我此而及單石業喜資富下我河下日沒一我臺空達的常景便物沒為……子大我別名解成?生賣的全直黑,我自我結毛分洲了世當,是政福那是東;斯說",
|
||||||
|
splitOutput: []string{
|
||||||
|
"不布人個我此而及單石業喜資富下 <clipped message>",
|
||||||
|
"我河下日沒一我臺空達的常景便物 <clipped message>",
|
||||||
|
"沒為……子大我別名解成?生賣的 <clipped message>",
|
||||||
|
"全直黑,我自我結毛分洲了世當, <clipped message>",
|
||||||
|
"是政福那是東;斯說",
|
||||||
|
},
|
||||||
|
nonSplitOutput: []string{"不布人個我此而及單石業喜資富下我河下日沒一我臺空達的常景便物沒為……子大我別名解成?生賣的全直黑,我自我結毛分洲了世當,是政福那是東;斯說"},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGetSubLines(t *testing.T) {
|
||||||
|
for testname, testcase := range lineSplittingTestCases {
|
||||||
|
splitLines := GetSubLines(testcase.input, testLineLength)
|
||||||
|
assert.Equalf(t, testcase.splitOutput, splitLines, "'%s' testcase should give expected lines with splitting.", testname)
|
||||||
|
for _, splitLine := range splitLines {
|
||||||
|
byteLength := len([]byte(splitLine))
|
||||||
|
assert.True(t, byteLength <= testLineLength, "Splitted line '%s' of testcase '%s' should not exceed the maximum byte-length (%d vs. %d).", splitLine, testcase, byteLength, testLineLength)
|
||||||
|
}
|
||||||
|
|
||||||
|
nonSplitLines := GetSubLines(testcase.input, 0)
|
||||||
|
assert.Equalf(t, testcase.nonSplitOutput, nonSplitLines, "'%s' testcase should give expected lines without splitting.", testname)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestConvertWebPToPNG(t *testing.T) {
|
||||||
|
if os.Getenv("LOCAL_TEST") == "" {
|
||||||
|
t.Skip()
|
||||||
|
}
|
||||||
|
input, err := ioutil.ReadFile("test.webp")
|
||||||
|
if err != nil {
|
||||||
|
t.Fail()
|
||||||
|
}
|
||||||
|
d := &input
|
||||||
|
err = ConvertWebPToPNG(d)
|
||||||
|
if err != nil {
|
||||||
|
t.Fail()
|
||||||
|
}
|
||||||
|
err = ioutil.WriteFile("test.png", *d, 0644)
|
||||||
|
if err != nil {
|
||||||
|
t.Fail()
|
||||||
|
}
|
||||||
|
}
|
239
bridge/irc/handlers.go
Normal file
@ -0,0 +1,239 @@
|
|||||||
|
package birc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/dfordsoft/golib/ic"
|
||||||
|
"github.com/lrstanley/girc"
|
||||||
|
"github.com/paulrosania/go-charset/charset"
|
||||||
|
"github.com/saintfish/chardet"
|
||||||
|
|
||||||
|
// We need to import the 'data' package as an implicit dependency.
|
||||||
|
// See: https://godoc.org/github.com/paulrosania/go-charset/charset
|
||||||
|
_ "github.com/paulrosania/go-charset/data"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Birc) handleCharset(msg *config.Message) error {
|
||||||
|
if b.GetString("Charset") != "" {
|
||||||
|
switch b.GetString("Charset") {
|
||||||
|
case "gbk", "gb18030", "gb2312", "big5", "euc-kr", "euc-jp", "shift-jis", "iso-2022-jp":
|
||||||
|
msg.Text = ic.ConvertString("utf-8", b.GetString("Charset"), msg.Text)
|
||||||
|
default:
|
||||||
|
buf := new(bytes.Buffer)
|
||||||
|
w, err := charset.NewWriter(b.GetString("Charset"), buf)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("charset from utf-8 conversion failed: %s", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Fprint(w, msg.Text)
|
||||||
|
w.Close()
|
||||||
|
msg.Text = buf.String()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleFiles returns true if we have handled the files, otherwise return false
|
||||||
|
func (b *Birc) handleFiles(msg *config.Message) bool {
|
||||||
|
if msg.Extra == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for _, rmsg := range helper.HandleExtra(msg, b.General) {
|
||||||
|
b.Local <- rmsg
|
||||||
|
}
|
||||||
|
if len(msg.Extra["file"]) == 0 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text += fi.Comment + ": "
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text = fi.URL
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text = fi.Comment + ": " + fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.Local <- config.Message{Text: msg.Text, Username: msg.Username, Channel: msg.Channel, Event: msg.Event}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handleJoinPart(client *girc.Client, event girc.Event) {
|
||||||
|
if len(event.Params) == 0 {
|
||||||
|
b.Log.Debugf("handleJoinPart: empty Params? %#v", event)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
channel := strings.ToLower(event.Params[0])
|
||||||
|
if event.Command == "KICK" && event.Params[1] == b.Nick {
|
||||||
|
b.Log.Infof("Got kicked from %s by %s", channel, event.Source.Name)
|
||||||
|
time.Sleep(time.Duration(b.GetInt("RejoinDelay")) * time.Second)
|
||||||
|
b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: channel, Account: b.Account, Event: config.EventRejoinChannels}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if event.Command == "QUIT" {
|
||||||
|
if event.Source.Name == b.Nick && strings.Contains(event.Last(), "Ping timeout") {
|
||||||
|
b.Log.Infof("%s reconnecting ..", b.Account)
|
||||||
|
b.Remote <- config.Message{Username: "system", Text: "reconnect", Channel: channel, Account: b.Account, Event: config.EventFailure}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if event.Source.Name != b.Nick {
|
||||||
|
if b.GetBool("nosendjoinpart") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Log.Debugf("<= Sending JOIN_LEAVE event from %s to gateway", b.Account)
|
||||||
|
// QUIT isn't channel bound, happens for all channels on the bridge
|
||||||
|
if event.Command == "QUIT" {
|
||||||
|
channel = ""
|
||||||
|
}
|
||||||
|
msg := config.Message{Username: "system", Text: event.Source.Name + " " + strings.ToLower(event.Command) + "s", Channel: channel, Account: b.Account, Event: config.EventJoinLeave}
|
||||||
|
b.Log.Debugf("<= Message is %#v", msg)
|
||||||
|
b.Remote <- msg
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Log.Debugf("handle %#v", event)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handleNewConnection(client *girc.Client, event girc.Event) {
|
||||||
|
b.Log.Debug("Registering callbacks")
|
||||||
|
i := b.i
|
||||||
|
b.Nick = event.Params[0]
|
||||||
|
|
||||||
|
i.Handlers.Add("PRIVMSG", b.handlePrivMsg)
|
||||||
|
i.Handlers.Add("CTCP_ACTION", b.handlePrivMsg)
|
||||||
|
i.Handlers.Add(girc.RPL_TOPICWHOTIME, b.handleTopicWhoTime)
|
||||||
|
i.Handlers.Add(girc.NOTICE, b.handleNotice)
|
||||||
|
i.Handlers.Add("JOIN", b.handleJoinPart)
|
||||||
|
i.Handlers.Add("PART", b.handleJoinPart)
|
||||||
|
i.Handlers.Add("QUIT", b.handleJoinPart)
|
||||||
|
i.Handlers.Add("KICK", b.handleJoinPart)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handleNickServ() {
|
||||||
|
if !b.GetBool("UseSASL") && b.GetString("NickServNick") != "" && b.GetString("NickServPassword") != "" {
|
||||||
|
b.Log.Debugf("Sending identify to nickserv %s", b.GetString("NickServNick"))
|
||||||
|
b.i.Cmd.Message(b.GetString("NickServNick"), "IDENTIFY "+b.GetString("NickServPassword"))
|
||||||
|
}
|
||||||
|
if strings.EqualFold(b.GetString("NickServNick"), "Q@CServe.quakenet.org") {
|
||||||
|
b.Log.Debugf("Authenticating %s against %s", b.GetString("NickServUsername"), b.GetString("NickServNick"))
|
||||||
|
b.i.Cmd.Message(b.GetString("NickServNick"), "AUTH "+b.GetString("NickServUsername")+" "+b.GetString("NickServPassword"))
|
||||||
|
}
|
||||||
|
// give nickserv some slack
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
b.authDone = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handleNotice(client *girc.Client, event girc.Event) {
|
||||||
|
if strings.Contains(event.String(), "This nickname is registered") && event.Source.Name == b.GetString("NickServNick") {
|
||||||
|
b.handleNickServ()
|
||||||
|
} else {
|
||||||
|
b.handlePrivMsg(client, event)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handleOther(client *girc.Client, event girc.Event) {
|
||||||
|
if b.GetInt("DebugLevel") == 1 {
|
||||||
|
if event.Command != "CLIENT_STATE_UPDATED" &&
|
||||||
|
event.Command != "CLIENT_GENERAL_UPDATED" {
|
||||||
|
b.Log.Debugf("%#v", event.String())
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
switch event.Command {
|
||||||
|
case "372", "375", "376", "250", "251", "252", "253", "254", "255", "265", "266", "002", "003", "004", "005":
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Log.Debugf("%#v", event.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handleOtherAuth(client *girc.Client, event girc.Event) {
|
||||||
|
b.handleNickServ()
|
||||||
|
b.handleRunCommands()
|
||||||
|
// we are now fully connected
|
||||||
|
b.connected <- nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handlePrivMsg(client *girc.Client, event girc.Event) {
|
||||||
|
if b.skipPrivMsg(event) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
rmsg := config.Message{Username: event.Source.Name, Channel: strings.ToLower(event.Params[0]), Account: b.Account, UserID: event.Source.Ident + "@" + event.Source.Host}
|
||||||
|
b.Log.Debugf("== Receiving PRIVMSG: %s %s %#v", event.Source.Name, event.Last(), event)
|
||||||
|
|
||||||
|
// set action event
|
||||||
|
if event.IsAction() {
|
||||||
|
rmsg.Event = config.EventUserAction
|
||||||
|
}
|
||||||
|
|
||||||
|
// strip action, we made an event if it was an action
|
||||||
|
rmsg.Text += event.StripAction()
|
||||||
|
|
||||||
|
// strip IRC colors
|
||||||
|
re := regexp.MustCompile(`\x03(?:\d{1,2}(?:,\d{1,2})?)?|[[:cntrl:]]`)
|
||||||
|
rmsg.Text = re.ReplaceAllString(rmsg.Text, "")
|
||||||
|
|
||||||
|
// start detecting the charset
|
||||||
|
mycharset := b.GetString("Charset")
|
||||||
|
if mycharset == "" {
|
||||||
|
// detect what were sending so that we convert it to utf-8
|
||||||
|
detector := chardet.NewTextDetector()
|
||||||
|
result, err := detector.DetectBest([]byte(rmsg.Text))
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Infof("detection failed for rmsg.Text: %#v", rmsg.Text)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Log.Debugf("detected %s confidence %#v", result.Charset, result.Confidence)
|
||||||
|
mycharset = result.Charset
|
||||||
|
// if we're not sure, just pick ISO-8859-1
|
||||||
|
if result.Confidence < 80 {
|
||||||
|
mycharset = "ISO-8859-1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
switch mycharset {
|
||||||
|
case "gbk", "gb18030", "gb2312", "big5", "euc-kr", "euc-jp", "shift-jis", "iso-2022-jp":
|
||||||
|
rmsg.Text = ic.ConvertString("utf-8", b.GetString("Charset"), rmsg.Text)
|
||||||
|
default:
|
||||||
|
r, err := charset.NewReader(mycharset, strings.NewReader(rmsg.Text))
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("charset to utf-8 conversion failed: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
output, _ := ioutil.ReadAll(r)
|
||||||
|
rmsg.Text = string(output)
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", event.Params[0], b.Account)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handleRunCommands() {
|
||||||
|
for _, cmd := range b.GetStringSlice("RunCommands") {
|
||||||
|
if err := b.i.Cmd.SendRaw(cmd); err != nil {
|
||||||
|
b.Log.Errorf("RunCommands %s failed: %s", cmd, err)
|
||||||
|
}
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) handleTopicWhoTime(client *girc.Client, event girc.Event) {
|
||||||
|
parts := strings.Split(event.Params[2], "!")
|
||||||
|
t, err := strconv.ParseInt(event.Params[3], 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("Invalid time stamp: %s", event.Params[3])
|
||||||
|
}
|
||||||
|
user := parts[0]
|
||||||
|
if len(parts) > 1 {
|
||||||
|
user += " [" + parts[1] + "]"
|
||||||
|
}
|
||||||
|
b.Log.Debugf("%s: Topic set by %s [%s]", event.Command, user, time.Unix(t, 0))
|
||||||
|
}
|
@ -1,59 +0,0 @@
|
|||||||
package birc
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
func tableformatter(nicks []string, nicksPerRow int, continued bool) string {
|
|
||||||
result := "|IRC users"
|
|
||||||
if continued {
|
|
||||||
result = "|(continued)"
|
|
||||||
}
|
|
||||||
for i := 0; i < 2; i++ {
|
|
||||||
for j := 1; j <= nicksPerRow && j <= len(nicks); j++ {
|
|
||||||
if i == 0 {
|
|
||||||
result += "|"
|
|
||||||
} else {
|
|
||||||
result += ":-|"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
result += "\r\n|"
|
|
||||||
}
|
|
||||||
result += nicks[0] + "|"
|
|
||||||
for i := 1; i < len(nicks); i++ {
|
|
||||||
if i%nicksPerRow == 0 {
|
|
||||||
result += "\r\n|" + nicks[i] + "|"
|
|
||||||
} else {
|
|
||||||
result += nicks[i] + "|"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
func plainformatter(nicks []string, nicksPerRow int) string {
|
|
||||||
return strings.Join(nicks, ", ") + " currently on IRC"
|
|
||||||
}
|
|
||||||
|
|
||||||
func IsMarkup(message string) bool {
|
|
||||||
switch message[0] {
|
|
||||||
case '|':
|
|
||||||
fallthrough
|
|
||||||
case '#':
|
|
||||||
fallthrough
|
|
||||||
case '_':
|
|
||||||
fallthrough
|
|
||||||
case '*':
|
|
||||||
fallthrough
|
|
||||||
case '~':
|
|
||||||
fallthrough
|
|
||||||
case '-':
|
|
||||||
fallthrough
|
|
||||||
case ':':
|
|
||||||
fallthrough
|
|
||||||
case '>':
|
|
||||||
fallthrough
|
|
||||||
case '=':
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
@ -3,255 +3,301 @@ package birc
|
|||||||
import (
|
import (
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
"hash/crc32"
|
||||||
log "github.com/Sirupsen/logrus"
|
"net"
|
||||||
ircm "github.com/sorcix/irc"
|
|
||||||
"github.com/thoj/go-ircevent"
|
|
||||||
"regexp"
|
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/lrstanley/girc"
|
||||||
|
|
||||||
|
// We need to import the 'data' package as an implicit dependency.
|
||||||
|
// See: https://godoc.org/github.com/paulrosania/go-charset/charset
|
||||||
|
_ "github.com/paulrosania/go-charset/data"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Birc struct {
|
type Birc struct {
|
||||||
i *irc.Connection
|
i *girc.Client
|
||||||
Nick string
|
Nick string
|
||||||
names map[string][]string
|
names map[string][]string
|
||||||
Config *config.Protocol
|
connected chan error
|
||||||
Remote chan config.Message
|
Local chan config.Message // local queue for flood control
|
||||||
connected chan struct{}
|
FirstConnection, authDone bool
|
||||||
Local chan config.Message // local queue for flood control
|
MessageDelay, MessageQueue, MessageLength int
|
||||||
Account string
|
|
||||||
|
*bridge.Config
|
||||||
}
|
}
|
||||||
|
|
||||||
var flog *log.Entry
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
var protocol = "irc"
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
flog = log.WithFields(log.Fields{"module": protocol})
|
|
||||||
}
|
|
||||||
|
|
||||||
func New(cfg config.Protocol, account string, c chan config.Message) *Birc {
|
|
||||||
b := &Birc{}
|
b := &Birc{}
|
||||||
b.Config = &cfg
|
b.Config = cfg
|
||||||
b.Nick = b.Config.Nick
|
b.Nick = b.GetString("Nick")
|
||||||
b.Remote = c
|
|
||||||
b.names = make(map[string][]string)
|
b.names = make(map[string][]string)
|
||||||
b.Account = account
|
b.connected = make(chan error)
|
||||||
b.connected = make(chan struct{})
|
if b.GetInt("MessageDelay") == 0 {
|
||||||
if b.Config.MessageDelay == 0 {
|
b.MessageDelay = 1300
|
||||||
b.Config.MessageDelay = 1300
|
} else {
|
||||||
|
b.MessageDelay = b.GetInt("MessageDelay")
|
||||||
}
|
}
|
||||||
if b.Config.MessageQueue == 0 {
|
if b.GetInt("MessageQueue") == 0 {
|
||||||
b.Config.MessageQueue = 30
|
b.MessageQueue = 30
|
||||||
|
} else {
|
||||||
|
b.MessageQueue = b.GetInt("MessageQueue")
|
||||||
}
|
}
|
||||||
b.Local = make(chan config.Message, b.Config.MessageQueue+10)
|
if b.GetInt("MessageLength") == 0 {
|
||||||
|
b.MessageLength = 400
|
||||||
|
} else {
|
||||||
|
b.MessageLength = b.GetInt("MessageLength")
|
||||||
|
}
|
||||||
|
b.FirstConnection = true
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) Command(msg *config.Message) string {
|
func (b *Birc) Command(msg *config.Message) string {
|
||||||
switch msg.Text {
|
if msg.Text == "!users" {
|
||||||
case "!users":
|
b.i.Handlers.Add(girc.RPL_NAMREPLY, b.storeNames)
|
||||||
b.i.AddCallback(ircm.RPL_NAMREPLY, b.storeNames)
|
b.i.Handlers.Add(girc.RPL_ENDOFNAMES, b.endNames)
|
||||||
b.i.AddCallback(ircm.RPL_ENDOFNAMES, b.endNames)
|
b.i.Cmd.SendRaw("NAMES " + msg.Channel) //nolint:errcheck
|
||||||
b.i.SendRaw("NAMES " + msg.Channel)
|
|
||||||
}
|
}
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) Connect() error {
|
func (b *Birc) Connect() error {
|
||||||
flog.Infof("Connecting %s", b.Config.Server)
|
b.Local = make(chan config.Message, b.MessageQueue+10)
|
||||||
i := irc.IRC(b.Config.Nick, b.Config.Nick)
|
b.Log.Infof("Connecting %s", b.GetString("Server"))
|
||||||
if log.GetLevel() == log.DebugLevel {
|
|
||||||
i.Debug = true
|
i, err := b.getClient()
|
||||||
}
|
|
||||||
i.UseTLS = b.Config.UseTLS
|
|
||||||
i.UseSASL = b.Config.UseSASL
|
|
||||||
i.SASLLogin = b.Config.NickServNick
|
|
||||||
i.SASLPassword = b.Config.NickServPassword
|
|
||||||
i.TLSConfig = &tls.Config{InsecureSkipVerify: b.Config.SkipTLSVerify}
|
|
||||||
if b.Config.Password != "" {
|
|
||||||
i.Password = b.Config.Password
|
|
||||||
}
|
|
||||||
i.AddCallback(ircm.RPL_WELCOME, b.handleNewConnection)
|
|
||||||
err := i.Connect(b.Config.Server)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
b.i = i
|
|
||||||
select {
|
if b.GetBool("UseSASL") {
|
||||||
case <-b.connected:
|
i.Config.SASL = &girc.SASLPlain{
|
||||||
flog.Info("Connection succeeded")
|
User: b.GetString("NickServNick"),
|
||||||
case <-time.After(time.Second * 30):
|
Pass: b.GetString("NickServPassword"),
|
||||||
return fmt.Errorf("connection timed out")
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
i.Handlers.Add(girc.RPL_WELCOME, b.handleNewConnection)
|
||||||
|
i.Handlers.Add(girc.RPL_ENDOFMOTD, b.handleOtherAuth)
|
||||||
|
i.Handlers.Add(girc.ERR_NOMOTD, b.handleOtherAuth)
|
||||||
|
i.Handlers.Add(girc.ALL_EVENTS, b.handleOther)
|
||||||
|
b.i = i
|
||||||
|
|
||||||
|
go b.doConnect()
|
||||||
|
|
||||||
|
err = <-b.connected
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("connection failed %s", err)
|
||||||
|
}
|
||||||
|
b.Log.Info("Connection succeeded")
|
||||||
|
b.FirstConnection = false
|
||||||
|
if b.GetInt("DebugLevel") == 0 {
|
||||||
|
i.Handlers.Clear(girc.ALL_EVENTS)
|
||||||
}
|
}
|
||||||
i.Debug = false
|
|
||||||
go b.doSend()
|
go b.doSend()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) JoinChannel(channel string) error {
|
func (b *Birc) Disconnect() error {
|
||||||
b.i.Join(channel)
|
b.i.Close()
|
||||||
|
close(b.Local)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) Send(msg config.Message) error {
|
func (b *Birc) JoinChannel(channel config.ChannelInfo) error {
|
||||||
flog.Debugf("Receiving %#v", msg)
|
// need to check if we have nickserv auth done before joining channels
|
||||||
if msg.Account == b.Account {
|
for {
|
||||||
return nil
|
if b.authDone {
|
||||||
}
|
break
|
||||||
if strings.HasPrefix(msg.Text, "!") {
|
|
||||||
b.Command(&msg)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
for _, text := range strings.Split(msg.Text, "\n") {
|
|
||||||
if len(b.Local) < b.Config.MessageQueue {
|
|
||||||
if len(b.Local) == b.Config.MessageQueue-1 {
|
|
||||||
text = text + " <message clipped>"
|
|
||||||
}
|
|
||||||
b.Local <- config.Message{Text: text, Username: msg.Username, Channel: msg.Channel}
|
|
||||||
} else {
|
|
||||||
flog.Debugf("flooding, dropping message (queue at %d)", len(b.Local))
|
|
||||||
}
|
}
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
}
|
||||||
|
if channel.Options.Key != "" {
|
||||||
|
b.Log.Debugf("using key %s for channel %s", channel.Options.Key, channel.Name)
|
||||||
|
b.i.Cmd.JoinKey(channel.Name, channel.Options.Key)
|
||||||
|
} else {
|
||||||
|
b.i.Cmd.Join(channel.Name)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Birc) Send(msg config.Message) (string, error) {
|
||||||
|
// ignore delete messages
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
|
||||||
|
// we can be in between reconnects #385
|
||||||
|
if !b.i.IsConnected() {
|
||||||
|
b.Log.Error("Not connected to server, dropping message")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Execute a command
|
||||||
|
if strings.HasPrefix(msg.Text, "!") {
|
||||||
|
b.Command(&msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// convert to specified charset
|
||||||
|
if err := b.handleCharset(&msg); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle files, return if we're done here
|
||||||
|
if ok := b.handleFiles(&msg); ok {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var msgLines []string
|
||||||
|
if b.GetBool("MessageSplit") {
|
||||||
|
msgLines = helper.GetSubLines(msg.Text, b.MessageLength)
|
||||||
|
} else {
|
||||||
|
msgLines = helper.GetSubLines(msg.Text, 0)
|
||||||
|
}
|
||||||
|
for i := range msgLines {
|
||||||
|
if len(b.Local) >= b.MessageQueue {
|
||||||
|
b.Log.Debugf("flooding, dropping message (queue at %d)", len(b.Local))
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Local <- config.Message{
|
||||||
|
Text: msgLines[i],
|
||||||
|
Username: msg.Username,
|
||||||
|
Channel: msg.Channel,
|
||||||
|
Event: msg.Event,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) doConnect() {
|
||||||
|
for {
|
||||||
|
if err := b.i.Connect(); err != nil {
|
||||||
|
b.Log.Errorf("disconnect: error: %s", err)
|
||||||
|
if b.FirstConnection {
|
||||||
|
b.connected <- err
|
||||||
|
return
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
b.Log.Info("disconnect: client requested quit")
|
||||||
|
}
|
||||||
|
b.Log.Info("reconnecting in 30 seconds...")
|
||||||
|
time.Sleep(30 * time.Second)
|
||||||
|
b.i.Handlers.Clear(girc.RPL_WELCOME)
|
||||||
|
b.i.Handlers.Add(girc.RPL_WELCOME, func(client *girc.Client, event girc.Event) {
|
||||||
|
b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: "", Account: b.Account, Event: config.EventRejoinChannels}
|
||||||
|
// set our correct nick on reconnect if necessary
|
||||||
|
b.Nick = event.Source.Name
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (b *Birc) doSend() {
|
func (b *Birc) doSend() {
|
||||||
rate := time.Millisecond * time.Duration(b.Config.MessageDelay)
|
rate := time.Millisecond * time.Duration(b.MessageDelay)
|
||||||
throttle := time.Tick(rate)
|
throttle := time.NewTicker(rate)
|
||||||
for msg := range b.Local {
|
for msg := range b.Local {
|
||||||
<-throttle
|
<-throttle.C
|
||||||
b.i.Privmsg(msg.Channel, msg.Username+msg.Text)
|
username := msg.Username
|
||||||
|
if b.GetBool("Colornicks") {
|
||||||
|
checksum := crc32.ChecksumIEEE([]byte(msg.Username))
|
||||||
|
colorCode := checksum%14 + 2 // quick fix - prevent white or black color codes
|
||||||
|
username = fmt.Sprintf("\x03%02d%s\x0F", colorCode, msg.Username)
|
||||||
|
}
|
||||||
|
if msg.Event == config.EventUserAction {
|
||||||
|
b.i.Cmd.Action(msg.Channel, username+msg.Text)
|
||||||
|
} else {
|
||||||
|
b.Log.Debugf("Sending to channel %s", msg.Channel)
|
||||||
|
b.i.Cmd.Message(msg.Channel, username+msg.Text)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) endNames(event *irc.Event) {
|
// validateInput validates the server/port/nick configuration. Returns a *girc.Client if successful
|
||||||
channel := event.Arguments[1]
|
func (b *Birc) getClient() (*girc.Client, error) {
|
||||||
|
server, portstr, err := net.SplitHostPort(b.GetString("Server"))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
port, err := strconv.Atoi(portstr)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// fix strict user handling of girc
|
||||||
|
user := b.GetString("Nick")
|
||||||
|
for !girc.IsValidUser(user) {
|
||||||
|
if len(user) == 1 {
|
||||||
|
user = "matterbridge"
|
||||||
|
break
|
||||||
|
}
|
||||||
|
user = user[1:]
|
||||||
|
}
|
||||||
|
|
||||||
|
i := girc.New(girc.Config{
|
||||||
|
Server: server,
|
||||||
|
ServerPass: b.GetString("Password"),
|
||||||
|
Port: port,
|
||||||
|
Nick: b.GetString("Nick"),
|
||||||
|
User: user,
|
||||||
|
Name: b.GetString("Nick"),
|
||||||
|
SSL: b.GetBool("UseTLS"),
|
||||||
|
TLSConfig: &tls.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), ServerName: server}, //nolint:gosec
|
||||||
|
PingDelay: time.Minute,
|
||||||
|
})
|
||||||
|
return i, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Birc) endNames(client *girc.Client, event girc.Event) {
|
||||||
|
channel := event.Params[1]
|
||||||
sort.Strings(b.names[channel])
|
sort.Strings(b.names[channel])
|
||||||
maxNamesPerPost := (300 / b.nicksPerRow()) * b.nicksPerRow()
|
maxNamesPerPost := (300 / b.nicksPerRow()) * b.nicksPerRow()
|
||||||
continued := false
|
|
||||||
for len(b.names[channel]) > maxNamesPerPost {
|
for len(b.names[channel]) > maxNamesPerPost {
|
||||||
b.Remote <- config.Message{Username: b.Nick, Text: b.formatnicks(b.names[channel][0:maxNamesPerPost], continued),
|
b.Remote <- config.Message{Username: b.Nick, Text: b.formatnicks(b.names[channel][0:maxNamesPerPost]),
|
||||||
Channel: channel, Account: b.Account}
|
Channel: channel, Account: b.Account}
|
||||||
b.names[channel] = b.names[channel][maxNamesPerPost:]
|
b.names[channel] = b.names[channel][maxNamesPerPost:]
|
||||||
continued = true
|
|
||||||
}
|
}
|
||||||
b.Remote <- config.Message{Username: b.Nick, Text: b.formatnicks(b.names[channel], continued),
|
b.Remote <- config.Message{Username: b.Nick, Text: b.formatnicks(b.names[channel]),
|
||||||
Channel: channel, Account: b.Account}
|
Channel: channel, Account: b.Account}
|
||||||
b.names[channel] = nil
|
b.names[channel] = nil
|
||||||
b.i.ClearCallback(ircm.RPL_NAMREPLY)
|
b.i.Handlers.Clear(girc.RPL_NAMREPLY)
|
||||||
b.i.ClearCallback(ircm.RPL_ENDOFNAMES)
|
b.i.Handlers.Clear(girc.RPL_ENDOFNAMES)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) handleNewConnection(event *irc.Event) {
|
func (b *Birc) skipPrivMsg(event girc.Event) bool {
|
||||||
flog.Debug("Registering callbacks")
|
// Our nick can be changed
|
||||||
i := b.i
|
b.Nick = b.i.GetNick()
|
||||||
b.Nick = event.Arguments[0]
|
|
||||||
i.AddCallback("PRIVMSG", b.handlePrivMsg)
|
|
||||||
i.AddCallback("CTCP_ACTION", b.handlePrivMsg)
|
|
||||||
i.AddCallback(ircm.RPL_TOPICWHOTIME, b.handleTopicWhoTime)
|
|
||||||
i.AddCallback(ircm.NOTICE, b.handleNotice)
|
|
||||||
//i.AddCallback(ircm.RPL_MYINFO, func(e *irc.Event) { flog.Infof("%s: %s", e.Code, strings.Join(e.Arguments[1:], " ")) })
|
|
||||||
i.AddCallback("PING", func(e *irc.Event) {
|
|
||||||
i.SendRaw("PONG :" + e.Message())
|
|
||||||
flog.Debugf("PING/PONG")
|
|
||||||
})
|
|
||||||
i.AddCallback("JOIN", b.handleJoinPart)
|
|
||||||
i.AddCallback("PART", b.handleJoinPart)
|
|
||||||
i.AddCallback("QUIT", b.handleJoinPart)
|
|
||||||
i.AddCallback("*", b.handleOther)
|
|
||||||
// we are now fully connected
|
|
||||||
b.connected <- struct{}{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Birc) handleJoinPart(event *irc.Event) {
|
// freenode doesn't send 001 as first reply
|
||||||
flog.Debugf("Sending JOIN_LEAVE event from %s to gateway", b.Account)
|
if event.Command == "NOTICE" {
|
||||||
channel := event.Arguments[0]
|
return true
|
||||||
if event.Code == "QUIT" {
|
|
||||||
channel = ""
|
|
||||||
}
|
}
|
||||||
b.Remote <- config.Message{Username: "system", Text: event.Nick + " " + strings.ToLower(event.Code) + "s", Channel: channel, Account: b.Account, Event: config.EVENT_JOIN_LEAVE}
|
|
||||||
flog.Debugf("handle %#v", event)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Birc) handleNotice(event *irc.Event) {
|
|
||||||
if strings.Contains(event.Message(), "This nickname is registered") && event.Nick == b.Config.NickServNick {
|
|
||||||
b.i.Privmsg(b.Config.NickServNick, "IDENTIFY "+b.Config.NickServPassword)
|
|
||||||
} else {
|
|
||||||
b.handlePrivMsg(event)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Birc) handleOther(event *irc.Event) {
|
|
||||||
switch event.Code {
|
|
||||||
case "372", "375", "376", "250", "251", "252", "253", "254", "255", "265", "266", "002", "003", "004", "005":
|
|
||||||
return
|
|
||||||
}
|
|
||||||
flog.Debugf("%#v", event.Raw)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Birc) handlePrivMsg(event *irc.Event) {
|
|
||||||
// don't forward queries to the bot
|
// don't forward queries to the bot
|
||||||
if event.Arguments[0] == b.Nick {
|
if event.Params[0] == b.Nick {
|
||||||
return
|
return true
|
||||||
}
|
}
|
||||||
// don't forward message from ourself
|
// don't forward message from ourself
|
||||||
if event.Nick == b.Nick {
|
if event.Source.Name == b.Nick {
|
||||||
return
|
return true
|
||||||
}
|
}
|
||||||
flog.Debugf("handlePrivMsg() %s %s %#v", event.Nick, event.Message(), event)
|
return false
|
||||||
msg := ""
|
|
||||||
if event.Code == "CTCP_ACTION" {
|
|
||||||
msg = event.Nick + " "
|
|
||||||
}
|
|
||||||
msg += event.Message()
|
|
||||||
// strip IRC colors
|
|
||||||
re := regexp.MustCompile(`[[:cntrl:]](\d+,|)\d+`)
|
|
||||||
msg = re.ReplaceAllString(msg, "")
|
|
||||||
flog.Debugf("Sending message from %s on %s to gateway", event.Arguments[0], b.Account)
|
|
||||||
b.Remote <- config.Message{Username: event.Nick, Text: msg, Channel: event.Arguments[0], Account: b.Account}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Birc) handleTopicWhoTime(event *irc.Event) {
|
|
||||||
parts := strings.Split(event.Arguments[2], "!")
|
|
||||||
t, err := strconv.ParseInt(event.Arguments[3], 10, 64)
|
|
||||||
if err != nil {
|
|
||||||
flog.Errorf("Invalid time stamp: %s", event.Arguments[3])
|
|
||||||
}
|
|
||||||
user := parts[0]
|
|
||||||
if len(parts) > 1 {
|
|
||||||
user += " [" + parts[1] + "]"
|
|
||||||
}
|
|
||||||
flog.Debugf("%s: Topic set by %s [%s]", event.Code, user, time.Unix(t, 0))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) nicksPerRow() int {
|
func (b *Birc) nicksPerRow() int {
|
||||||
return 4
|
return 4
|
||||||
/*
|
|
||||||
if b.Config.Mattermost.NicksPerRow < 1 {
|
|
||||||
return 4
|
|
||||||
}
|
|
||||||
return b.Config.Mattermost.NicksPerRow
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) storeNames(event *irc.Event) {
|
func (b *Birc) storeNames(client *girc.Client, event girc.Event) {
|
||||||
channel := event.Arguments[2]
|
channel := event.Params[2]
|
||||||
b.names[channel] = append(
|
b.names[channel] = append(
|
||||||
b.names[channel],
|
b.names[channel],
|
||||||
strings.Split(strings.TrimSpace(event.Message()), " ")...)
|
strings.Split(strings.TrimSpace(event.Last()), " ")...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Birc) formatnicks(nicks []string, continued bool) string {
|
func (b *Birc) formatnicks(nicks []string) string {
|
||||||
return plainformatter(nicks, b.nicksPerRow())
|
return strings.Join(nicks, ", ") + " currently on IRC"
|
||||||
/*
|
|
||||||
switch b.Config.Mattermost.NickFormatter {
|
|
||||||
case "table":
|
|
||||||
return tableformatter(nicks, b.nicksPerRow(), continued)
|
|
||||||
default:
|
|
||||||
return plainformatter(nicks, b.nicksPerRow())
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
347
bridge/matrix/matrix.go
Normal file
@ -0,0 +1,347 @@
|
|||||||
|
package bmatrix
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"html"
|
||||||
|
"mime"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
matrix "github.com/matterbridge/gomatrix"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Bmatrix struct {
|
||||||
|
mc *matrix.Client
|
||||||
|
UserID string
|
||||||
|
RoomMap map[string]string
|
||||||
|
sync.RWMutex
|
||||||
|
htmlTag *regexp.Regexp
|
||||||
|
*bridge.Config
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
b := &Bmatrix{Config: cfg}
|
||||||
|
b.htmlTag = regexp.MustCompile("</.*?>")
|
||||||
|
b.RoomMap = make(map[string]string)
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmatrix) Connect() error {
|
||||||
|
var err error
|
||||||
|
b.Log.Infof("Connecting %s", b.GetString("Server"))
|
||||||
|
b.mc, err = matrix.NewClient(b.GetString("Server"), "", "")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
resp, err := b.mc.Login(&matrix.ReqLogin{
|
||||||
|
Type: "m.login.password",
|
||||||
|
User: b.GetString("Login"),
|
||||||
|
Password: b.GetString("Password"),
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.mc.SetCredentials(resp.UserID, resp.AccessToken)
|
||||||
|
b.UserID = resp.UserID
|
||||||
|
b.Log.Info("Connection succeeded")
|
||||||
|
go b.handlematrix()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmatrix) Disconnect() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmatrix) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
resp, err := b.mc.JoinRoom(channel.Name, "", nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.Lock()
|
||||||
|
b.RoomMap[resp.RoomID] = channel.Name
|
||||||
|
b.Unlock()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmatrix) Send(msg config.Message) (string, error) {
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
|
||||||
|
channel := b.getRoomID(msg.Channel)
|
||||||
|
b.Log.Debugf("Channel %s maps to channel id %s", msg.Channel, channel)
|
||||||
|
|
||||||
|
// Make a action /me of the message
|
||||||
|
if msg.Event == config.EventUserAction {
|
||||||
|
m := matrix.TextMessage{
|
||||||
|
MsgType: "m.emote",
|
||||||
|
Body: msg.Username + msg.Text,
|
||||||
|
}
|
||||||
|
resp, err := b.mc.SendMessageEvent(channel, "m.room.message", m)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return resp.EventID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete message
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
if msg.ID == "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
resp, err := b.mc.RedactEvent(channel, msg.ID, &matrix.ReqRedact{})
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return resp.EventID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload a file if it exists
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
if _, err := b.mc.SendText(channel, rmsg.Username+rmsg.Text); err != nil {
|
||||||
|
b.Log.Errorf("sendText failed: %s", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// check if we have files to upload (from slack, telegram or mattermost)
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
return b.handleUploadFiles(&msg, channel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Edit message if we have an ID
|
||||||
|
// matrix has no editing support
|
||||||
|
|
||||||
|
// Use notices to send join/leave events
|
||||||
|
if msg.Event == config.EventJoinLeave {
|
||||||
|
resp, err := b.mc.SendNotice(channel, msg.Username+msg.Text)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return resp.EventID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
username := html.EscapeString(msg.Username)
|
||||||
|
// check if we have a </tag>. if we have, we don't escape HTML. #696
|
||||||
|
if b.htmlTag.MatchString(msg.Username) {
|
||||||
|
username = msg.Username
|
||||||
|
}
|
||||||
|
// Post normal message with HTML support (eg riot.im)
|
||||||
|
resp, err := b.mc.SendHTML(channel, msg.Username+msg.Text, username+helper.ParseMarkdown(msg.Text))
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return resp.EventID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmatrix) getRoomID(channel string) string {
|
||||||
|
b.RLock()
|
||||||
|
defer b.RUnlock()
|
||||||
|
for ID, name := range b.RoomMap {
|
||||||
|
if name == channel {
|
||||||
|
return ID
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmatrix) handlematrix() {
|
||||||
|
syncer := b.mc.Syncer.(*matrix.DefaultSyncer)
|
||||||
|
syncer.OnEventType("m.room.redaction", b.handleEvent)
|
||||||
|
syncer.OnEventType("m.room.message", b.handleEvent)
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
if err := b.mc.Sync(); err != nil {
|
||||||
|
b.Log.Println("Sync() returned ", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmatrix) handleEvent(ev *matrix.Event) {
|
||||||
|
b.Log.Debugf("== Receiving event: %#v", ev)
|
||||||
|
if ev.Sender != b.UserID {
|
||||||
|
b.RLock()
|
||||||
|
channel, ok := b.RoomMap[ev.RoomID]
|
||||||
|
b.RUnlock()
|
||||||
|
if !ok {
|
||||||
|
b.Log.Debugf("Unknown room %s", ev.RoomID)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO download avatar
|
||||||
|
|
||||||
|
// Create our message
|
||||||
|
rmsg := config.Message{Username: ev.Sender[1:], Channel: channel, Account: b.Account, UserID: ev.Sender, ID: ev.ID}
|
||||||
|
|
||||||
|
// Text must be a string
|
||||||
|
if rmsg.Text, ok = ev.Content["body"].(string); !ok {
|
||||||
|
b.Log.Errorf("Content[body] is not a string: %T\n%#v",
|
||||||
|
ev.Content["body"], ev.Content)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove homeserver suffix if configured
|
||||||
|
if b.GetBool("NoHomeServerSuffix") {
|
||||||
|
re := regexp.MustCompile("(.*?):.*")
|
||||||
|
rmsg.Username = re.ReplaceAllString(rmsg.Username, `$1`)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete event
|
||||||
|
if ev.Type == "m.room.redaction" {
|
||||||
|
rmsg.Event = config.EventMsgDelete
|
||||||
|
rmsg.ID = ev.Redacts
|
||||||
|
rmsg.Text = config.EventMsgDelete
|
||||||
|
b.Remote <- rmsg
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do we have a /me action
|
||||||
|
if ev.Content["msgtype"].(string) == "m.emote" {
|
||||||
|
rmsg.Event = config.EventUserAction
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do we have attachments
|
||||||
|
if b.containsAttachment(ev.Content) {
|
||||||
|
err := b.handleDownloadFile(&rmsg, ev.Content)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("download failed: %#v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", ev.Sender, b.Account)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleDownloadFile handles file download
|
||||||
|
func (b *Bmatrix) handleDownloadFile(rmsg *config.Message, content map[string]interface{}) error {
|
||||||
|
var (
|
||||||
|
ok bool
|
||||||
|
url, name, msgtype, mtype string
|
||||||
|
info map[string]interface{}
|
||||||
|
size float64
|
||||||
|
)
|
||||||
|
|
||||||
|
rmsg.Extra = make(map[string][]interface{})
|
||||||
|
if url, ok = content["url"].(string); !ok {
|
||||||
|
return fmt.Errorf("url isn't a %T", url)
|
||||||
|
}
|
||||||
|
url = strings.Replace(url, "mxc://", b.GetString("Server")+"/_matrix/media/v1/download/", -1)
|
||||||
|
|
||||||
|
if info, ok = content["info"].(map[string]interface{}); !ok {
|
||||||
|
return fmt.Errorf("info isn't a %T", info)
|
||||||
|
}
|
||||||
|
if size, ok = info["size"].(float64); !ok {
|
||||||
|
return fmt.Errorf("size isn't a %T", size)
|
||||||
|
}
|
||||||
|
if name, ok = content["body"].(string); !ok {
|
||||||
|
return fmt.Errorf("name isn't a %T", name)
|
||||||
|
}
|
||||||
|
if msgtype, ok = content["msgtype"].(string); !ok {
|
||||||
|
return fmt.Errorf("msgtype isn't a %T", msgtype)
|
||||||
|
}
|
||||||
|
if mtype, ok = info["mimetype"].(string); !ok {
|
||||||
|
return fmt.Errorf("mtype isn't a %T", mtype)
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if we have an image uploaded without extension
|
||||||
|
if !strings.Contains(name, ".") {
|
||||||
|
if msgtype == "m.image" {
|
||||||
|
mext, _ := mime.ExtensionsByType(mtype)
|
||||||
|
if len(mext) > 0 {
|
||||||
|
name += mext[0]
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// just a default .png extension if we don't have mime info
|
||||||
|
name += ".png"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if the size is ok
|
||||||
|
err := helper.HandleDownloadSize(b.Log, rmsg, name, int64(size), b.General)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// actually download the file
|
||||||
|
data, err := helper.DownloadFile(url)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("download %s failed %#v", url, err)
|
||||||
|
}
|
||||||
|
// add the downloaded data to the message
|
||||||
|
helper.HandleDownloadData(b.Log, rmsg, name, "", url, data, b.General)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleUploadFiles handles native upload of files.
|
||||||
|
func (b *Bmatrix) handleUploadFiles(msg *config.Message, channel string) (string, error) {
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
if fi, ok := f.(config.FileInfo); ok {
|
||||||
|
b.handleUploadFile(msg, channel, &fi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleUploadFile handles native upload of a file.
|
||||||
|
func (b *Bmatrix) handleUploadFile(msg *config.Message, channel string, fi *config.FileInfo) {
|
||||||
|
content := bytes.NewReader(*fi.Data)
|
||||||
|
sp := strings.Split(fi.Name, ".")
|
||||||
|
mtype := mime.TypeByExtension("." + sp[len(sp)-1])
|
||||||
|
if !strings.Contains(mtype, "image") && !strings.Contains(mtype, "video") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if fi.Comment != "" {
|
||||||
|
_, err := b.mc.SendText(channel, msg.Username+fi.Comment)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("file comment failed: %#v", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// image and video uploads send no username, we have to do this ourself here #715
|
||||||
|
_, err := b.mc.SendText(channel, msg.Username)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("file comment failed: %#v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.Log.Debugf("uploading file: %s %s", fi.Name, mtype)
|
||||||
|
res, err := b.mc.UploadToContentRepo(content, mtype, int64(len(*fi.Data)))
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("file upload failed: %#v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case strings.Contains(mtype, "video"):
|
||||||
|
b.Log.Debugf("sendVideo %s", res.ContentURI)
|
||||||
|
_, err = b.mc.SendVideo(channel, fi.Name, res.ContentURI)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("sendVideo failed: %#v", err)
|
||||||
|
}
|
||||||
|
case strings.Contains(mtype, "image"):
|
||||||
|
b.Log.Debugf("sendImage %s", res.ContentURI)
|
||||||
|
_, err = b.mc.SendImage(channel, fi.Name, res.ContentURI)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("sendImage failed: %#v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.Log.Debugf("result: %#v", res)
|
||||||
|
}
|
||||||
|
|
||||||
|
// skipMessages returns true if this message should not be handled
|
||||||
|
func (b *Bmatrix) containsAttachment(content map[string]interface{}) bool {
|
||||||
|
// Skip empty messages
|
||||||
|
if content["msgtype"] == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only allow image,video or file msgtypes
|
||||||
|
if !(content["msgtype"].(string) == "m.image" ||
|
||||||
|
content["msgtype"].(string) == "m.video" ||
|
||||||
|
content["msgtype"].(string) == "m.file") {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
195
bridge/mattermost/handlers.go
Normal file
@ -0,0 +1,195 @@
|
|||||||
|
package bmattermost
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/42wim/matterbridge/matterclient"
|
||||||
|
"github.com/mattermost/mattermost-server/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
// handleDownloadAvatar downloads the avatar of userid from channel
|
||||||
|
// sends a EVENT_AVATAR_DOWNLOAD message to the gateway if successful.
|
||||||
|
// logs an error message if it fails
|
||||||
|
func (b *Bmattermost) handleDownloadAvatar(userid string, channel string) {
|
||||||
|
rmsg := config.Message{
|
||||||
|
Username: "system",
|
||||||
|
Text: "avatar",
|
||||||
|
Channel: channel,
|
||||||
|
Account: b.Account,
|
||||||
|
UserID: userid,
|
||||||
|
Event: config.EventAvatarDownload,
|
||||||
|
Extra: make(map[string][]interface{}),
|
||||||
|
}
|
||||||
|
if _, ok := b.avatarMap[userid]; !ok {
|
||||||
|
data, resp := b.mc.Client.GetProfileImage(userid, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
b.Log.Errorf("ProfileImage download failed for %#v %s", userid, resp.Error)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
err := helper.HandleDownloadSize(b.Log, &rmsg, userid+".png", int64(len(data)), b.General)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Error(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
helper.HandleDownloadData(b.Log, &rmsg, userid+".png", rmsg.Text, "", &data, b.General)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleDownloadFile handles file download
|
||||||
|
func (b *Bmattermost) handleDownloadFile(rmsg *config.Message, id string) error {
|
||||||
|
url, _ := b.mc.Client.GetFileLink(id)
|
||||||
|
finfo, resp := b.mc.Client.GetFileInfo(id)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
err := helper.HandleDownloadSize(b.Log, rmsg, finfo.Name, finfo.Size, b.General)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
data, resp := b.mc.Client.DownloadFile(id, true)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
helper.HandleDownloadData(b.Log, rmsg, finfo.Name, rmsg.Text, url, &data, b.General)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmattermost) handleMatter() {
|
||||||
|
messages := make(chan *config.Message)
|
||||||
|
if b.GetString("WebhookBindAddress") != "" {
|
||||||
|
b.Log.Debugf("Choosing webhooks based receiving")
|
||||||
|
go b.handleMatterHook(messages)
|
||||||
|
} else {
|
||||||
|
if b.GetString("Token") != "" {
|
||||||
|
b.Log.Debugf("Choosing token based receiving")
|
||||||
|
} else {
|
||||||
|
b.Log.Debugf("Choosing login/password based receiving")
|
||||||
|
}
|
||||||
|
go b.handleMatterClient(messages)
|
||||||
|
}
|
||||||
|
var ok bool
|
||||||
|
for message := range messages {
|
||||||
|
message.Avatar = helper.GetAvatar(b.avatarMap, message.UserID, b.General)
|
||||||
|
message.Account = b.Account
|
||||||
|
message.Text, ok = b.replaceAction(message.Text)
|
||||||
|
if ok {
|
||||||
|
message.Event = config.EventUserAction
|
||||||
|
}
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", message.Username, b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", message)
|
||||||
|
b.Remote <- *message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmattermost) handleMatterClient(messages chan *config.Message) {
|
||||||
|
for message := range b.mc.MessageChan {
|
||||||
|
b.Log.Debugf("%#v", message.Raw.Data)
|
||||||
|
|
||||||
|
if b.skipMessage(message) {
|
||||||
|
b.Log.Debugf("Skipped message: %#v", message)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// only download avatars if we have a place to upload them (configured mediaserver)
|
||||||
|
if b.General.MediaServerUpload != "" || b.General.MediaDownloadPath != "" {
|
||||||
|
b.handleDownloadAvatar(message.UserID, message.Channel)
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("== Receiving event %#v", message)
|
||||||
|
|
||||||
|
rmsg := &config.Message{
|
||||||
|
Username: message.Username,
|
||||||
|
UserID: message.UserID,
|
||||||
|
Channel: message.Channel,
|
||||||
|
Text: message.Text,
|
||||||
|
ID: message.Post.Id,
|
||||||
|
ParentID: message.Post.ParentId,
|
||||||
|
Extra: make(map[string][]interface{}),
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle mattermost post properties (override username and attachments)
|
||||||
|
b.handleProps(rmsg, message)
|
||||||
|
|
||||||
|
// create a text for bridges that don't support native editing
|
||||||
|
if message.Raw.Event == model.WEBSOCKET_EVENT_POST_EDITED && !b.GetBool("EditDisable") {
|
||||||
|
rmsg.Text = message.Text + b.GetString("EditSuffix")
|
||||||
|
}
|
||||||
|
|
||||||
|
if message.Raw.Event == model.WEBSOCKET_EVENT_POST_DELETED {
|
||||||
|
rmsg.Event = config.EventMsgDelete
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, id := range message.Post.FileIds {
|
||||||
|
err := b.handleDownloadFile(rmsg, id)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("download failed: %s", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use nickname instead of username if defined
|
||||||
|
if nick := b.mc.GetNickName(rmsg.UserID); nick != "" {
|
||||||
|
rmsg.Username = nick
|
||||||
|
}
|
||||||
|
|
||||||
|
messages <- rmsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmattermost) handleMatterHook(messages chan *config.Message) {
|
||||||
|
for {
|
||||||
|
message := b.mh.Receive()
|
||||||
|
b.Log.Debugf("Receiving from matterhook %#v", message)
|
||||||
|
messages <- &config.Message{
|
||||||
|
UserID: message.UserID,
|
||||||
|
Username: message.UserName,
|
||||||
|
Text: message.Text,
|
||||||
|
Channel: message.ChannelName,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleUploadFile handles native upload of files
|
||||||
|
func (b *Bmattermost) handleUploadFile(msg *config.Message) (string, error) {
|
||||||
|
var err error
|
||||||
|
var res, id string
|
||||||
|
channelID := b.mc.GetChannelId(msg.Channel, b.TeamID)
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
id, err = b.mc.UploadFile(*fi.Data, channelID, fi.Name)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
msg.Text = fi.Comment
|
||||||
|
if b.GetBool("PrefixMessagesWithNick") {
|
||||||
|
msg.Text = msg.Username + msg.Text
|
||||||
|
}
|
||||||
|
res, err = b.mc.PostMessageWithFiles(channelID, msg.Text, msg.ParentID, []string{id})
|
||||||
|
}
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmattermost) handleProps(rmsg *config.Message, message *matterclient.Message) {
|
||||||
|
props := message.Post.Props
|
||||||
|
if props == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if _, ok := props["override_username"].(string); ok {
|
||||||
|
rmsg.Username = props["override_username"].(string)
|
||||||
|
}
|
||||||
|
if _, ok := props["attachments"].([]interface{}); ok {
|
||||||
|
rmsg.Extra["attachments"] = props["attachments"].([]interface{})
|
||||||
|
if rmsg.Text == "" {
|
||||||
|
for _, attachment := range rmsg.Extra["attachments"] {
|
||||||
|
attach := attachment.(map[string]interface{})
|
||||||
|
if attach["text"].(string) != "" {
|
||||||
|
rmsg.Text += attach["text"].(string)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if attach["fallback"].(string) != "" {
|
||||||
|
rmsg.Text += attach["fallback"].(string)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
218
bridge/mattermost/helpers.go
Normal file
@ -0,0 +1,218 @@
|
|||||||
|
package bmattermost
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/42wim/matterbridge/matterclient"
|
||||||
|
"github.com/42wim/matterbridge/matterhook"
|
||||||
|
"github.com/mattermost/mattermost-server/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Bmattermost) doConnectWebhookBind() error {
|
||||||
|
switch {
|
||||||
|
case b.GetString("WebhookURL") != "":
|
||||||
|
b.Log.Info("Connecting using webhookurl (sending) and webhookbindaddress (receiving)")
|
||||||
|
b.mh = matterhook.New(b.GetString("WebhookURL"),
|
||||||
|
matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"),
|
||||||
|
BindAddress: b.GetString("WebhookBindAddress")})
|
||||||
|
case b.GetString("Token") != "":
|
||||||
|
b.Log.Info("Connecting using token (sending)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
case b.GetString("Login") != "":
|
||||||
|
b.Log.Info("Connecting using login/password (sending)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
b.Log.Info("Connecting using webhookbindaddress (receiving)")
|
||||||
|
b.mh = matterhook.New(b.GetString("WebhookURL"),
|
||||||
|
matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"),
|
||||||
|
BindAddress: b.GetString("WebhookBindAddress")})
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmattermost) doConnectWebhookURL() error {
|
||||||
|
b.Log.Info("Connecting using webhookurl (sending)")
|
||||||
|
b.mh = matterhook.New(b.GetString("WebhookURL"),
|
||||||
|
matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"),
|
||||||
|
DisableServer: true})
|
||||||
|
if b.GetString("Token") != "" {
|
||||||
|
b.Log.Info("Connecting using token (receiving)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else if b.GetString("Login") != "" {
|
||||||
|
b.Log.Info("Connecting using login/password (receiving)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmattermost) apiLogin() error {
|
||||||
|
password := b.GetString("Password")
|
||||||
|
if b.GetString("Token") != "" {
|
||||||
|
password = "token=" + b.GetString("Token")
|
||||||
|
}
|
||||||
|
|
||||||
|
b.mc = matterclient.New(b.GetString("Login"), password, b.GetString("Team"), b.GetString("Server"))
|
||||||
|
if b.GetBool("debug") {
|
||||||
|
b.mc.SetLogLevel("debug")
|
||||||
|
}
|
||||||
|
b.mc.SkipTLSVerify = b.GetBool("SkipTLSVerify")
|
||||||
|
b.mc.NoTLS = b.GetBool("NoTLS")
|
||||||
|
b.Log.Infof("Connecting %s (team: %s) on %s", b.GetString("Login"), b.GetString("Team"), b.GetString("Server"))
|
||||||
|
err := b.mc.Login()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.Log.Info("Connection succeeded")
|
||||||
|
b.TeamID = b.mc.GetTeamId()
|
||||||
|
go b.mc.WsReceiver()
|
||||||
|
go b.mc.StatusLoop()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// replaceAction replace the message with the correct action (/me) code
|
||||||
|
func (b *Bmattermost) replaceAction(text string) (string, bool) {
|
||||||
|
if strings.HasPrefix(text, "*") && strings.HasSuffix(text, "*") {
|
||||||
|
return strings.Replace(text, "*", "", -1), true
|
||||||
|
}
|
||||||
|
return text, false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bmattermost) cacheAvatar(msg *config.Message) (string, error) {
|
||||||
|
fi := msg.Extra["file"][0].(config.FileInfo)
|
||||||
|
/* if we have a sha we have successfully uploaded the file to the media server,
|
||||||
|
so we can now cache the sha */
|
||||||
|
if fi.SHA != "" {
|
||||||
|
b.Log.Debugf("Added %s to %s in avatarMap", fi.SHA, msg.UserID)
|
||||||
|
b.avatarMap[msg.UserID] = fi.SHA
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// sendWebhook uses the configured WebhookURL to send the message
|
||||||
|
func (b *Bmattermost) sendWebhook(msg config.Message) (string, error) {
|
||||||
|
// skip events
|
||||||
|
if msg.Event != "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if b.GetBool("PrefixMessagesWithNick") {
|
||||||
|
msg.Text = msg.Username + msg.Text
|
||||||
|
}
|
||||||
|
if msg.Extra != nil {
|
||||||
|
// this sends a message only if we received a config.EVENT_FILE_FAILURE_SIZE
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
rmsg := rmsg // scopelint
|
||||||
|
iconURL := config.GetIconURL(&rmsg, b.GetString("iconurl"))
|
||||||
|
matterMessage := matterhook.OMessage{
|
||||||
|
IconURL: iconURL,
|
||||||
|
Channel: rmsg.Channel,
|
||||||
|
UserName: rmsg.Username,
|
||||||
|
Text: rmsg.Text,
|
||||||
|
Props: make(map[string]interface{}),
|
||||||
|
}
|
||||||
|
matterMessage.Props["matterbridge_"+b.uuid] = true
|
||||||
|
if err := b.mh.Send(matterMessage); err != nil {
|
||||||
|
b.Log.Errorf("sendWebhook failed: %s ", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// webhook doesn't support file uploads, so we add the url manually
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text += fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
iconURL := config.GetIconURL(&msg, b.GetString("iconurl"))
|
||||||
|
matterMessage := matterhook.OMessage{
|
||||||
|
IconURL: iconURL,
|
||||||
|
Channel: msg.Channel,
|
||||||
|
UserName: msg.Username,
|
||||||
|
Text: msg.Text,
|
||||||
|
Props: make(map[string]interface{}),
|
||||||
|
}
|
||||||
|
if msg.Avatar != "" {
|
||||||
|
matterMessage.IconURL = msg.Avatar
|
||||||
|
}
|
||||||
|
matterMessage.Props["matterbridge_"+b.uuid] = true
|
||||||
|
err := b.mh.Send(matterMessage)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Info(err)
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// skipMessages returns true if this message should not be handled
|
||||||
|
func (b *Bmattermost) skipMessage(message *matterclient.Message) bool {
|
||||||
|
// Handle join/leave
|
||||||
|
if message.Type == "system_join_leave" ||
|
||||||
|
message.Type == "system_join_channel" ||
|
||||||
|
message.Type == "system_leave_channel" {
|
||||||
|
if b.GetBool("nosendjoinpart") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
b.Log.Debugf("Sending JOIN_LEAVE event from %s to gateway", b.Account)
|
||||||
|
b.Remote <- config.Message{
|
||||||
|
Username: "system",
|
||||||
|
Text: message.Text,
|
||||||
|
Channel: message.Channel,
|
||||||
|
Account: b.Account,
|
||||||
|
Event: config.EventJoinLeave,
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle edited messages
|
||||||
|
if (message.Raw.Event == model.WEBSOCKET_EVENT_POST_EDITED) && b.GetBool("EditDisable") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore messages sent from matterbridge
|
||||||
|
if message.Post.Props != nil {
|
||||||
|
if _, ok := message.Post.Props["matterbridge_"+b.uuid].(bool); ok {
|
||||||
|
b.Log.Debugf("sent by matterbridge, ignoring")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore messages sent from a user logged in as the bot
|
||||||
|
if b.mc.User.Username == message.Username {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// if the message has reactions don't repost it (for now, until we can correlate reaction with message)
|
||||||
|
if message.Post.HasReactions {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// ignore messages from other teams than ours
|
||||||
|
if message.Raw.Data["team_id"].(string) != b.TeamID {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// only handle posted, edited or deleted events
|
||||||
|
if !(message.Raw.Event == "posted" || message.Raw.Event == model.WEBSOCKET_EVENT_POST_EDITED ||
|
||||||
|
message.Raw.Event == model.WEBSOCKET_EVENT_POST_DELETED) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
@ -1,51 +1,31 @@
|
|||||||
package bmattermost
|
package bmattermost
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
"github.com/42wim/matterbridge/matterclient"
|
"github.com/42wim/matterbridge/matterclient"
|
||||||
"github.com/42wim/matterbridge/matterhook"
|
"github.com/42wim/matterbridge/matterhook"
|
||||||
log "github.com/Sirupsen/logrus"
|
"github.com/rs/xid"
|
||||||
)
|
)
|
||||||
|
|
||||||
type MMhook struct {
|
|
||||||
mh *matterhook.Client
|
|
||||||
}
|
|
||||||
|
|
||||||
type MMapi struct {
|
|
||||||
mc *matterclient.MMClient
|
|
||||||
mmMap map[string]string
|
|
||||||
mmIgnoreNicks []string
|
|
||||||
}
|
|
||||||
|
|
||||||
type MMMessage struct {
|
|
||||||
Text string
|
|
||||||
Channel string
|
|
||||||
Username string
|
|
||||||
}
|
|
||||||
|
|
||||||
type Bmattermost struct {
|
type Bmattermost struct {
|
||||||
MMhook
|
mh *matterhook.Client
|
||||||
MMapi
|
mc *matterclient.MMClient
|
||||||
Config *config.Protocol
|
uuid string
|
||||||
Remote chan config.Message
|
TeamID string
|
||||||
name string
|
*bridge.Config
|
||||||
TeamId string
|
avatarMap map[string]string
|
||||||
Account string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var flog *log.Entry
|
const mattermostPlugin = "mattermost.plugin"
|
||||||
var protocol = "mattermost"
|
|
||||||
|
|
||||||
func init() {
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
flog = log.WithFields(log.Fields{"module": protocol})
|
b := &Bmattermost{Config: cfg, avatarMap: make(map[string]string)}
|
||||||
}
|
b.uuid = xid.New().String()
|
||||||
|
|
||||||
func New(cfg config.Protocol, account string, c chan config.Message) *Bmattermost {
|
|
||||||
b := &Bmattermost{}
|
|
||||||
b.Config = &cfg
|
|
||||||
b.Remote = c
|
|
||||||
b.Account = account
|
|
||||||
b.mmMap = make(map[string]string)
|
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -54,110 +34,115 @@ func (b *Bmattermost) Command(cmd string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bmattermost) Connect() error {
|
func (b *Bmattermost) Connect() error {
|
||||||
if !b.Config.UseAPI {
|
if b.Account == mattermostPlugin {
|
||||||
flog.Info("Connecting webhooks")
|
|
||||||
b.mh = matterhook.New(b.Config.URL,
|
|
||||||
matterhook.Config{InsecureSkipVerify: b.Config.SkipTLSVerify,
|
|
||||||
BindAddress: b.Config.BindAddress})
|
|
||||||
} else {
|
|
||||||
b.mc = matterclient.New(b.Config.Login, b.Config.Password,
|
|
||||||
b.Config.Team, b.Config.Server)
|
|
||||||
b.mc.SkipTLSVerify = b.Config.SkipTLSVerify
|
|
||||||
b.mc.NoTLS = b.Config.NoTLS
|
|
||||||
flog.Infof("Connecting %s (team: %s) on %s", b.Config.Login, b.Config.Team, b.Config.Server)
|
|
||||||
err := b.mc.Login()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
flog.Info("Connection succeeded")
|
|
||||||
b.TeamId = b.mc.GetTeamId()
|
|
||||||
go b.mc.WsReceiver()
|
|
||||||
}
|
|
||||||
go b.handleMatter()
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bmattermost) JoinChannel(channel string) error {
|
|
||||||
// we can only join channels using the API
|
|
||||||
if b.Config.UseAPI {
|
|
||||||
return b.mc.JoinChannel(b.mc.GetChannelId(channel, ""))
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bmattermost) Send(msg config.Message) error {
|
|
||||||
flog.Debugf("Receiving %#v", msg)
|
|
||||||
nick := msg.Username
|
|
||||||
message := msg.Text
|
|
||||||
channel := msg.Channel
|
|
||||||
|
|
||||||
if b.Config.PrefixMessagesWithNick {
|
|
||||||
/*if IsMarkup(message) {
|
|
||||||
message = nick + "\n\n" + message
|
|
||||||
} else {
|
|
||||||
*/
|
|
||||||
message = nick + " " + message
|
|
||||||
//}
|
|
||||||
}
|
|
||||||
if !b.Config.UseAPI {
|
|
||||||
matterMessage := matterhook.OMessage{IconURL: b.Config.IconURL}
|
|
||||||
matterMessage.Channel = channel
|
|
||||||
matterMessage.UserName = nick
|
|
||||||
matterMessage.Type = ""
|
|
||||||
matterMessage.Text = message
|
|
||||||
err := b.mh.Send(matterMessage)
|
|
||||||
if err != nil {
|
|
||||||
flog.Info(err)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
b.mc.PostMessage(b.mc.GetChannelId(channel, ""), message)
|
if b.GetString("WebhookBindAddress") != "" {
|
||||||
|
if err := b.doConnectWebhookBind(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
go b.handleMatter()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
switch {
|
||||||
|
case b.GetString("WebhookURL") != "":
|
||||||
|
if err := b.doConnectWebhookURL(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
go b.handleMatter()
|
||||||
|
return nil
|
||||||
|
case b.GetString("Token") != "":
|
||||||
|
b.Log.Info("Connecting using token (sending and receiving)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
go b.handleMatter()
|
||||||
|
case b.GetString("Login") != "":
|
||||||
|
b.Log.Info("Connecting using login/password (sending and receiving)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
go b.handleMatter()
|
||||||
|
}
|
||||||
|
if b.GetString("WebhookBindAddress") == "" && b.GetString("WebhookURL") == "" &&
|
||||||
|
b.GetString("Login") == "" && b.GetString("Token") == "" {
|
||||||
|
return errors.New("no connection method found. See that you have WebhookBindAddress, WebhookURL or Token/Login/Password/Server/Team configured")
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bmattermost) handleMatter() {
|
func (b *Bmattermost) Disconnect() error {
|
||||||
flog.Debugf("Choosing API based Mattermost connection: %t", b.Config.UseAPI)
|
return nil
|
||||||
mchan := make(chan *MMMessage)
|
|
||||||
if b.Config.UseAPI {
|
|
||||||
go b.handleMatterClient(mchan)
|
|
||||||
} else {
|
|
||||||
go b.handleMatterHook(mchan)
|
|
||||||
}
|
|
||||||
for message := range mchan {
|
|
||||||
flog.Debugf("Sending message from %s on %s to gateway", message.Username, b.Account)
|
|
||||||
b.Remote <- config.Message{Text: message.Text, Username: message.Username, Channel: message.Channel, Account: b.Account}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bmattermost) handleMatterClient(mchan chan *MMMessage) {
|
func (b *Bmattermost) JoinChannel(channel config.ChannelInfo) error {
|
||||||
for message := range b.mc.MessageChan {
|
if b.Account == mattermostPlugin {
|
||||||
// do not post our own messages back to irc
|
return nil
|
||||||
// only listen to message from our team
|
}
|
||||||
if message.Raw.Event == "posted" && b.mc.User.Username != message.Username && message.Raw.Data["team_id"].(string) == b.TeamId {
|
// we can only join channels using the API
|
||||||
flog.Debugf("Receiving from matterclient %#v", message)
|
if b.GetString("WebhookURL") == "" && b.GetString("WebhookBindAddress") == "" {
|
||||||
m := &MMMessage{}
|
id := b.mc.GetChannelId(channel.Name, b.TeamID)
|
||||||
m.Username = message.Username
|
if id == "" {
|
||||||
m.Channel = message.Channel
|
return fmt.Errorf("Could not find channel ID for channel %s", channel.Name)
|
||||||
m.Text = message.Text
|
}
|
||||||
if len(message.Post.FileIds) > 0 {
|
return b.mc.JoinChannel(id)
|
||||||
for _, link := range b.mc.GetPublicLinks(message.Post.FileIds) {
|
}
|
||||||
m.Text = m.Text + "\n" + link
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Bmattermost) Send(msg config.Message) (string, error) {
|
||||||
|
if b.Account == mattermostPlugin {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
|
||||||
|
// Make a action /me of the message
|
||||||
|
if msg.Event == config.EventUserAction {
|
||||||
|
msg.Text = "*" + msg.Text + "*"
|
||||||
|
}
|
||||||
|
|
||||||
|
// map the file SHA to our user (caches the avatar)
|
||||||
|
if msg.Event == config.EventAvatarDownload {
|
||||||
|
return b.cacheAvatar(&msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use webhook to send the message
|
||||||
|
if b.GetString("WebhookURL") != "" {
|
||||||
|
return b.sendWebhook(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete message
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
if msg.ID == "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
return msg.ID, b.mc.DeleteMessage(msg.ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload a file if it exists
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
if _, err := b.mc.PostMessage(b.mc.GetChannelId(rmsg.Channel, b.TeamID), rmsg.Username+rmsg.Text, msg.ParentID); err != nil {
|
||||||
|
b.Log.Errorf("PostMessage failed: %s", err)
|
||||||
}
|
}
|
||||||
mchan <- m
|
}
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
return b.handleUploadFile(&msg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bmattermost) handleMatterHook(mchan chan *MMMessage) {
|
// Prepend nick if configured
|
||||||
for {
|
if b.GetBool("PrefixMessagesWithNick") {
|
||||||
message := b.mh.Receive()
|
msg.Text = msg.Username + msg.Text
|
||||||
flog.Debugf("Receiving from matterhook %#v", message)
|
|
||||||
m := &MMMessage{}
|
|
||||||
m.Username = message.UserName
|
|
||||||
m.Text = message.Text
|
|
||||||
m.Channel = message.ChannelName
|
|
||||||
mchan <- m
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Edit message if we have an ID
|
||||||
|
if msg.ID != "" {
|
||||||
|
return b.mc.EditMessage(msg.ID, msg.Text)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Post normal message
|
||||||
|
return b.mc.PostMessage(b.mc.GetChannelId(msg.Channel, b.TeamID), msg.Text, msg.ParentID)
|
||||||
}
|
}
|
||||||
|
74
bridge/rocketchat/handlers.go
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
package brocketchat
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Brocketchat) handleRocket() {
|
||||||
|
messages := make(chan *config.Message)
|
||||||
|
if b.GetString("WebhookBindAddress") != "" {
|
||||||
|
b.Log.Debugf("Choosing webhooks based receiving")
|
||||||
|
go b.handleRocketHook(messages)
|
||||||
|
} else {
|
||||||
|
b.Log.Debugf("Choosing login/password based receiving")
|
||||||
|
go b.handleRocketClient(messages)
|
||||||
|
}
|
||||||
|
for message := range messages {
|
||||||
|
message.Account = b.Account
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", message.Username, b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", message)
|
||||||
|
b.Remote <- *message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) handleRocketHook(messages chan *config.Message) {
|
||||||
|
for {
|
||||||
|
message := b.rh.Receive()
|
||||||
|
b.Log.Debugf("Receiving from rockethook %#v", message)
|
||||||
|
// do not loop
|
||||||
|
if message.UserName == b.GetString("Nick") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
messages <- &config.Message{
|
||||||
|
UserID: message.UserID,
|
||||||
|
Username: message.UserName,
|
||||||
|
Text: message.Text,
|
||||||
|
Channel: message.ChannelName,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) handleRocketClient(messages chan *config.Message) {
|
||||||
|
for message := range b.messageChan {
|
||||||
|
// skip messages with same ID, apparently messages get duplicated for an unknown reason
|
||||||
|
if _, ok := b.cache.Get(message.ID); ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
b.cache.Add(message.ID, true)
|
||||||
|
b.Log.Debugf("message %#v", message)
|
||||||
|
m := message
|
||||||
|
if b.skipMessage(&m) {
|
||||||
|
b.Log.Debugf("Skipped message: %#v", message)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
rmsg := &config.Message{Text: message.Msg,
|
||||||
|
Username: message.User.UserName,
|
||||||
|
Channel: b.getChannelName(message.RoomID),
|
||||||
|
Account: b.Account,
|
||||||
|
UserID: message.User.ID,
|
||||||
|
ID: message.ID,
|
||||||
|
}
|
||||||
|
messages <- rmsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) handleUploadFile(msg *config.Message) error {
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if err := b.uploadFile(&fi, b.getChannelID(msg.Channel)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
198
bridge/rocketchat/helpers.go
Normal file
@ -0,0 +1,198 @@
|
|||||||
|
package brocketchat
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io/ioutil"
|
||||||
|
"mime"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/42wim/matterbridge/hook/rockethook"
|
||||||
|
"github.com/42wim/matterbridge/matterhook"
|
||||||
|
"github.com/matterbridge/Rocket.Chat.Go.SDK/models"
|
||||||
|
"github.com/matterbridge/Rocket.Chat.Go.SDK/realtime"
|
||||||
|
"github.com/matterbridge/Rocket.Chat.Go.SDK/rest"
|
||||||
|
"github.com/nelsonken/gomf"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Brocketchat) doConnectWebhookBind() error {
|
||||||
|
switch {
|
||||||
|
case b.GetString("WebhookURL") != "":
|
||||||
|
b.Log.Info("Connecting using webhookurl (sending) and webhookbindaddress (receiving)")
|
||||||
|
b.mh = matterhook.New(b.GetString("WebhookURL"),
|
||||||
|
matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"),
|
||||||
|
DisableServer: true})
|
||||||
|
b.rh = rockethook.New(b.GetString("WebhookURL"), rockethook.Config{BindAddress: b.GetString("WebhookBindAddress")})
|
||||||
|
case b.GetString("Login") != "":
|
||||||
|
b.Log.Info("Connecting using login/password (sending)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
b.Log.Info("Connecting using webhookbindaddress (receiving)")
|
||||||
|
b.rh = rockethook.New(b.GetString("WebhookURL"), rockethook.Config{BindAddress: b.GetString("WebhookBindAddress")})
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) doConnectWebhookURL() error {
|
||||||
|
b.Log.Info("Connecting using webhookurl (sending)")
|
||||||
|
b.mh = matterhook.New(b.GetString("WebhookURL"),
|
||||||
|
matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"),
|
||||||
|
DisableServer: true})
|
||||||
|
if b.GetString("Login") != "" {
|
||||||
|
b.Log.Info("Connecting using login/password (receiving)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) apiLogin() error {
|
||||||
|
b.Log.Debugf("handling apiLogin()")
|
||||||
|
credentials := &models.UserCredentials{Email: b.GetString("login"), Password: b.GetString("password")}
|
||||||
|
myURL, err := url.Parse(b.GetString("server"))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
client, err := realtime.NewClient(myURL, b.GetBool("debug"))
|
||||||
|
b.c = client
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
restclient := rest.NewClient(myURL, b.GetBool("debug"))
|
||||||
|
user, err := b.c.Login(credentials)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.user = user
|
||||||
|
b.r = restclient
|
||||||
|
err = b.r.Login(credentials)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.Log.Info("Connection succeeded")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) getChannelName(id string) string {
|
||||||
|
b.RLock()
|
||||||
|
defer b.RUnlock()
|
||||||
|
if name, ok := b.channelMap[id]; ok {
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) getChannelID(name string) string {
|
||||||
|
b.RLock()
|
||||||
|
defer b.RUnlock()
|
||||||
|
for k, v := range b.channelMap {
|
||||||
|
if v == name {
|
||||||
|
return k
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) skipMessage(message *models.Message) bool {
|
||||||
|
return message.User.ID == b.user.ID
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Brocketchat) uploadFile(fi *config.FileInfo, channel string) error {
|
||||||
|
fb := gomf.New()
|
||||||
|
if err := fb.WriteField("description", fi.Comment); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
sp := strings.Split(fi.Name, ".")
|
||||||
|
mtype := mime.TypeByExtension("." + sp[len(sp)-1])
|
||||||
|
if !strings.Contains(mtype, "image") && !strings.Contains(mtype, "video") {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if err := fb.WriteFile("file", fi.Name, mtype, *fi.Data); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
req, err := fb.GetHTTPRequest(context.TODO(), b.GetString("server")+"/api/v1/rooms.upload/"+channel)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
req.Header.Add("X-Auth-Token", b.user.Token)
|
||||||
|
req.Header.Add("X-User-Id", b.user.ID)
|
||||||
|
client := &http.Client{
|
||||||
|
Timeout: time.Second * 5,
|
||||||
|
}
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
body, err := ioutil.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if resp.StatusCode != 200 {
|
||||||
|
b.Log.Errorf("failed: %#v", string(body))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// sendWebhook uses the configured WebhookURL to send the message
|
||||||
|
func (b *Brocketchat) sendWebhook(msg *config.Message) error {
|
||||||
|
// skip events
|
||||||
|
if msg.Event != "" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if b.GetBool("PrefixMessagesWithNick") {
|
||||||
|
msg.Text = msg.Username + msg.Text
|
||||||
|
}
|
||||||
|
if msg.Extra != nil {
|
||||||
|
// this sends a message only if we received a config.EVENT_FILE_FAILURE_SIZE
|
||||||
|
for _, rmsg := range helper.HandleExtra(msg, b.General) {
|
||||||
|
rmsg := rmsg // scopelint
|
||||||
|
iconURL := config.GetIconURL(&rmsg, b.GetString("iconurl"))
|
||||||
|
matterMessage := matterhook.OMessage{
|
||||||
|
IconURL: iconURL,
|
||||||
|
Channel: rmsg.Channel,
|
||||||
|
UserName: rmsg.Username,
|
||||||
|
Text: rmsg.Text,
|
||||||
|
Props: make(map[string]interface{}),
|
||||||
|
}
|
||||||
|
if err := b.mh.Send(matterMessage); err != nil {
|
||||||
|
b.Log.Errorf("sendWebhook failed: %s ", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// webhook doesn't support file uploads, so we add the url manually
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text += fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
iconURL := config.GetIconURL(msg, b.GetString("iconurl"))
|
||||||
|
matterMessage := matterhook.OMessage{
|
||||||
|
IconURL: iconURL,
|
||||||
|
Channel: msg.Channel,
|
||||||
|
UserName: msg.Username,
|
||||||
|
Text: msg.Text,
|
||||||
|
}
|
||||||
|
if msg.Avatar != "" {
|
||||||
|
matterMessage.IconURL = msg.Avatar
|
||||||
|
}
|
||||||
|
err := b.mh.Send(matterMessage)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Info(err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
@ -1,37 +1,46 @@
|
|||||||
package brocketchat
|
package brocketchat
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
"github.com/42wim/matterbridge/hook/rockethook"
|
"github.com/42wim/matterbridge/hook/rockethook"
|
||||||
"github.com/42wim/matterbridge/matterhook"
|
"github.com/42wim/matterbridge/matterhook"
|
||||||
log "github.com/Sirupsen/logrus"
|
lru "github.com/hashicorp/golang-lru"
|
||||||
|
"github.com/matterbridge/Rocket.Chat.Go.SDK/models"
|
||||||
|
"github.com/matterbridge/Rocket.Chat.Go.SDK/realtime"
|
||||||
|
"github.com/matterbridge/Rocket.Chat.Go.SDK/rest"
|
||||||
)
|
)
|
||||||
|
|
||||||
type MMhook struct {
|
|
||||||
mh *matterhook.Client
|
|
||||||
rh *rockethook.Client
|
|
||||||
}
|
|
||||||
|
|
||||||
type Brocketchat struct {
|
type Brocketchat struct {
|
||||||
MMhook
|
mh *matterhook.Client
|
||||||
Config *config.Protocol
|
rh *rockethook.Client
|
||||||
Remote chan config.Message
|
c *realtime.Client
|
||||||
name string
|
r *rest.Client
|
||||||
Account string
|
cache *lru.Cache
|
||||||
|
*bridge.Config
|
||||||
|
messageChan chan models.Message
|
||||||
|
channelMap map[string]string
|
||||||
|
user *models.User
|
||||||
|
sync.RWMutex
|
||||||
}
|
}
|
||||||
|
|
||||||
var flog *log.Entry
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
var protocol = "rocketchat"
|
newCache, err := lru.New(100)
|
||||||
|
if err != nil {
|
||||||
func init() {
|
cfg.Log.Fatalf("Could not create LRU cache for rocketchat bridge: %v", err)
|
||||||
flog = log.WithFields(log.Fields{"module": protocol})
|
}
|
||||||
}
|
b := &Brocketchat{
|
||||||
|
Config: cfg,
|
||||||
func New(cfg config.Protocol, account string, c chan config.Message) *Brocketchat {
|
messageChan: make(chan models.Message),
|
||||||
b := &Brocketchat{}
|
channelMap: make(map[string]string),
|
||||||
b.Config = &cfg
|
cache: newCache,
|
||||||
b.Remote = c
|
}
|
||||||
b.Account = account
|
b.Log.Debugf("enabling rocketchat")
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -40,43 +49,122 @@ func (b *Brocketchat) Command(cmd string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (b *Brocketchat) Connect() error {
|
func (b *Brocketchat) Connect() error {
|
||||||
flog.Info("Connecting webhooks")
|
if b.GetString("WebhookBindAddress") != "" {
|
||||||
b.mh = matterhook.New(b.Config.URL,
|
if err := b.doConnectWebhookBind(); err != nil {
|
||||||
matterhook.Config{InsecureSkipVerify: b.Config.SkipTLSVerify,
|
return err
|
||||||
DisableServer: true})
|
}
|
||||||
b.rh = rockethook.New(b.Config.URL, rockethook.Config{BindAddress: b.Config.BindAddress})
|
go b.handleRocket()
|
||||||
go b.handleRocketHook()
|
return nil
|
||||||
|
}
|
||||||
|
switch {
|
||||||
|
case b.GetString("WebhookURL") != "":
|
||||||
|
if err := b.doConnectWebhookURL(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
go b.handleRocket()
|
||||||
|
return nil
|
||||||
|
case b.GetString("Login") != "":
|
||||||
|
b.Log.Info("Connecting using login/password (sending and receiving)")
|
||||||
|
err := b.apiLogin()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
go b.handleRocket()
|
||||||
|
}
|
||||||
|
if b.GetString("WebhookBindAddress") == "" && b.GetString("WebhookURL") == "" &&
|
||||||
|
b.GetString("Login") == "" {
|
||||||
|
return errors.New("no connection method found. See that you have WebhookBindAddress, WebhookURL or Login/Password/Server configured")
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Brocketchat) JoinChannel(channel string) error {
|
func (b *Brocketchat) Disconnect() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Brocketchat) Send(msg config.Message) error {
|
func (b *Brocketchat) JoinChannel(channel config.ChannelInfo) error {
|
||||||
flog.Debugf("Receiving %#v", msg)
|
if b.c == nil {
|
||||||
matterMessage := matterhook.OMessage{IconURL: b.Config.IconURL}
|
return nil
|
||||||
matterMessage.Channel = msg.Channel
|
}
|
||||||
matterMessage.UserName = msg.Username
|
id, err := b.c.GetChannelId(strings.TrimPrefix(channel.Name, "#"))
|
||||||
matterMessage.Type = ""
|
|
||||||
matterMessage.Text = msg.Text
|
|
||||||
err := b.mh.Send(matterMessage)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Info(err)
|
return err
|
||||||
|
}
|
||||||
|
b.Lock()
|
||||||
|
b.channelMap[id] = channel.Name
|
||||||
|
b.Unlock()
|
||||||
|
mychannel := &models.Channel{ID: id, Name: strings.TrimPrefix(channel.Name, "#")}
|
||||||
|
if err := b.c.JoinChannel(id); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := b.c.SubscribeToMessageStream(mychannel, b.messageChan); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Brocketchat) handleRocketHook() {
|
func (b *Brocketchat) Send(msg config.Message) (string, error) {
|
||||||
for {
|
// strip the # if people has set this
|
||||||
message := b.rh.Receive()
|
msg.Channel = strings.TrimPrefix(msg.Channel, "#")
|
||||||
flog.Debugf("Receiving from rockethook %#v", message)
|
channel := &models.Channel{ID: b.getChannelID(msg.Channel), Name: msg.Channel}
|
||||||
// do not loop
|
|
||||||
if message.UserName == b.Config.Nick {
|
// Delete message
|
||||||
continue
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
if msg.ID == "" {
|
||||||
|
return "", nil
|
||||||
}
|
}
|
||||||
flog.Debugf("Sending message from %s on %s to gateway", message.UserName, b.Account)
|
return msg.ID, b.c.DeleteMessage(&models.Message{ID: msg.ID})
|
||||||
b.Remote <- config.Message{Text: message.Text, Username: message.UserName, Channel: message.ChannelName, Account: b.Account}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Use webhook to send the message
|
||||||
|
if b.GetString("WebhookURL") != "" {
|
||||||
|
return "", b.sendWebhook(&msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prepend nick if configured
|
||||||
|
if b.GetBool("PrefixMessagesWithNick") {
|
||||||
|
msg.Text = msg.Username + msg.Text
|
||||||
|
}
|
||||||
|
|
||||||
|
// Edit message if we have an ID
|
||||||
|
if msg.ID != "" {
|
||||||
|
return msg.ID, b.c.EditMessage(&models.Message{ID: msg.ID, Msg: msg.Text, RoomID: b.getChannelID(msg.Channel)})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload a file if it exists
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
// strip the # if people has set this
|
||||||
|
rmsg.Channel = strings.TrimPrefix(rmsg.Channel, "#")
|
||||||
|
smsg := &models.Message{
|
||||||
|
RoomID: b.getChannelID(rmsg.Channel),
|
||||||
|
Msg: rmsg.Username + rmsg.Text,
|
||||||
|
PostMessage: models.PostMessage{
|
||||||
|
Avatar: rmsg.Avatar,
|
||||||
|
Alias: rmsg.Username,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
if _, err := b.c.SendMessage(smsg); err != nil {
|
||||||
|
b.Log.Errorf("SendMessage failed: %s", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
return "", b.handleUploadFile(&msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
smsg := &models.Message{
|
||||||
|
RoomID: channel.ID,
|
||||||
|
Msg: msg.Text,
|
||||||
|
PostMessage: models.PostMessage{
|
||||||
|
Avatar: msg.Avatar,
|
||||||
|
Alias: msg.Username,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
rmsg, err := b.c.SendMessage(smsg)
|
||||||
|
if rmsg == nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return rmsg.ID, err
|
||||||
}
|
}
|
||||||
|
347
bridge/slack/handlers.go
Normal file
@ -0,0 +1,347 @@
|
|||||||
|
package bslack
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"html"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/nlopes/slack"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Bslack) handleSlack() {
|
||||||
|
messages := make(chan *config.Message)
|
||||||
|
if b.GetString(incomingWebhookConfig) != "" {
|
||||||
|
b.Log.Debugf("Choosing webhooks based receiving")
|
||||||
|
go b.handleMatterHook(messages)
|
||||||
|
} else {
|
||||||
|
b.Log.Debugf("Choosing token based receiving")
|
||||||
|
go b.handleSlackClient(messages)
|
||||||
|
}
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
b.Log.Debug("Start listening for Slack messages")
|
||||||
|
for message := range messages {
|
||||||
|
if message.Event != config.EventUserTyping {
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", message.Username, b.Account)
|
||||||
|
}
|
||||||
|
|
||||||
|
// cleanup the message
|
||||||
|
message.Text = b.replaceMention(message.Text)
|
||||||
|
message.Text = b.replaceVariable(message.Text)
|
||||||
|
message.Text = b.replaceChannel(message.Text)
|
||||||
|
message.Text = b.replaceURL(message.Text)
|
||||||
|
message.Text = html.UnescapeString(message.Text)
|
||||||
|
|
||||||
|
// Add the avatar
|
||||||
|
message.Avatar = b.users.getAvatar(message.UserID)
|
||||||
|
|
||||||
|
b.Log.Debugf("<= Message is %#v", message)
|
||||||
|
b.Remote <- *message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) handleSlackClient(messages chan *config.Message) {
|
||||||
|
for msg := range b.rtm.IncomingEvents {
|
||||||
|
if msg.Type != sUserTyping && msg.Type != sLatencyReport {
|
||||||
|
b.Log.Debugf("== Receiving event %#v", msg.Data)
|
||||||
|
}
|
||||||
|
switch ev := msg.Data.(type) {
|
||||||
|
case *slack.UserTypingEvent:
|
||||||
|
if !b.GetBool("ShowUserTyping") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
rmsg, err := b.handleTypingEvent(ev)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("%#v", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
messages <- rmsg
|
||||||
|
case *slack.MessageEvent:
|
||||||
|
if b.skipMessageEvent(ev) {
|
||||||
|
b.Log.Debugf("Skipped message: %#v", ev)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
rmsg, err := b.handleMessageEvent(ev)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("%#v", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
messages <- rmsg
|
||||||
|
case *slack.OutgoingErrorEvent:
|
||||||
|
b.Log.Debugf("%#v", ev.Error())
|
||||||
|
case *slack.ChannelJoinedEvent:
|
||||||
|
// When we join a channel we update the full list of users as
|
||||||
|
// well as the information for the channel that we joined as this
|
||||||
|
// should now tell that we are a member of it.
|
||||||
|
b.channels.registerChannel(ev.Channel)
|
||||||
|
case *slack.ConnectedEvent:
|
||||||
|
b.si = ev.Info
|
||||||
|
b.channels.populateChannels(true)
|
||||||
|
b.users.populateUsers(true)
|
||||||
|
case *slack.InvalidAuthEvent:
|
||||||
|
b.Log.Fatalf("Invalid Token %#v", ev)
|
||||||
|
case *slack.ConnectionErrorEvent:
|
||||||
|
b.Log.Errorf("Connection failed %#v %#v", ev.Error(), ev.ErrorObj)
|
||||||
|
case *slack.MemberJoinedChannelEvent:
|
||||||
|
b.users.populateUser(ev.User)
|
||||||
|
case *slack.LatencyReport:
|
||||||
|
continue
|
||||||
|
default:
|
||||||
|
b.Log.Debugf("Unhandled incoming event: %T", ev)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) handleMatterHook(messages chan *config.Message) {
|
||||||
|
for {
|
||||||
|
message := b.mh.Receive()
|
||||||
|
b.Log.Debugf("receiving from matterhook (slack) %#v", message)
|
||||||
|
if message.UserName == "slackbot" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
messages <- &config.Message{
|
||||||
|
Username: message.UserName,
|
||||||
|
Text: message.Text,
|
||||||
|
Channel: message.ChannelName,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// skipMessageEvent skips event that need to be skipped :-)
|
||||||
|
func (b *Bslack) skipMessageEvent(ev *slack.MessageEvent) bool {
|
||||||
|
switch ev.SubType {
|
||||||
|
case sChannelLeave, sChannelJoin:
|
||||||
|
return b.GetBool(noSendJoinConfig)
|
||||||
|
case sPinnedItem, sUnpinnedItem:
|
||||||
|
return true
|
||||||
|
case sChannelTopic, sChannelPurpose:
|
||||||
|
// Skip the event if our bot/user account changed the topic/purpose
|
||||||
|
if ev.User == b.si.User.ID {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skip any messages that we made ourselves or from 'slackbot' (see #527).
|
||||||
|
if ev.Username == sSlackBotUser ||
|
||||||
|
(b.rtm != nil && ev.Username == b.si.User.Name) ||
|
||||||
|
(len(ev.Attachments) > 0 && ev.Attachments[0].CallbackID == "matterbridge_"+b.uuid) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// It seems ev.SubMessage.Edited == nil when slack unfurls.
|
||||||
|
// Do not forward these messages. See Github issue #266.
|
||||||
|
if ev.SubMessage != nil &&
|
||||||
|
ev.SubMessage.ThreadTimestamp != ev.SubMessage.Timestamp &&
|
||||||
|
ev.SubMessage.Edited == nil {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(ev.Files) > 0 {
|
||||||
|
return b.filesCached(ev.Files)
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) filesCached(files []slack.File) bool {
|
||||||
|
for i := range files {
|
||||||
|
if !b.fileCached(&files[i]) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleMessageEvent handles the message events. Together with any called sub-methods,
|
||||||
|
// this method implements the following event processing pipeline:
|
||||||
|
//
|
||||||
|
// 1. Check if the message should be ignored.
|
||||||
|
// NOTE: This is not actually part of the method below but is done just before it
|
||||||
|
// is called via the 'skipMessageEvent()' method.
|
||||||
|
// 2. Populate the Matterbridge message that will be sent to the router based on the
|
||||||
|
// received event and logic that is common to all events that are not skipped.
|
||||||
|
// 3. Detect and handle any message that is "status" related (think join channel, etc.).
|
||||||
|
// This might result in an early exit from the pipeline and passing of the
|
||||||
|
// pre-populated message to the Matterbridge router.
|
||||||
|
// 4. Handle the specific case of messages that edit existing messages depending on
|
||||||
|
// configuration.
|
||||||
|
// 5. Handle any attachments of the received event.
|
||||||
|
// 6. Check that the Matterbridge message that we end up with after at the end of the
|
||||||
|
// pipeline is valid before sending it to the Matterbridge router.
|
||||||
|
func (b *Bslack) handleMessageEvent(ev *slack.MessageEvent) (*config.Message, error) {
|
||||||
|
rmsg, err := b.populateReceivedMessage(ev)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle some message types early.
|
||||||
|
if b.handleStatusEvent(ev, rmsg) {
|
||||||
|
return rmsg, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
b.handleAttachments(ev, rmsg)
|
||||||
|
|
||||||
|
// Verify that we have the right information and the message
|
||||||
|
// is well-formed before sending it out to the router.
|
||||||
|
if len(ev.Files) == 0 && (rmsg.Text == "" || rmsg.Username == "") {
|
||||||
|
if ev.BotID != "" {
|
||||||
|
// This is probably a webhook we couldn't resolve.
|
||||||
|
return nil, fmt.Errorf("message handling resulted in an empty bot message (probably an incoming webhook we couldn't resolve): %#v", ev)
|
||||||
|
}
|
||||||
|
if ev.SubMessage != nil {
|
||||||
|
return nil, fmt.Errorf("message handling resulted in an empty message: %#v with submessage %#v", ev, ev.SubMessage)
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("message handling resulted in an empty message: %#v", ev)
|
||||||
|
}
|
||||||
|
return rmsg, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) handleStatusEvent(ev *slack.MessageEvent, rmsg *config.Message) bool {
|
||||||
|
switch ev.SubType {
|
||||||
|
case sChannelJoined, sMemberJoined:
|
||||||
|
// There's no further processing needed on channel events
|
||||||
|
// so we return 'true'.
|
||||||
|
return true
|
||||||
|
case sChannelJoin, sChannelLeave:
|
||||||
|
rmsg.Username = sSystemUser
|
||||||
|
rmsg.Event = config.EventJoinLeave
|
||||||
|
case sChannelTopic, sChannelPurpose:
|
||||||
|
b.channels.populateChannels(false)
|
||||||
|
rmsg.Event = config.EventTopicChange
|
||||||
|
case sMessageChanged:
|
||||||
|
rmsg.Text = ev.SubMessage.Text
|
||||||
|
// handle deleted thread starting messages
|
||||||
|
if ev.SubMessage.Text == "This message was deleted." {
|
||||||
|
rmsg.Event = config.EventMsgDelete
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
case sMessageDeleted:
|
||||||
|
rmsg.Text = config.EventMsgDelete
|
||||||
|
rmsg.Event = config.EventMsgDelete
|
||||||
|
rmsg.ID = ev.DeletedTimestamp
|
||||||
|
// If a message is being deleted we do not need to process
|
||||||
|
// the event any further so we return 'true'.
|
||||||
|
return true
|
||||||
|
case sMeMessage:
|
||||||
|
rmsg.Event = config.EventUserAction
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) handleAttachments(ev *slack.MessageEvent, rmsg *config.Message) {
|
||||||
|
// File comments are set by the system (because there is no username given).
|
||||||
|
if ev.SubType == sFileComment {
|
||||||
|
rmsg.Username = sSystemUser
|
||||||
|
}
|
||||||
|
|
||||||
|
// See if we have some text in the attachments.
|
||||||
|
if rmsg.Text == "" {
|
||||||
|
for _, attach := range ev.Attachments {
|
||||||
|
if attach.Text != "" {
|
||||||
|
if attach.Title != "" {
|
||||||
|
rmsg.Text = attach.Title + "\n"
|
||||||
|
}
|
||||||
|
rmsg.Text += attach.Text
|
||||||
|
} else {
|
||||||
|
rmsg.Text = attach.Fallback
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save the attachments, so that we can send them to other slack (compatible) bridges.
|
||||||
|
if len(ev.Attachments) > 0 {
|
||||||
|
rmsg.Extra[sSlackAttachment] = append(rmsg.Extra[sSlackAttachment], ev.Attachments)
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we have files attached, download them (in memory) and put a pointer to it in msg.Extra.
|
||||||
|
for i := range ev.Files {
|
||||||
|
if err := b.handleDownloadFile(rmsg, &ev.Files[i], false); err != nil {
|
||||||
|
b.Log.Errorf("Could not download incoming file: %#v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) handleTypingEvent(ev *slack.UserTypingEvent) (*config.Message, error) {
|
||||||
|
channelInfo, err := b.channels.getChannelByID(ev.Channel)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &config.Message{
|
||||||
|
Channel: channelInfo.Name,
|
||||||
|
Account: b.Account,
|
||||||
|
Event: config.EventUserTyping,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleDownloadFile handles file download
|
||||||
|
func (b *Bslack) handleDownloadFile(rmsg *config.Message, file *slack.File, retry bool) error {
|
||||||
|
if b.fileCached(file) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
// Check that the file is neither too large nor blacklisted.
|
||||||
|
if err := helper.HandleDownloadSize(b.Log, rmsg, file.Name, int64(file.Size), b.General); err != nil {
|
||||||
|
b.Log.WithError(err).Infof("Skipping download of incoming file.")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Actually download the file.
|
||||||
|
data, err := helper.DownloadFileAuth(file.URLPrivateDownload, "Bearer "+b.GetString(tokenConfig))
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("download %s failed %#v", file.URLPrivateDownload, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(*data) != file.Size && !retry {
|
||||||
|
b.Log.Debugf("Data size (%d) is not equal to size declared (%d)\n", len(*data), file.Size)
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
return b.handleDownloadFile(rmsg, file, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
// If a comment is attached to the file(s) it is in the 'Text' field of the Slack messge event
|
||||||
|
// and should be added as comment to only one of the files. We reset the 'Text' field to ensure
|
||||||
|
// that the comment is not duplicated.
|
||||||
|
comment := rmsg.Text
|
||||||
|
rmsg.Text = ""
|
||||||
|
helper.HandleDownloadData(b.Log, rmsg, file.Name, comment, file.URLPrivateDownload, data, b.General)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleGetChannelMembers handles messages containing the GetChannelMembers event
|
||||||
|
// Sends a message to the router containing *config.ChannelMembers
|
||||||
|
func (b *Bslack) handleGetChannelMembers(rmsg *config.Message) bool {
|
||||||
|
if rmsg.Event != config.EventGetChannelMembers {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
cMembers := b.channels.getChannelMembers(b.users)
|
||||||
|
|
||||||
|
extra := make(map[string][]interface{})
|
||||||
|
extra[config.EventGetChannelMembers] = append(extra[config.EventGetChannelMembers], cMembers)
|
||||||
|
msg := config.Message{
|
||||||
|
Extra: extra,
|
||||||
|
Event: config.EventGetChannelMembers,
|
||||||
|
Account: b.Account,
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("sending msg to remote %#v", msg)
|
||||||
|
b.Remote <- msg
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// fileCached implements Matterbridge's caching logic for files
|
||||||
|
// shared via Slack.
|
||||||
|
//
|
||||||
|
// We consider that a file was cached if its ID was added in the last minute or
|
||||||
|
// it's name was registered in the last 10 seconds. This ensures that an
|
||||||
|
// identically named file but with different content will be uploaded correctly
|
||||||
|
// (the assumption is that such name collisions will not occur within the given
|
||||||
|
// timeframes).
|
||||||
|
func (b *Bslack) fileCached(file *slack.File) bool {
|
||||||
|
if ts, ok := b.cache.Get("file" + file.ID); ok && time.Since(ts.(time.Time)) < time.Minute {
|
||||||
|
return true
|
||||||
|
} else if ts, ok = b.cache.Get("filename" + file.Name); ok && time.Since(ts.(time.Time)) < 10*time.Second {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
229
bridge/slack/helpers.go
Normal file
@ -0,0 +1,229 @@
|
|||||||
|
package bslack
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/nlopes/slack"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
// populateReceivedMessage shapes the initial Matterbridge message that we will forward to the
|
||||||
|
// router before we apply message-dependent modifications.
|
||||||
|
func (b *Bslack) populateReceivedMessage(ev *slack.MessageEvent) (*config.Message, error) {
|
||||||
|
// Use our own func because rtm.GetChannelInfo doesn't work for private channels.
|
||||||
|
channel, err := b.channels.getChannelByID(ev.Channel)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
rmsg := &config.Message{
|
||||||
|
Text: ev.Text,
|
||||||
|
Channel: channel.Name,
|
||||||
|
Account: b.Account,
|
||||||
|
ID: ev.Timestamp,
|
||||||
|
Extra: make(map[string][]interface{}),
|
||||||
|
ParentID: ev.ThreadTimestamp,
|
||||||
|
Protocol: b.Protocol,
|
||||||
|
}
|
||||||
|
if b.useChannelID {
|
||||||
|
rmsg.Channel = "ID:" + channel.ID
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle 'edit' messages.
|
||||||
|
if ev.SubMessage != nil && !b.GetBool(editDisableConfig) {
|
||||||
|
rmsg.ID = ev.SubMessage.Timestamp
|
||||||
|
if ev.SubMessage.ThreadTimestamp != ev.SubMessage.Timestamp {
|
||||||
|
b.Log.Debugf("SubMessage %#v", ev.SubMessage)
|
||||||
|
rmsg.Text = ev.SubMessage.Text + b.GetString(editSuffixConfig)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// For edits, only submessage has thread ts.
|
||||||
|
// Ensures edits to threaded messages maintain their prefix hint on the
|
||||||
|
// unthreaded end.
|
||||||
|
if ev.SubMessage != nil {
|
||||||
|
rmsg.ParentID = ev.SubMessage.ThreadTimestamp
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = b.populateMessageWithUserInfo(ev, rmsg); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return rmsg, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) populateMessageWithUserInfo(ev *slack.MessageEvent, rmsg *config.Message) error {
|
||||||
|
if ev.SubType == sMessageDeleted || ev.SubType == sFileComment {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// First, deal with bot-originating messages but only do so when not using webhooks: we
|
||||||
|
// would not be able to distinguish which bot would be sending them.
|
||||||
|
if err := b.populateMessageWithBotInfo(ev, rmsg); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Second, deal with "real" users if we have the necessary information.
|
||||||
|
var userID string
|
||||||
|
switch {
|
||||||
|
case ev.User != "":
|
||||||
|
userID = ev.User
|
||||||
|
case ev.SubMessage != nil && ev.SubMessage.User != "":
|
||||||
|
userID = ev.SubMessage.User
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
user := b.users.getUser(userID)
|
||||||
|
if user == nil {
|
||||||
|
return fmt.Errorf("could not find information for user with id %s", ev.User)
|
||||||
|
}
|
||||||
|
|
||||||
|
rmsg.UserID = user.ID
|
||||||
|
rmsg.Username = user.Name
|
||||||
|
if user.Profile.DisplayName != "" {
|
||||||
|
rmsg.Username = user.Profile.DisplayName
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) populateMessageWithBotInfo(ev *slack.MessageEvent, rmsg *config.Message) error {
|
||||||
|
if ev.BotID == "" || b.GetString(outgoingWebhookConfig) != "" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var err error
|
||||||
|
var bot *slack.Bot
|
||||||
|
for {
|
||||||
|
bot, err = b.rtm.GetBotInfo(ev.BotID)
|
||||||
|
if err == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = handleRateLimit(b.Log, err); err != nil {
|
||||||
|
b.Log.Errorf("Could not retrieve bot information: %#v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.Log.Debugf("Found bot %#v", bot)
|
||||||
|
|
||||||
|
if bot.Name != "" {
|
||||||
|
rmsg.Username = bot.Name
|
||||||
|
if ev.Username != "" {
|
||||||
|
rmsg.Username = ev.Username
|
||||||
|
}
|
||||||
|
rmsg.UserID = bot.ID
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
mentionRE = regexp.MustCompile(`<@([a-zA-Z0-9]+)>`)
|
||||||
|
channelRE = regexp.MustCompile(`<#[a-zA-Z0-9]+\|(.+?)>`)
|
||||||
|
variableRE = regexp.MustCompile(`<!((?:subteam\^)?[a-zA-Z0-9]+)(?:\|@?(.+?))?>`)
|
||||||
|
urlRE = regexp.MustCompile(`<(.*?)(\|.*?)?>`)
|
||||||
|
codeFenceRE = regexp.MustCompile(`(?m)^` + "```" + `\w+$`)
|
||||||
|
topicOrPurposeRE = regexp.MustCompile(`(?s)(@.+) (cleared|set)(?: the)? channel (topic|purpose)(?:: (.*))?`)
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Bslack) extractTopicOrPurpose(text string) (string, string) {
|
||||||
|
r := topicOrPurposeRE.FindStringSubmatch(text)
|
||||||
|
if len(r) == 5 {
|
||||||
|
action, updateType, extracted := r[2], r[3], r[4]
|
||||||
|
switch action {
|
||||||
|
case "set":
|
||||||
|
return updateType, extracted
|
||||||
|
case "cleared":
|
||||||
|
return updateType, ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.Log.Warnf("Encountered channel topic or purpose change message with unexpected format: %s", text)
|
||||||
|
return "unknown", ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// @see https://api.slack.com/docs/message-formatting#linking_to_channels_and_users
|
||||||
|
func (b *Bslack) replaceMention(text string) string {
|
||||||
|
replaceFunc := func(match string) string {
|
||||||
|
userID := strings.Trim(match, "@<>")
|
||||||
|
if username := b.users.getUsername(userID); userID != "" {
|
||||||
|
return "@" + username
|
||||||
|
}
|
||||||
|
return match
|
||||||
|
}
|
||||||
|
return mentionRE.ReplaceAllStringFunc(text, replaceFunc)
|
||||||
|
}
|
||||||
|
|
||||||
|
// @see https://api.slack.com/docs/message-formatting#linking_to_channels_and_users
|
||||||
|
func (b *Bslack) replaceChannel(text string) string {
|
||||||
|
for _, r := range channelRE.FindAllStringSubmatch(text, -1) {
|
||||||
|
text = strings.Replace(text, r[0], "#"+r[1], 1)
|
||||||
|
}
|
||||||
|
return text
|
||||||
|
}
|
||||||
|
|
||||||
|
// @see https://api.slack.com/docs/message-formatting#variables
|
||||||
|
func (b *Bslack) replaceVariable(text string) string {
|
||||||
|
for _, r := range variableRE.FindAllStringSubmatch(text, -1) {
|
||||||
|
if r[2] != "" {
|
||||||
|
text = strings.Replace(text, r[0], "@"+r[2], 1)
|
||||||
|
} else {
|
||||||
|
text = strings.Replace(text, r[0], "@"+r[1], 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return text
|
||||||
|
}
|
||||||
|
|
||||||
|
// @see https://api.slack.com/docs/message-formatting#linking_to_urls
|
||||||
|
func (b *Bslack) replaceURL(text string) string {
|
||||||
|
for _, r := range urlRE.FindAllStringSubmatch(text, -1) {
|
||||||
|
if len(strings.TrimSpace(r[2])) == 1 { // A display text separator was found, but the text was blank
|
||||||
|
text = strings.Replace(text, r[0], "", 1)
|
||||||
|
} else {
|
||||||
|
text = strings.Replace(text, r[0], r[1], 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return text
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) replaceCodeFence(text string) string {
|
||||||
|
return codeFenceRE.ReplaceAllString(text, "```")
|
||||||
|
}
|
||||||
|
|
||||||
|
// getUsersInConversation returns an array of userIDs that are members of channelID
|
||||||
|
func (b *Bslack) getUsersInConversation(channelID string) ([]string, error) {
|
||||||
|
channelMembers := []string{}
|
||||||
|
for {
|
||||||
|
queryParams := &slack.GetUsersInConversationParameters{
|
||||||
|
ChannelID: channelID,
|
||||||
|
}
|
||||||
|
|
||||||
|
members, nextCursor, err := b.sc.GetUsersInConversation(queryParams)
|
||||||
|
if err != nil {
|
||||||
|
if err = handleRateLimit(b.Log, err); err != nil {
|
||||||
|
return channelMembers, fmt.Errorf("Could not retrieve users in channels: %#v", err)
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
channelMembers = append(channelMembers, members...)
|
||||||
|
|
||||||
|
if nextCursor == "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
queryParams.Cursor = nextCursor
|
||||||
|
}
|
||||||
|
return channelMembers, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleRateLimit(log *logrus.Entry, err error) error {
|
||||||
|
rateLimit, ok := err.(*slack.RateLimitedError)
|
||||||
|
if !ok {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
log.Infof("Rate-limited by Slack. Sleeping for %v", rateLimit.RetryAfter)
|
||||||
|
time.Sleep(rateLimit.RetryAfter)
|
||||||
|
return nil
|
||||||
|
}
|
36
bridge/slack/helpers_test.go
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
package bslack
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestExtractTopicOrPurpose(t *testing.T) {
|
||||||
|
testcases := map[string]struct {
|
||||||
|
input string
|
||||||
|
wantChangeType string
|
||||||
|
wantOutput string
|
||||||
|
}{
|
||||||
|
"success - topic type": {"@someone set channel topic: foo bar", "topic", "foo bar"},
|
||||||
|
"success - purpose type": {"@someone set channel purpose: foo bar", "purpose", "foo bar"},
|
||||||
|
"success - one line": {"@someone set channel topic: foo bar", "topic", "foo bar"},
|
||||||
|
"success - multi-line": {"@someone set channel topic: foo\nbar", "topic", "foo\nbar"},
|
||||||
|
"success - cleared": {"@someone cleared channel topic", "topic", ""},
|
||||||
|
"error - unhandled": {"some unmatched message", "unknown", ""},
|
||||||
|
}
|
||||||
|
|
||||||
|
logger := logrus.New()
|
||||||
|
logger.SetOutput(ioutil.Discard)
|
||||||
|
cfg := &bridge.Config{Bridge: &bridge.Bridge{Log: logrus.NewEntry(logger)}}
|
||||||
|
b := newBridge(cfg)
|
||||||
|
for name, tc := range testcases {
|
||||||
|
gotChangeType, gotOutput := b.extractTopicOrPurpose(tc.input)
|
||||||
|
|
||||||
|
assert.Equalf(t, tc.wantChangeType, gotChangeType, "This testcase failed: %s", name)
|
||||||
|
assert.Equalf(t, tc.wantOutput, gotOutput, "This testcase failed: %s", name)
|
||||||
|
}
|
||||||
|
}
|
78
bridge/slack/legacy.go
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
package bslack
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/matterhook"
|
||||||
|
"github.com/nlopes/slack"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BLegacy struct {
|
||||||
|
*Bslack
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewLegacy(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
return &BLegacy{Bslack: newBridge(cfg)}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *BLegacy) Connect() error {
|
||||||
|
b.RLock()
|
||||||
|
defer b.RUnlock()
|
||||||
|
if b.GetString(incomingWebhookConfig) != "" {
|
||||||
|
switch {
|
||||||
|
case b.GetString(outgoingWebhookConfig) != "":
|
||||||
|
b.Log.Info("Connecting using webhookurl (sending) and webhookbindaddress (receiving)")
|
||||||
|
b.mh = matterhook.New(b.GetString(outgoingWebhookConfig), matterhook.Config{
|
||||||
|
InsecureSkipVerify: b.GetBool(skipTLSConfig),
|
||||||
|
BindAddress: b.GetString(incomingWebhookConfig),
|
||||||
|
})
|
||||||
|
case b.GetString(tokenConfig) != "":
|
||||||
|
b.Log.Info("Connecting using token (sending)")
|
||||||
|
b.sc = slack.New(b.GetString(tokenConfig))
|
||||||
|
b.rtm = b.sc.NewRTM()
|
||||||
|
go b.rtm.ManageConnection()
|
||||||
|
b.Log.Info("Connecting using webhookbindaddress (receiving)")
|
||||||
|
b.mh = matterhook.New(b.GetString(outgoingWebhookConfig), matterhook.Config{
|
||||||
|
InsecureSkipVerify: b.GetBool(skipTLSConfig),
|
||||||
|
BindAddress: b.GetString(incomingWebhookConfig),
|
||||||
|
})
|
||||||
|
default:
|
||||||
|
b.Log.Info("Connecting using webhookbindaddress (receiving)")
|
||||||
|
b.mh = matterhook.New(b.GetString(outgoingWebhookConfig), matterhook.Config{
|
||||||
|
InsecureSkipVerify: b.GetBool(skipTLSConfig),
|
||||||
|
BindAddress: b.GetString(incomingWebhookConfig),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
go b.handleSlack()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if b.GetString(outgoingWebhookConfig) != "" {
|
||||||
|
b.Log.Info("Connecting using webhookurl (sending)")
|
||||||
|
b.mh = matterhook.New(b.GetString(outgoingWebhookConfig), matterhook.Config{
|
||||||
|
InsecureSkipVerify: b.GetBool(skipTLSConfig),
|
||||||
|
DisableServer: true,
|
||||||
|
})
|
||||||
|
if b.GetString(tokenConfig) != "" {
|
||||||
|
b.Log.Info("Connecting using token (receiving)")
|
||||||
|
b.sc = slack.New(b.GetString(tokenConfig), slack.OptionDebug(b.GetBool("debug")))
|
||||||
|
b.channels = newChannelManager(b.Log, b.sc)
|
||||||
|
b.users = newUserManager(b.Log, b.sc)
|
||||||
|
b.rtm = b.sc.NewRTM()
|
||||||
|
go b.rtm.ManageConnection()
|
||||||
|
go b.handleSlack()
|
||||||
|
}
|
||||||
|
} else if b.GetString(tokenConfig) != "" {
|
||||||
|
b.Log.Info("Connecting using token (sending and receiving)")
|
||||||
|
b.sc = slack.New(b.GetString(tokenConfig), slack.OptionDebug(b.GetBool("debug")))
|
||||||
|
b.channels = newChannelManager(b.Log, b.sc)
|
||||||
|
b.users = newUserManager(b.Log, b.sc)
|
||||||
|
b.rtm = b.sc.NewRTM()
|
||||||
|
go b.rtm.ManageConnection()
|
||||||
|
go b.handleSlack()
|
||||||
|
}
|
||||||
|
if b.GetString(incomingWebhookConfig) == "" && b.GetString(outgoingWebhookConfig) == "" && b.GetString(tokenConfig) == "" {
|
||||||
|
return errors.New("no connection method found. See that you have WebhookBindAddress, WebhookURL or Token configured")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
@ -1,48 +1,91 @@
|
|||||||
package bslack
|
package bslack
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
|
||||||
"github.com/42wim/matterbridge/matterhook"
|
|
||||||
log "github.com/Sirupsen/logrus"
|
|
||||||
"github.com/nlopes/slack"
|
|
||||||
"regexp"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/42wim/matterbridge/matterhook"
|
||||||
|
"github.com/hashicorp/golang-lru"
|
||||||
|
"github.com/nlopes/slack"
|
||||||
|
"github.com/rs/xid"
|
||||||
)
|
)
|
||||||
|
|
||||||
type MMMessage struct {
|
|
||||||
Text string
|
|
||||||
Channel string
|
|
||||||
Username string
|
|
||||||
Raw *slack.MessageEvent
|
|
||||||
}
|
|
||||||
|
|
||||||
type Bslack struct {
|
type Bslack struct {
|
||||||
mh *matterhook.Client
|
sync.RWMutex
|
||||||
sc *slack.Client
|
*bridge.Config
|
||||||
Config *config.Protocol
|
|
||||||
rtm *slack.RTM
|
mh *matterhook.Client
|
||||||
Plus bool
|
sc *slack.Client
|
||||||
Remote chan config.Message
|
rtm *slack.RTM
|
||||||
Users []slack.User
|
si *slack.Info
|
||||||
Account string
|
|
||||||
si *slack.Info
|
cache *lru.Cache
|
||||||
channels []slack.Channel
|
uuid string
|
||||||
|
useChannelID bool
|
||||||
|
|
||||||
|
channels *channels
|
||||||
|
users *users
|
||||||
}
|
}
|
||||||
|
|
||||||
var flog *log.Entry
|
const (
|
||||||
var protocol = "slack"
|
sChannelJoin = "channel_join"
|
||||||
|
sChannelLeave = "channel_leave"
|
||||||
|
sChannelJoined = "channel_joined"
|
||||||
|
sMemberJoined = "member_joined_channel"
|
||||||
|
sMessageChanged = "message_changed"
|
||||||
|
sMessageDeleted = "message_deleted"
|
||||||
|
sSlackAttachment = "slack_attachment"
|
||||||
|
sPinnedItem = "pinned_item"
|
||||||
|
sUnpinnedItem = "unpinned_item"
|
||||||
|
sChannelTopic = "channel_topic"
|
||||||
|
sChannelPurpose = "channel_purpose"
|
||||||
|
sFileComment = "file_comment"
|
||||||
|
sMeMessage = "me_message"
|
||||||
|
sUserTyping = "user_typing"
|
||||||
|
sLatencyReport = "latency_report"
|
||||||
|
sSystemUser = "system"
|
||||||
|
sSlackBotUser = "slackbot"
|
||||||
|
|
||||||
func init() {
|
tokenConfig = "Token"
|
||||||
flog = log.WithFields(log.Fields{"module": protocol})
|
incomingWebhookConfig = "WebhookBindAddress"
|
||||||
|
outgoingWebhookConfig = "WebhookURL"
|
||||||
|
skipTLSConfig = "SkipTLSVerify"
|
||||||
|
useNickPrefixConfig = "PrefixMessagesWithNick"
|
||||||
|
editDisableConfig = "EditDisable"
|
||||||
|
editSuffixConfig = "EditSuffix"
|
||||||
|
iconURLConfig = "iconurl"
|
||||||
|
noSendJoinConfig = "nosendjoinpart"
|
||||||
|
)
|
||||||
|
|
||||||
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
// Print a deprecation warning for legacy non-bot tokens (#527).
|
||||||
|
token := cfg.GetString(tokenConfig)
|
||||||
|
if token != "" && !strings.HasPrefix(token, "xoxb") {
|
||||||
|
cfg.Log.Warn("Non-bot token detected. It is STRONGLY recommended to use a proper bot-token instead.")
|
||||||
|
cfg.Log.Warn("Legacy tokens may be deprecated by Slack at short notice. See the Matterbridge GitHub wiki for a migration guide.")
|
||||||
|
cfg.Log.Warn("See https://github.com/42wim/matterbridge/wiki/Slack-bot-setup")
|
||||||
|
return NewLegacy(cfg)
|
||||||
|
}
|
||||||
|
return newBridge(cfg)
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(cfg config.Protocol, account string, c chan config.Message) *Bslack {
|
func newBridge(cfg *bridge.Config) *Bslack {
|
||||||
b := &Bslack{}
|
newCache, err := lru.New(5000)
|
||||||
b.Config = &cfg
|
if err != nil {
|
||||||
b.Remote = c
|
cfg.Log.Fatalf("Could not create LRU cache for Slack bridge: %v", err)
|
||||||
b.Account = account
|
}
|
||||||
|
b := &Bslack{
|
||||||
|
Config: cfg,
|
||||||
|
uuid: xid.New().String(),
|
||||||
|
cache: newCache,
|
||||||
|
}
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -51,201 +94,440 @@ func (b *Bslack) Command(cmd string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) Connect() error {
|
func (b *Bslack) Connect() error {
|
||||||
flog.Info("Connecting")
|
b.RLock()
|
||||||
if !b.Config.UseAPI {
|
defer b.RUnlock()
|
||||||
b.mh = matterhook.New(b.Config.URL,
|
|
||||||
matterhook.Config{BindAddress: b.Config.BindAddress})
|
if b.GetString(incomingWebhookConfig) == "" && b.GetString(outgoingWebhookConfig) == "" && b.GetString(tokenConfig) == "" {
|
||||||
} else {
|
return errors.New("no connection method found: WebhookBindAddress, WebhookURL or Token need to be configured")
|
||||||
b.sc = slack.New(b.Config.Token)
|
}
|
||||||
|
|
||||||
|
// If we have a token we use the Slack websocket-based RTM for both sending and receiving.
|
||||||
|
if token := b.GetString(tokenConfig); token != "" {
|
||||||
|
b.Log.Info("Connecting using token")
|
||||||
|
|
||||||
|
b.sc = slack.New(token, slack.OptionDebug(b.GetBool("Debug")))
|
||||||
|
|
||||||
|
b.channels = newChannelManager(b.Log, b.sc)
|
||||||
|
b.users = newUserManager(b.Log, b.sc)
|
||||||
|
|
||||||
b.rtm = b.sc.NewRTM()
|
b.rtm = b.sc.NewRTM()
|
||||||
go b.rtm.ManageConnection()
|
go b.rtm.ManageConnection()
|
||||||
|
go b.handleSlack()
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
flog.Info("Connection succeeded")
|
|
||||||
go b.handleSlack()
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bslack) JoinChannel(channel string) error {
|
// In absence of a token we fall back to incoming and outgoing Webhooks.
|
||||||
// we can only join channels using the API
|
b.mh = matterhook.New(
|
||||||
if b.Config.UseAPI {
|
"",
|
||||||
_, err := b.sc.JoinChannel(channel)
|
matterhook.Config{
|
||||||
if err != nil {
|
InsecureSkipVerify: b.GetBool("SkipTLSVerify"),
|
||||||
return err
|
DisableServer: true,
|
||||||
}
|
},
|
||||||
|
)
|
||||||
|
if b.GetString(outgoingWebhookConfig) != "" {
|
||||||
|
b.Log.Info("Using specified webhook for outgoing messages.")
|
||||||
|
b.mh.Url = b.GetString(outgoingWebhookConfig)
|
||||||
|
}
|
||||||
|
if b.GetString(incomingWebhookConfig) != "" {
|
||||||
|
b.Log.Info("Setting up local webhook for incoming messages.")
|
||||||
|
b.mh.BindAddress = b.GetString(incomingWebhookConfig)
|
||||||
|
b.mh.DisableServer = false
|
||||||
|
go b.handleSlack()
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) Send(msg config.Message) error {
|
func (b *Bslack) Disconnect() error {
|
||||||
flog.Debugf("Receiving %#v", msg)
|
return b.rtm.Disconnect()
|
||||||
if msg.Account == b.Account {
|
}
|
||||||
|
|
||||||
|
// JoinChannel only acts as a verification method that checks whether Matterbridge's
|
||||||
|
// Slack integration is already member of the channel. This is because Slack does not
|
||||||
|
// allow apps or bots to join channels themselves and they need to be invited
|
||||||
|
// manually by a user.
|
||||||
|
func (b *Bslack) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
// We can only join a channel through the Slack API.
|
||||||
|
if b.sc == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
nick := msg.Username
|
|
||||||
message := msg.Text
|
b.channels.populateChannels(false)
|
||||||
channel := msg.Channel
|
|
||||||
if b.Config.PrefixMessagesWithNick {
|
channelInfo, err := b.channels.getChannel(channel.Name)
|
||||||
message = nick + " " + message
|
|
||||||
}
|
|
||||||
if !b.Config.UseAPI {
|
|
||||||
matterMessage := matterhook.OMessage{IconURL: b.Config.IconURL}
|
|
||||||
matterMessage.Channel = channel
|
|
||||||
matterMessage.UserName = nick
|
|
||||||
matterMessage.Type = ""
|
|
||||||
matterMessage.Text = message
|
|
||||||
err := b.mh.Send(matterMessage)
|
|
||||||
if err != nil {
|
|
||||||
flog.Info(err)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
schannel, err := b.getChannelByName(channel)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
return fmt.Errorf("could not join channel: %#v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.HasPrefix(channel.Name, "ID:") {
|
||||||
|
b.useChannelID = true
|
||||||
|
channel.Name = channelInfo.Name
|
||||||
|
}
|
||||||
|
|
||||||
|
if !channelInfo.IsMember {
|
||||||
|
return fmt.Errorf("slack integration that matterbridge is using is not member of channel '%s', please add it manually", channelInfo.Name)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) Reload(cfg *bridge.Config) (string, error) {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) Send(msg config.Message) (string, error) {
|
||||||
|
// Too noisy to log like other events
|
||||||
|
if msg.Event != config.EventUserTyping {
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
msg.Text = b.replaceCodeFence(msg.Text)
|
||||||
|
|
||||||
|
// Make a action /me of the message
|
||||||
|
if msg.Event == config.EventUserAction {
|
||||||
|
msg.Text = "_" + msg.Text + "_"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use webhook to send the message
|
||||||
|
if b.GetString(outgoingWebhookConfig) != "" {
|
||||||
|
return "", b.sendWebhook(msg)
|
||||||
|
}
|
||||||
|
return b.sendRTM(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// sendWebhook uses the configured WebhookURL to send the message
|
||||||
|
func (b *Bslack) sendWebhook(msg config.Message) error {
|
||||||
|
// Skip events.
|
||||||
|
if msg.Event != "" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if b.GetBool(useNickPrefixConfig) {
|
||||||
|
msg.Text = msg.Username + msg.Text
|
||||||
|
}
|
||||||
|
|
||||||
|
if msg.Extra != nil {
|
||||||
|
// This sends a message only if we received a config.EVENT_FILE_FAILURE_SIZE.
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
rmsg := rmsg // scopelint
|
||||||
|
iconURL := config.GetIconURL(&rmsg, b.GetString(iconURLConfig))
|
||||||
|
matterMessage := matterhook.OMessage{
|
||||||
|
IconURL: iconURL,
|
||||||
|
Channel: msg.Channel,
|
||||||
|
UserName: rmsg.Username,
|
||||||
|
Text: rmsg.Text,
|
||||||
|
}
|
||||||
|
if err := b.mh.Send(matterMessage); err != nil {
|
||||||
|
b.Log.Errorf("Failed to send message: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Webhook doesn't support file uploads, so we add the URL manually.
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi, ok := f.(config.FileInfo)
|
||||||
|
if !ok {
|
||||||
|
b.Log.Errorf("Received a file with unexpected content: %#v", f)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text += " " + fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we have native slack_attachments add them.
|
||||||
|
var attachs []slack.Attachment
|
||||||
|
for _, attach := range msg.Extra[sSlackAttachment] {
|
||||||
|
attachs = append(attachs, attach.([]slack.Attachment)...)
|
||||||
|
}
|
||||||
|
|
||||||
|
iconURL := config.GetIconURL(&msg, b.GetString(iconURLConfig))
|
||||||
|
matterMessage := matterhook.OMessage{
|
||||||
|
IconURL: iconURL,
|
||||||
|
Attachments: attachs,
|
||||||
|
Channel: msg.Channel,
|
||||||
|
UserName: msg.Username,
|
||||||
|
Text: msg.Text,
|
||||||
|
}
|
||||||
|
if msg.Avatar != "" {
|
||||||
|
matterMessage.IconURL = msg.Avatar
|
||||||
|
}
|
||||||
|
if err := b.mh.Send(matterMessage); err != nil {
|
||||||
|
b.Log.Errorf("Failed to send message via webhook: %#v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
np := slack.NewPostMessageParameters()
|
|
||||||
if b.Config.PrefixMessagesWithNick == true {
|
|
||||||
np.AsUser = true
|
|
||||||
}
|
|
||||||
np.Username = nick
|
|
||||||
np.IconURL = config.GetIconURL(&msg, b.Config)
|
|
||||||
if msg.Avatar != "" {
|
|
||||||
np.IconURL = msg.Avatar
|
|
||||||
}
|
|
||||||
b.sc.PostMessage(schannel.ID, message, np)
|
|
||||||
|
|
||||||
/*
|
|
||||||
newmsg := b.rtm.NewOutgoingMessage(message, schannel.ID)
|
|
||||||
b.rtm.SendMessage(newmsg)
|
|
||||||
*/
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) getAvatar(user string) string {
|
func (b *Bslack) sendRTM(msg config.Message) (string, error) {
|
||||||
var avatar string
|
// Handle channelmember messages.
|
||||||
if b.Users != nil {
|
if handled := b.handleGetChannelMembers(&msg); handled {
|
||||||
for _, u := range b.Users {
|
return "", nil
|
||||||
if user == u.Name {
|
}
|
||||||
return u.Profile.Image48
|
|
||||||
|
channelInfo, err := b.channels.getChannel(msg.Channel)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("could not send message: %v", err)
|
||||||
|
}
|
||||||
|
if msg.Event == config.EventUserTyping {
|
||||||
|
if b.GetBool("ShowUserTyping") {
|
||||||
|
b.rtm.SendMessage(b.rtm.NewTypingMessage(channelInfo.ID))
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var handled bool
|
||||||
|
|
||||||
|
// Handle topic/purpose updates.
|
||||||
|
if handled, err = b.handleTopicOrPurpose(&msg, channelInfo); handled {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle prefix hint for unthreaded messages.
|
||||||
|
if msg.ParentID == "msg-parent-not-found" {
|
||||||
|
msg.ParentID = ""
|
||||||
|
msg.Text = fmt.Sprintf("[thread]: %s", msg.Text)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle message deletions.
|
||||||
|
if handled, err = b.deleteMessage(&msg, channelInfo); handled {
|
||||||
|
return msg.ID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prepend nickname if configured.
|
||||||
|
if b.GetBool(useNickPrefixConfig) {
|
||||||
|
msg.Text = msg.Username + msg.Text
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle message edits.
|
||||||
|
if handled, err = b.editMessage(&msg, channelInfo); handled {
|
||||||
|
return msg.ID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload a file if it exists.
|
||||||
|
if msg.Extra != nil {
|
||||||
|
extraMsgs := helper.HandleExtra(&msg, b.General)
|
||||||
|
for i := range extraMsgs {
|
||||||
|
rmsg := &extraMsgs[i]
|
||||||
|
rmsg.Text = rmsg.Username + rmsg.Text
|
||||||
|
_, err = b.postMessage(rmsg, channelInfo)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Error(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Upload files if necessary (from Slack, Telegram or Mattermost).
|
||||||
|
b.uploadFile(&msg, channelInfo.ID)
|
||||||
}
|
}
|
||||||
return avatar
|
|
||||||
|
// Post message.
|
||||||
|
return b.postMessage(&msg, channelInfo)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) getChannelByName(name string) (*slack.Channel, error) {
|
func (b *Bslack) updateTopicOrPurpose(msg *config.Message, channelInfo *slack.Channel) error {
|
||||||
if b.channels == nil {
|
var updateFunc func(channelID string, value string) (*slack.Channel, error)
|
||||||
return nil, fmt.Errorf("%s: channel %s not found (no channels found)", b.Account, name)
|
|
||||||
|
incomingChangeType, text := b.extractTopicOrPurpose(msg.Text)
|
||||||
|
switch incomingChangeType {
|
||||||
|
case "topic":
|
||||||
|
updateFunc = b.rtm.SetTopicOfConversation
|
||||||
|
case "purpose":
|
||||||
|
updateFunc = b.rtm.SetPurposeOfConversation
|
||||||
|
default:
|
||||||
|
b.Log.Errorf("Unhandled type received from extractTopicOrPurpose: %s", incomingChangeType)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
for _, channel := range b.channels {
|
for {
|
||||||
if channel.Name == name {
|
_, err := updateFunc(channelInfo.ID, text)
|
||||||
return &channel, nil
|
if err == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if err = handleRateLimit(b.Log, err); err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil, fmt.Errorf("%s: channel %s not found", b.Account, name)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) handleSlack() {
|
// handles updating topic/purpose and determining whether to further propagate update messages.
|
||||||
flog.Debugf("Choosing API based slack connection: %t", b.Config.UseAPI)
|
func (b *Bslack) handleTopicOrPurpose(msg *config.Message, channelInfo *slack.Channel) (bool, error) {
|
||||||
mchan := make(chan *MMMessage)
|
if msg.Event != config.EventTopicChange {
|
||||||
if b.Config.UseAPI {
|
return false, nil
|
||||||
go b.handleSlackClient(mchan)
|
|
||||||
} else {
|
|
||||||
go b.handleMatterHook(mchan)
|
|
||||||
}
|
}
|
||||||
time.Sleep(time.Second)
|
|
||||||
flog.Debug("Start listening for Slack messages")
|
if b.GetBool("SyncTopic") {
|
||||||
for message := range mchan {
|
return true, b.updateTopicOrPurpose(msg, channelInfo)
|
||||||
// do not send messages from ourself
|
}
|
||||||
if b.Config.UseAPI && message.Username == b.si.User.Name {
|
|
||||||
|
// Pass along to normal message handlers.
|
||||||
|
if b.GetBool("ShowTopicChange") {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Swallow message as handled no-op.
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) deleteMessage(msg *config.Message, channelInfo *slack.Channel) (bool, error) {
|
||||||
|
if msg.Event != config.EventMsgDelete {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Some protocols echo deletes, but with an empty ID.
|
||||||
|
if msg.ID == "" {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
_, _, err := b.rtm.DeleteMessage(channelInfo.ID, msg.ID)
|
||||||
|
if err == nil {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = handleRateLimit(b.Log, err); err != nil {
|
||||||
|
b.Log.Errorf("Failed to delete user message from Slack: %#v", err)
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) editMessage(msg *config.Message, channelInfo *slack.Channel) (bool, error) {
|
||||||
|
if msg.ID == "" {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
messageOptions := b.prepareMessageOptions(msg)
|
||||||
|
for {
|
||||||
|
messageOptions = append(messageOptions, slack.MsgOptionText(msg.Text, false))
|
||||||
|
_, _, _, err := b.rtm.UpdateMessage(channelInfo.ID, msg.ID, messageOptions...)
|
||||||
|
if err == nil {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = handleRateLimit(b.Log, err); err != nil {
|
||||||
|
b.Log.Errorf("Failed to edit user message on Slack: %#v", err)
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bslack) postMessage(msg *config.Message, channelInfo *slack.Channel) (string, error) {
|
||||||
|
// don't post empty messages
|
||||||
|
if msg.Text == "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
messageOptions := b.prepareMessageOptions(msg)
|
||||||
|
messageOptions = append(
|
||||||
|
messageOptions,
|
||||||
|
slack.MsgOptionText(msg.Text, false),
|
||||||
|
slack.MsgOptionEnableLinkUnfurl(),
|
||||||
|
)
|
||||||
|
for {
|
||||||
|
_, id, err := b.rtm.PostMessage(channelInfo.ID, messageOptions...)
|
||||||
|
if err == nil {
|
||||||
|
return id, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = handleRateLimit(b.Log, err); err != nil {
|
||||||
|
b.Log.Errorf("Failed to sent user message to Slack: %#v", err)
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// uploadFile handles native upload of files
|
||||||
|
func (b *Bslack) uploadFile(msg *config.Message, channelID string) {
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi, ok := f.(config.FileInfo)
|
||||||
|
if !ok {
|
||||||
|
b.Log.Errorf("Received a file with unexpected content: %#v", f)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
texts := strings.Split(message.Text, "\n")
|
if msg.Text == fi.Comment {
|
||||||
for _, text := range texts {
|
msg.Text = ""
|
||||||
flog.Debugf("Sending message from %s on %s to gateway", message.Username, b.Account)
|
}
|
||||||
b.Remote <- config.Message{Text: text, Username: message.Username, Channel: message.Channel, Account: b.Account, Avatar: b.getAvatar(message.Username)}
|
// Because the result of the UploadFile is slower than the MessageEvent from slack
|
||||||
|
// we can't match on the file ID yet, so we have to match on the filename too.
|
||||||
|
ts := time.Now()
|
||||||
|
b.Log.Debugf("Adding file %s to cache at %s with timestamp", fi.Name, ts.String())
|
||||||
|
b.cache.Add("filename"+fi.Name, ts)
|
||||||
|
initialComment := fmt.Sprintf("File from %s", msg.Username)
|
||||||
|
if fi.Comment != "" {
|
||||||
|
initialComment += fmt.Sprintf("with comment: %s", fi.Comment)
|
||||||
|
}
|
||||||
|
res, err := b.sc.UploadFile(slack.FileUploadParameters{
|
||||||
|
Reader: bytes.NewReader(*fi.Data),
|
||||||
|
Filename: fi.Name,
|
||||||
|
Channels: []string{channelID},
|
||||||
|
InitialComment: initialComment,
|
||||||
|
ThreadTimestamp: msg.ParentID,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("uploadfile %#v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if res.ID != "" {
|
||||||
|
b.Log.Debugf("Adding file ID %s to cache with timestamp %s", res.ID, ts.String())
|
||||||
|
b.cache.Add("file"+res.ID, ts)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) handleSlackClient(mchan chan *MMMessage) {
|
func (b *Bslack) prepareMessageOptions(msg *config.Message) []slack.MsgOption {
|
||||||
count := 0
|
params := slack.NewPostMessageParameters()
|
||||||
for msg := range b.rtm.IncomingEvents {
|
if b.GetBool(useNickPrefixConfig) {
|
||||||
switch ev := msg.Data.(type) {
|
params.AsUser = true
|
||||||
case *slack.MessageEvent:
|
}
|
||||||
// ignore first message
|
params.Username = msg.Username
|
||||||
if count > 0 {
|
params.LinkNames = 1 // replace mentions
|
||||||
flog.Debugf("Receiving from slackclient %#v", ev)
|
params.IconURL = config.GetIconURL(msg, b.GetString(iconURLConfig))
|
||||||
//ev.ReplyTo
|
params.ThreadTimestamp = msg.ParentID
|
||||||
channel, err := b.rtm.GetChannelInfo(ev.Channel)
|
if msg.Avatar != "" {
|
||||||
if err != nil {
|
params.IconURL = msg.Avatar
|
||||||
continue
|
}
|
||||||
}
|
|
||||||
user, err := b.rtm.GetUserInfo(ev.User)
|
var attachments []slack.Attachment
|
||||||
if err != nil {
|
// add a callback ID so we can see we created it
|
||||||
continue
|
attachments = append(attachments, slack.Attachment{CallbackID: "matterbridge_" + b.uuid})
|
||||||
}
|
// add file attachments
|
||||||
m := &MMMessage{}
|
attachments = append(attachments, b.createAttach(msg.Extra)...)
|
||||||
m.Username = user.Name
|
// add slack attachments (from another slack bridge)
|
||||||
m.Channel = channel.Name
|
if msg.Extra != nil {
|
||||||
m.Text = ev.Text
|
for _, attach := range msg.Extra[sSlackAttachment] {
|
||||||
m.Raw = ev
|
attachments = append(attachments, attach.([]slack.Attachment)...)
|
||||||
m.Text = b.replaceMention(m.Text)
|
|
||||||
mchan <- m
|
|
||||||
}
|
|
||||||
count++
|
|
||||||
case *slack.OutgoingErrorEvent:
|
|
||||||
flog.Debugf("%#v", ev.Error())
|
|
||||||
case *slack.ChannelJoinedEvent:
|
|
||||||
b.Users, _ = b.sc.GetUsers()
|
|
||||||
case *slack.ConnectedEvent:
|
|
||||||
b.channels = ev.Info.Channels
|
|
||||||
b.si = ev.Info
|
|
||||||
b.Users, _ = b.sc.GetUsers()
|
|
||||||
// add private channels
|
|
||||||
groups, _ := b.sc.GetGroups(true)
|
|
||||||
for _, g := range groups {
|
|
||||||
channel := new(slack.Channel)
|
|
||||||
channel.ID = g.ID
|
|
||||||
channel.Name = g.Name
|
|
||||||
b.channels = append(b.channels, *channel)
|
|
||||||
}
|
|
||||||
case *slack.InvalidAuthEvent:
|
|
||||||
flog.Fatalf("Invalid Token %#v", ev)
|
|
||||||
default:
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var opts []slack.MsgOption
|
||||||
|
opts = append(opts, slack.MsgOptionAttachments(attachments...))
|
||||||
|
opts = append(opts, slack.MsgOptionPostMessageParameters(params))
|
||||||
|
return opts
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) handleMatterHook(mchan chan *MMMessage) {
|
func (b *Bslack) createAttach(extra map[string][]interface{}) []slack.Attachment {
|
||||||
for {
|
var attachements []slack.Attachment
|
||||||
message := b.mh.Receive()
|
for _, v := range extra["attachments"] {
|
||||||
flog.Debugf("receiving from matterhook (slack) %#v", message)
|
entry := v.(map[string]interface{})
|
||||||
m := &MMMessage{}
|
s := slack.Attachment{
|
||||||
m.Username = message.UserName
|
Fallback: extractStringField(entry, "fallback"),
|
||||||
m.Text = message.Text
|
Color: extractStringField(entry, "color"),
|
||||||
m.Text = b.replaceMention(m.Text)
|
Pretext: extractStringField(entry, "pretext"),
|
||||||
m.Channel = message.ChannelName
|
AuthorName: extractStringField(entry, "author_name"),
|
||||||
mchan <- m
|
AuthorLink: extractStringField(entry, "author_link"),
|
||||||
|
AuthorIcon: extractStringField(entry, "author_icon"),
|
||||||
|
Title: extractStringField(entry, "title"),
|
||||||
|
TitleLink: extractStringField(entry, "title_link"),
|
||||||
|
Text: extractStringField(entry, "text"),
|
||||||
|
ImageURL: extractStringField(entry, "image_url"),
|
||||||
|
ThumbURL: extractStringField(entry, "thumb_url"),
|
||||||
|
Footer: extractStringField(entry, "footer"),
|
||||||
|
FooterIcon: extractStringField(entry, "footer_icon"),
|
||||||
|
}
|
||||||
|
attachements = append(attachements, s)
|
||||||
}
|
}
|
||||||
|
return attachements
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) userName(id string) string {
|
func extractStringField(data map[string]interface{}, field string) string {
|
||||||
for _, u := range b.Users {
|
if rawValue, found := data[field]; found {
|
||||||
if u.ID == id {
|
if value, ok := rawValue.(string); ok {
|
||||||
return u.Name
|
return value
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bslack) replaceMention(text string) string {
|
|
||||||
results := regexp.MustCompile(`<@([a-zA-z0-9]+)>`).FindAllStringSubmatch(text, -1)
|
|
||||||
for _, r := range results {
|
|
||||||
text = strings.Replace(text, "<@"+r[1]+">", "@"+b.userName(r[1]), -1)
|
|
||||||
|
|
||||||
}
|
|
||||||
return text
|
|
||||||
}
|
|
||||||
|
335
bridge/slack/users_channels.go
Normal file
@ -0,0 +1,335 @@
|
|||||||
|
package bslack
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/nlopes/slack"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
const minimumRefreshInterval = 10 * time.Second
|
||||||
|
|
||||||
|
type users struct {
|
||||||
|
log *logrus.Entry
|
||||||
|
sc *slack.Client
|
||||||
|
|
||||||
|
users map[string]*slack.User
|
||||||
|
usersMutex sync.RWMutex
|
||||||
|
usersSyncPoints map[string]chan struct{}
|
||||||
|
|
||||||
|
refreshInProgress bool
|
||||||
|
earliestRefresh time.Time
|
||||||
|
refreshMutex sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func newUserManager(log *logrus.Entry, sc *slack.Client) *users {
|
||||||
|
return &users{
|
||||||
|
log: log,
|
||||||
|
sc: sc,
|
||||||
|
users: make(map[string]*slack.User),
|
||||||
|
usersSyncPoints: make(map[string]chan struct{}),
|
||||||
|
earliestRefresh: time.Now(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *users) getUser(id string) *slack.User {
|
||||||
|
b.usersMutex.RLock()
|
||||||
|
user, ok := b.users[id]
|
||||||
|
b.usersMutex.RUnlock()
|
||||||
|
if ok {
|
||||||
|
return user
|
||||||
|
}
|
||||||
|
b.populateUser(id)
|
||||||
|
b.usersMutex.RLock()
|
||||||
|
defer b.usersMutex.RUnlock()
|
||||||
|
|
||||||
|
return b.users[id]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *users) getUsername(id string) string {
|
||||||
|
if user := b.getUser(id); user != nil {
|
||||||
|
if user.Profile.DisplayName != "" {
|
||||||
|
return user.Profile.DisplayName
|
||||||
|
}
|
||||||
|
return user.Name
|
||||||
|
}
|
||||||
|
b.log.Warnf("Could not find user with ID '%s'", id)
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *users) getAvatar(id string) string {
|
||||||
|
if user := b.getUser(id); user != nil {
|
||||||
|
return user.Profile.Image48
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *users) populateUser(userID string) {
|
||||||
|
for {
|
||||||
|
b.usersMutex.Lock()
|
||||||
|
_, exists := b.users[userID]
|
||||||
|
if exists {
|
||||||
|
// already in cache
|
||||||
|
b.usersMutex.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if syncPoint, ok := b.usersSyncPoints[userID]; ok {
|
||||||
|
// Another goroutine is already populating this user for us so wait on it to finish.
|
||||||
|
b.usersMutex.Unlock()
|
||||||
|
<-syncPoint
|
||||||
|
// We do not return and iterate again to check that the entry does indeed exist
|
||||||
|
// in case the previous query failed for some reason.
|
||||||
|
} else {
|
||||||
|
b.usersSyncPoints[userID] = make(chan struct{})
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do not hold the lock while fetching information from Slack
|
||||||
|
// as this might take an unbounded amount of time.
|
||||||
|
b.usersMutex.Unlock()
|
||||||
|
|
||||||
|
user, err := b.sc.GetUserInfo(userID)
|
||||||
|
if err != nil {
|
||||||
|
b.log.Debugf("GetUserInfo failed for %v: %v", userID, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
b.usersMutex.Lock()
|
||||||
|
defer b.usersMutex.Unlock()
|
||||||
|
|
||||||
|
// Register user information.
|
||||||
|
b.users[userID] = user
|
||||||
|
|
||||||
|
// Wake up any waiting goroutines and remove the synchronization point.
|
||||||
|
close(b.usersSyncPoints[userID])
|
||||||
|
delete(b.usersSyncPoints, userID)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *users) populateUsers(wait bool) {
|
||||||
|
b.refreshMutex.Lock()
|
||||||
|
if !wait && (time.Now().Before(b.earliestRefresh) || b.refreshInProgress) {
|
||||||
|
b.log.Debugf("Not refreshing user list as it was done less than %v ago.", minimumRefreshInterval)
|
||||||
|
b.refreshMutex.Unlock()
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for b.refreshInProgress {
|
||||||
|
b.refreshMutex.Unlock()
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
b.refreshMutex.Lock()
|
||||||
|
}
|
||||||
|
b.refreshInProgress = true
|
||||||
|
b.refreshMutex.Unlock()
|
||||||
|
|
||||||
|
newUsers := map[string]*slack.User{}
|
||||||
|
pagination := b.sc.GetUsersPaginated(slack.GetUsersOptionLimit(200))
|
||||||
|
count := 0
|
||||||
|
for {
|
||||||
|
var err error
|
||||||
|
pagination, err = pagination.Next(context.Background())
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
if err != nil {
|
||||||
|
if pagination.Done(err) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = handleRateLimit(b.log, err); err != nil {
|
||||||
|
b.log.Errorf("Could not retrieve users: %#v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range pagination.Users {
|
||||||
|
newUsers[pagination.Users[i].ID] = &pagination.Users[i]
|
||||||
|
}
|
||||||
|
b.log.Debugf("getting %d users", len(pagination.Users))
|
||||||
|
count++
|
||||||
|
// more > 2000 users, slack will complain and ratelimit. break
|
||||||
|
if count > 10 {
|
||||||
|
b.log.Info("Large slack detected > 2000 users, skipping loading complete userlist.")
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
b.usersMutex.Lock()
|
||||||
|
defer b.usersMutex.Unlock()
|
||||||
|
b.users = newUsers
|
||||||
|
|
||||||
|
b.refreshMutex.Lock()
|
||||||
|
defer b.refreshMutex.Unlock()
|
||||||
|
b.earliestRefresh = time.Now().Add(minimumRefreshInterval)
|
||||||
|
b.refreshInProgress = false
|
||||||
|
}
|
||||||
|
|
||||||
|
type channels struct {
|
||||||
|
log *logrus.Entry
|
||||||
|
sc *slack.Client
|
||||||
|
|
||||||
|
channelsByID map[string]*slack.Channel
|
||||||
|
channelsByName map[string]*slack.Channel
|
||||||
|
channelsMutex sync.RWMutex
|
||||||
|
|
||||||
|
channelMembers map[string][]string
|
||||||
|
channelMembersMutex sync.RWMutex
|
||||||
|
|
||||||
|
refreshInProgress bool
|
||||||
|
earliestRefresh time.Time
|
||||||
|
refreshMutex sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func newChannelManager(log *logrus.Entry, sc *slack.Client) *channels {
|
||||||
|
return &channels{
|
||||||
|
log: log,
|
||||||
|
sc: sc,
|
||||||
|
channelsByID: make(map[string]*slack.Channel),
|
||||||
|
channelsByName: make(map[string]*slack.Channel),
|
||||||
|
earliestRefresh: time.Now(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *channels) getChannel(channel string) (*slack.Channel, error) {
|
||||||
|
if strings.HasPrefix(channel, "ID:") {
|
||||||
|
return b.getChannelByID(strings.TrimPrefix(channel, "ID:"))
|
||||||
|
}
|
||||||
|
return b.getChannelByName(channel)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *channels) getChannelByName(name string) (*slack.Channel, error) {
|
||||||
|
return b.getChannelBy(name, b.channelsByName)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *channels) getChannelByID(id string) (*slack.Channel, error) {
|
||||||
|
return b.getChannelBy(id, b.channelsByID)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *channels) getChannelBy(lookupKey string, lookupMap map[string]*slack.Channel) (*slack.Channel, error) {
|
||||||
|
b.channelsMutex.RLock()
|
||||||
|
defer b.channelsMutex.RUnlock()
|
||||||
|
|
||||||
|
if channel, ok := lookupMap[lookupKey]; ok {
|
||||||
|
return channel, nil
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("channel %s not found", lookupKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *channels) getChannelMembers(users *users) config.ChannelMembers {
|
||||||
|
b.channelMembersMutex.RLock()
|
||||||
|
defer b.channelMembersMutex.RUnlock()
|
||||||
|
|
||||||
|
membersInfo := config.ChannelMembers{}
|
||||||
|
for channelID, members := range b.channelMembers {
|
||||||
|
for _, member := range members {
|
||||||
|
channelName := ""
|
||||||
|
userName := ""
|
||||||
|
userNick := ""
|
||||||
|
user := users.getUser(member)
|
||||||
|
if user != nil {
|
||||||
|
userName = user.Name
|
||||||
|
userNick = user.Profile.DisplayName
|
||||||
|
}
|
||||||
|
channel, _ := b.getChannelByID(channelID)
|
||||||
|
if channel != nil {
|
||||||
|
channelName = channel.Name
|
||||||
|
}
|
||||||
|
memberInfo := config.ChannelMember{
|
||||||
|
Username: userName,
|
||||||
|
Nick: userNick,
|
||||||
|
UserID: member,
|
||||||
|
ChannelID: channelID,
|
||||||
|
ChannelName: channelName,
|
||||||
|
}
|
||||||
|
membersInfo = append(membersInfo, memberInfo)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return membersInfo
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *channels) registerChannel(channel slack.Channel) {
|
||||||
|
b.channelsMutex.Lock()
|
||||||
|
defer b.channelsMutex.Unlock()
|
||||||
|
|
||||||
|
b.channelsByID[channel.ID] = &channel
|
||||||
|
b.channelsByName[channel.Name] = &channel
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *channels) populateChannels(wait bool) {
|
||||||
|
b.refreshMutex.Lock()
|
||||||
|
if !wait && (time.Now().Before(b.earliestRefresh) || b.refreshInProgress) {
|
||||||
|
b.log.Debugf("Not refreshing channel list as it was done less than %v seconds ago.", minimumRefreshInterval)
|
||||||
|
b.refreshMutex.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for b.refreshInProgress {
|
||||||
|
b.refreshMutex.Unlock()
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
b.refreshMutex.Lock()
|
||||||
|
}
|
||||||
|
b.refreshInProgress = true
|
||||||
|
b.refreshMutex.Unlock()
|
||||||
|
|
||||||
|
newChannelsByID := map[string]*slack.Channel{}
|
||||||
|
newChannelsByName := map[string]*slack.Channel{}
|
||||||
|
newChannelMembers := make(map[string][]string)
|
||||||
|
|
||||||
|
// We only retrieve public and private channels, not IMs
|
||||||
|
// and MPIMs as those do not have a channel name.
|
||||||
|
queryParams := &slack.GetConversationsParameters{
|
||||||
|
ExcludeArchived: "true",
|
||||||
|
Types: []string{"public_channel,private_channel"},
|
||||||
|
}
|
||||||
|
for {
|
||||||
|
channels, nextCursor, err := b.sc.GetConversations(queryParams)
|
||||||
|
if err != nil {
|
||||||
|
if err = handleRateLimit(b.log, err); err != nil {
|
||||||
|
b.log.Errorf("Could not retrieve channels: %#v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range channels {
|
||||||
|
newChannelsByID[channels[i].ID] = &channels[i]
|
||||||
|
newChannelsByName[channels[i].Name] = &channels[i]
|
||||||
|
// also find all the members in every channel
|
||||||
|
// comment for now, issues on big slacks
|
||||||
|
/*
|
||||||
|
members, err := b.getUsersInConversation(channels[i].ID)
|
||||||
|
if err != nil {
|
||||||
|
if err = b.handleRateLimit(err); err != nil {
|
||||||
|
b.Log.Errorf("Could not retrieve channel members: %#v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
newChannelMembers[channels[i].ID] = members
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
if nextCursor == "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
queryParams.Cursor = nextCursor
|
||||||
|
}
|
||||||
|
|
||||||
|
b.channelsMutex.Lock()
|
||||||
|
defer b.channelsMutex.Unlock()
|
||||||
|
b.channelsByID = newChannelsByID
|
||||||
|
b.channelsByName = newChannelsByName
|
||||||
|
|
||||||
|
b.channelMembersMutex.Lock()
|
||||||
|
defer b.channelMembersMutex.Unlock()
|
||||||
|
b.channelMembers = newChannelMembers
|
||||||
|
|
||||||
|
b.refreshMutex.Lock()
|
||||||
|
defer b.refreshMutex.Unlock()
|
||||||
|
b.earliestRefresh = time.Now().Add(minimumRefreshInterval)
|
||||||
|
b.refreshInProgress = false
|
||||||
|
}
|
165
bridge/sshchat/sshchat.go
Normal file
@ -0,0 +1,165 @@
|
|||||||
|
package bsshchat
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"io"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/shazow/ssh-chat/sshd"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Bsshchat struct {
|
||||||
|
r *bufio.Scanner
|
||||||
|
w io.WriteCloser
|
||||||
|
*bridge.Config
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
return &Bsshchat{Config: cfg}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsshchat) Connect() error {
|
||||||
|
b.Log.Infof("Connecting %s", b.GetString("Server"))
|
||||||
|
|
||||||
|
// connHandler will be called by 'sshd.ConnectShell()' below
|
||||||
|
// once the connection is established in order to handle it.
|
||||||
|
connErr := make(chan error, 1) // Needs to be buffered.
|
||||||
|
connSignal := make(chan struct{})
|
||||||
|
connHandler := func(r io.Reader, w io.WriteCloser) error {
|
||||||
|
b.r = bufio.NewScanner(r)
|
||||||
|
b.r.Scan()
|
||||||
|
b.w = w
|
||||||
|
if _, err := b.w.Write([]byte("/theme mono\r\n/quiet\r\n")); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
close(connSignal) // Connection is established so we can signal the success.
|
||||||
|
return b.handleSSHChat()
|
||||||
|
}
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
// As a successful connection will result in this returning after the Connection
|
||||||
|
// method has already returned point we NEED to have a buffered channel to still
|
||||||
|
// be able to write.
|
||||||
|
connErr <- sshd.ConnectShell(b.GetString("Server"), b.GetString("Nick"), connHandler)
|
||||||
|
}()
|
||||||
|
|
||||||
|
select {
|
||||||
|
case err := <-connErr:
|
||||||
|
b.Log.Error("Connection failed")
|
||||||
|
return err
|
||||||
|
case <-connSignal:
|
||||||
|
}
|
||||||
|
b.Log.Info("Connection succeeded")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsshchat) Disconnect() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsshchat) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsshchat) Send(msg config.Message) (string, error) {
|
||||||
|
// ignore delete messages
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
if _, err := b.w.Write([]byte(rmsg.Username + rmsg.Text + "\r\n")); err != nil {
|
||||||
|
b.Log.Errorf("Could not send extra message: %#v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
return b.handleUploadFile(&msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_, err := b.w.Write([]byte(msg.Username + msg.Text + "\r\n"))
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
func (b *Bsshchat) sshchatKeepAlive() chan bool {
|
||||||
|
done := make(chan bool)
|
||||||
|
go func() {
|
||||||
|
ticker := time.NewTicker(90 * time.Second)
|
||||||
|
defer ticker.Stop()
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-ticker.C:
|
||||||
|
b.Log.Debugf("PING")
|
||||||
|
err := b.xc.PingC2S("", "")
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Debugf("PING failed %#v", err)
|
||||||
|
}
|
||||||
|
case <-done:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
return done
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
func stripPrompt(s string) string {
|
||||||
|
pos := strings.LastIndex(s, "\033[K")
|
||||||
|
if pos < 0 {
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
return s[pos+3:]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsshchat) handleSSHChat() error {
|
||||||
|
/*
|
||||||
|
done := b.sshchatKeepAlive()
|
||||||
|
defer close(done)
|
||||||
|
*/
|
||||||
|
wait := true
|
||||||
|
for {
|
||||||
|
if b.r.Scan() {
|
||||||
|
// ignore messages from ourselves
|
||||||
|
if !strings.Contains(b.r.Text(), "\033[K") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if strings.Contains(b.r.Text(), "Rate limiting is in effect") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
res := strings.Split(stripPrompt(b.r.Text()), ":")
|
||||||
|
if res[0] == "-> Set theme" {
|
||||||
|
wait = false
|
||||||
|
b.Log.Debugf("mono found, allowing")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if !wait {
|
||||||
|
b.Log.Debugf("<= Message %#v", res)
|
||||||
|
rmsg := config.Message{Username: res[0], Text: strings.TrimSpace(strings.Join(res[1:], ":")), Channel: "sshchat", Account: b.Account, UserID: "nick"}
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsshchat) handleUploadFile(msg *config.Message) (string, error) {
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text += fi.Comment + ": "
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text = fi.URL
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text = fi.Comment + ": " + fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if _, err := b.w.Write([]byte(msg.Username + msg.Text + "\r\n")); err != nil {
|
||||||
|
b.Log.Errorf("Could not send file message: %#v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
126
bridge/steam/handlers.go
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
package bsteam
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/Philipp15b/go-steam"
|
||||||
|
"github.com/Philipp15b/go-steam/protocol/steamlang"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Bsteam) handleChatMsg(e *steam.ChatMsgEvent) {
|
||||||
|
b.Log.Debugf("Receiving ChatMsgEvent: %#v", e)
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", b.getNick(e.ChatterId), b.Account)
|
||||||
|
var channel int64
|
||||||
|
if e.ChatRoomId == 0 {
|
||||||
|
channel = int64(e.ChatterId)
|
||||||
|
} else {
|
||||||
|
// for some reason we have to remove 0x18000000000000
|
||||||
|
// TODO
|
||||||
|
// https://github.com/42wim/matterbridge/pull/630#discussion_r238102751
|
||||||
|
// channel = int64(e.ChatRoomId) & 0xfffffffffffff
|
||||||
|
channel = int64(e.ChatRoomId) - 0x18000000000000
|
||||||
|
}
|
||||||
|
msg := config.Message{
|
||||||
|
Username: b.getNick(e.ChatterId),
|
||||||
|
Text: e.Message,
|
||||||
|
Channel: strconv.FormatInt(channel, 10),
|
||||||
|
Account: b.Account,
|
||||||
|
UserID: strconv.FormatInt(int64(e.ChatterId), 10),
|
||||||
|
}
|
||||||
|
b.Remote <- msg
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsteam) handleEvents() {
|
||||||
|
myLoginInfo := &steam.LogOnDetails{
|
||||||
|
Username: b.GetString("Login"),
|
||||||
|
Password: b.GetString("Password"),
|
||||||
|
AuthCode: b.GetString("AuthCode"),
|
||||||
|
}
|
||||||
|
// TODO Attempt to read existing auth hash to avoid steam guard.
|
||||||
|
// Maybe works
|
||||||
|
//myLoginInfo.SentryFileHash, _ = ioutil.ReadFile("sentry")
|
||||||
|
for event := range b.c.Events() {
|
||||||
|
switch e := event.(type) {
|
||||||
|
case *steam.ChatMsgEvent:
|
||||||
|
b.handleChatMsg(e)
|
||||||
|
case *steam.PersonaStateEvent:
|
||||||
|
b.Log.Debugf("PersonaStateEvent: %#v\n", e)
|
||||||
|
b.Lock()
|
||||||
|
b.userMap[e.FriendId] = e.Name
|
||||||
|
b.Unlock()
|
||||||
|
case *steam.ConnectedEvent:
|
||||||
|
b.c.Auth.LogOn(myLoginInfo)
|
||||||
|
case *steam.MachineAuthUpdateEvent:
|
||||||
|
// TODO sentry files for 2 auth
|
||||||
|
/*
|
||||||
|
b.Log.Info("authupdate", e)
|
||||||
|
b.Log.Info("hash", e.Hash)
|
||||||
|
ioutil.WriteFile("sentry", e.Hash, 0666)
|
||||||
|
*/
|
||||||
|
case *steam.LogOnFailedEvent:
|
||||||
|
b.Log.Info("Logon failed", e)
|
||||||
|
err := b.handleLogOnFailed(e, myLoginInfo)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Error(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
case *steam.LoggedOnEvent:
|
||||||
|
b.Log.Debugf("LoggedOnEvent: %#v", e)
|
||||||
|
b.connected <- struct{}{}
|
||||||
|
b.Log.Debugf("setting online")
|
||||||
|
b.c.Social.SetPersonaState(steamlang.EPersonaState_Online)
|
||||||
|
case *steam.DisconnectedEvent:
|
||||||
|
b.Log.Info("Disconnected")
|
||||||
|
b.Log.Info("Attempting to reconnect...")
|
||||||
|
b.c.Connect()
|
||||||
|
case steam.FatalErrorEvent:
|
||||||
|
b.Log.Errorf("steam FatalErrorEvent: %#v", e)
|
||||||
|
default:
|
||||||
|
b.Log.Debugf("unknown event %#v", e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsteam) handleLogOnFailed(e *steam.LogOnFailedEvent, myLoginInfo *steam.LogOnDetails) error {
|
||||||
|
switch e.Result {
|
||||||
|
case steamlang.EResult_AccountLogonDeniedNeedTwoFactorCode:
|
||||||
|
b.Log.Info("Steam guard isn't letting me in! Enter 2FA code:")
|
||||||
|
var code string
|
||||||
|
fmt.Scanf("%s", &code)
|
||||||
|
// TODO https://github.com/42wim/matterbridge/pull/630#discussion_r238103978
|
||||||
|
myLoginInfo.TwoFactorCode = code
|
||||||
|
case steamlang.EResult_AccountLogonDenied:
|
||||||
|
b.Log.Info("Steam guard isn't letting me in! Enter auth code:")
|
||||||
|
var code string
|
||||||
|
fmt.Scanf("%s", &code)
|
||||||
|
// TODO https://github.com/42wim/matterbridge/pull/630#discussion_r238103978
|
||||||
|
myLoginInfo.AuthCode = code
|
||||||
|
case steamlang.EResult_InvalidLoginAuthCode:
|
||||||
|
return fmt.Errorf("Steam guard: invalid login auth code: %#v ", e.Result)
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("LogOnFailedEvent: %#v ", e.Result)
|
||||||
|
// TODO: Handle EResult_InvalidLoginAuthCode
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleFileInfo handles config.FileInfo and adds correct file comment or URL to msg.Text.
|
||||||
|
// Returns error if cast fails.
|
||||||
|
func (b *Bsteam) handleFileInfo(msg *config.Message, f interface{}) error {
|
||||||
|
if _, ok := f.(config.FileInfo); !ok {
|
||||||
|
return fmt.Errorf("handleFileInfo cast failed %#v", f)
|
||||||
|
}
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text += fi.Comment + ": "
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text = fi.URL
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text = fi.Comment + ": " + fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
95
bridge/steam/steam.go
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
package bsteam
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/Philipp15b/go-steam"
|
||||||
|
"github.com/Philipp15b/go-steam/protocol/steamlang"
|
||||||
|
"github.com/Philipp15b/go-steam/steamid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Bsteam struct {
|
||||||
|
c *steam.Client
|
||||||
|
connected chan struct{}
|
||||||
|
userMap map[steamid.SteamId]string
|
||||||
|
sync.RWMutex
|
||||||
|
*bridge.Config
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
b := &Bsteam{Config: cfg}
|
||||||
|
b.userMap = make(map[steamid.SteamId]string)
|
||||||
|
b.connected = make(chan struct{})
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsteam) Connect() error {
|
||||||
|
b.Log.Info("Connecting")
|
||||||
|
b.c = steam.NewClient()
|
||||||
|
go b.handleEvents()
|
||||||
|
go b.c.Connect()
|
||||||
|
select {
|
||||||
|
case <-b.connected:
|
||||||
|
b.Log.Info("Connection succeeded")
|
||||||
|
case <-time.After(time.Second * 30):
|
||||||
|
return fmt.Errorf("connection timed out")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsteam) Disconnect() error {
|
||||||
|
b.c.Disconnect()
|
||||||
|
return nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsteam) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
id, err := steamid.NewId(channel.Name)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.c.Social.JoinChat(id)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsteam) Send(msg config.Message) (string, error) {
|
||||||
|
// ignore delete messages
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
id, err := steamid.NewId(msg.Channel)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle files
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
b.c.Social.SendMessage(id, steamlang.EChatEntryType_ChatMsg, rmsg.Username+rmsg.Text)
|
||||||
|
}
|
||||||
|
for i := range msg.Extra["file"] {
|
||||||
|
if err := b.handleFileInfo(&msg, msg.Extra["file"][i]); err != nil {
|
||||||
|
b.Log.Error(err)
|
||||||
|
}
|
||||||
|
b.c.Social.SendMessage(id, steamlang.EChatEntryType_ChatMsg, msg.Username+msg.Text)
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
b.c.Social.SendMessage(id, steamlang.EChatEntryType_ChatMsg, msg.Username+msg.Text)
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsteam) getNick(id steamid.SteamId) string {
|
||||||
|
b.RLock()
|
||||||
|
defer b.RUnlock()
|
||||||
|
if name, ok := b.userMap[id]; ok {
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
return "unknown"
|
||||||
|
}
|
377
bridge/telegram/handlers.go
Normal file
@ -0,0 +1,377 @@
|
|||||||
|
package btelegram
|
||||||
|
|
||||||
|
import (
|
||||||
|
"html"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/go-telegram-bot-api/telegram-bot-api"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Btelegram) handleUpdate(rmsg *config.Message, message, posted, edited *tgbotapi.Message) *tgbotapi.Message {
|
||||||
|
// handle channels
|
||||||
|
if posted != nil {
|
||||||
|
message = posted
|
||||||
|
rmsg.Text = message.Text
|
||||||
|
}
|
||||||
|
|
||||||
|
// edited channel message
|
||||||
|
if edited != nil && !b.GetBool("EditDisable") {
|
||||||
|
message = edited
|
||||||
|
rmsg.Text = rmsg.Text + message.Text + b.GetString("EditSuffix")
|
||||||
|
}
|
||||||
|
return message
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleChannels checks if it's a channel message and if the message is a new or edited messages
|
||||||
|
func (b *Btelegram) handleChannels(rmsg *config.Message, message *tgbotapi.Message, update tgbotapi.Update) *tgbotapi.Message {
|
||||||
|
return b.handleUpdate(rmsg, message, update.ChannelPost, update.EditedChannelPost)
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleGroups checks if it's a group message and if the message is a new or edited messages
|
||||||
|
func (b *Btelegram) handleGroups(rmsg *config.Message, message *tgbotapi.Message, update tgbotapi.Update) *tgbotapi.Message {
|
||||||
|
return b.handleUpdate(rmsg, message, update.Message, update.EditedMessage)
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleForwarded handles forwarded messages
|
||||||
|
func (b *Btelegram) handleForwarded(rmsg *config.Message, message *tgbotapi.Message) {
|
||||||
|
if message.ForwardFrom != nil {
|
||||||
|
usernameForward := ""
|
||||||
|
if b.GetBool("UseFirstName") {
|
||||||
|
usernameForward = message.ForwardFrom.FirstName
|
||||||
|
}
|
||||||
|
if usernameForward == "" {
|
||||||
|
usernameForward = message.ForwardFrom.UserName
|
||||||
|
if usernameForward == "" {
|
||||||
|
usernameForward = message.ForwardFrom.FirstName
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if usernameForward == "" {
|
||||||
|
usernameForward = unknownUser
|
||||||
|
}
|
||||||
|
rmsg.Text = "Forwarded from " + usernameForward + ": " + rmsg.Text
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleQuoting handles quoting of previous messages
|
||||||
|
func (b *Btelegram) handleQuoting(rmsg *config.Message, message *tgbotapi.Message) {
|
||||||
|
if message.ReplyToMessage != nil {
|
||||||
|
usernameReply := ""
|
||||||
|
if message.ReplyToMessage.From != nil {
|
||||||
|
if b.GetBool("UseFirstName") {
|
||||||
|
usernameReply = message.ReplyToMessage.From.FirstName
|
||||||
|
}
|
||||||
|
if usernameReply == "" {
|
||||||
|
usernameReply = message.ReplyToMessage.From.UserName
|
||||||
|
if usernameReply == "" {
|
||||||
|
usernameReply = message.ReplyToMessage.From.FirstName
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if usernameReply == "" {
|
||||||
|
usernameReply = unknownUser
|
||||||
|
}
|
||||||
|
if !b.GetBool("QuoteDisable") {
|
||||||
|
rmsg.Text = b.handleQuote(rmsg.Text, usernameReply, message.ReplyToMessage.Text)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleUsername handles the correct setting of the username
|
||||||
|
func (b *Btelegram) handleUsername(rmsg *config.Message, message *tgbotapi.Message) {
|
||||||
|
if message.From != nil {
|
||||||
|
rmsg.UserID = strconv.Itoa(message.From.ID)
|
||||||
|
if b.GetBool("UseFirstName") {
|
||||||
|
rmsg.Username = message.From.FirstName
|
||||||
|
}
|
||||||
|
if rmsg.Username == "" {
|
||||||
|
rmsg.Username = message.From.UserName
|
||||||
|
if rmsg.Username == "" {
|
||||||
|
rmsg.Username = message.From.FirstName
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// only download avatars if we have a place to upload them (configured mediaserver)
|
||||||
|
if b.General.MediaServerUpload != "" {
|
||||||
|
b.handleDownloadAvatar(message.From.ID, rmsg.Channel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// if we really didn't find a username, set it to unknown
|
||||||
|
if rmsg.Username == "" {
|
||||||
|
rmsg.Username = unknownUser
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Btelegram) handleRecv(updates <-chan tgbotapi.Update) {
|
||||||
|
for update := range updates {
|
||||||
|
b.Log.Debugf("== Receiving event: %#v", update.Message)
|
||||||
|
|
||||||
|
if update.Message == nil && update.ChannelPost == nil &&
|
||||||
|
update.EditedMessage == nil && update.EditedChannelPost == nil {
|
||||||
|
b.Log.Error("Getting nil messages, this shouldn't happen.")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
var message *tgbotapi.Message
|
||||||
|
|
||||||
|
rmsg := config.Message{Account: b.Account, Extra: make(map[string][]interface{})}
|
||||||
|
|
||||||
|
// handle channels
|
||||||
|
message = b.handleChannels(&rmsg, message, update)
|
||||||
|
|
||||||
|
// handle groups
|
||||||
|
message = b.handleGroups(&rmsg, message, update)
|
||||||
|
|
||||||
|
// set the ID's from the channel or group message
|
||||||
|
rmsg.ID = strconv.Itoa(message.MessageID)
|
||||||
|
rmsg.Channel = strconv.FormatInt(message.Chat.ID, 10)
|
||||||
|
|
||||||
|
// handle username
|
||||||
|
b.handleUsername(&rmsg, message)
|
||||||
|
|
||||||
|
// handle any downloads
|
||||||
|
err := b.handleDownload(&rmsg, message)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("download failed: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle forwarded messages
|
||||||
|
b.handleForwarded(&rmsg, message)
|
||||||
|
|
||||||
|
// quote the previous message
|
||||||
|
b.handleQuoting(&rmsg, message)
|
||||||
|
|
||||||
|
// handle entities (adding URLs)
|
||||||
|
b.handleEntities(&rmsg, message)
|
||||||
|
|
||||||
|
if rmsg.Text != "" || len(rmsg.Extra) > 0 {
|
||||||
|
rmsg.Text = helper.RemoveEmptyNewLines(rmsg.Text)
|
||||||
|
// channels don't have (always?) user information. see #410
|
||||||
|
if message.From != nil {
|
||||||
|
rmsg.Avatar = helper.GetAvatar(b.avatarMap, strconv.Itoa(message.From.ID), b.General)
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", rmsg.Username, b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleDownloadAvatar downloads the avatar of userid from channel
|
||||||
|
// sends a EVENT_AVATAR_DOWNLOAD message to the gateway if successful.
|
||||||
|
// logs an error message if it fails
|
||||||
|
func (b *Btelegram) handleDownloadAvatar(userid int, channel string) {
|
||||||
|
rmsg := config.Message{Username: "system",
|
||||||
|
Text: "avatar",
|
||||||
|
Channel: channel,
|
||||||
|
Account: b.Account,
|
||||||
|
UserID: strconv.Itoa(userid),
|
||||||
|
Event: config.EventAvatarDownload,
|
||||||
|
Extra: make(map[string][]interface{})}
|
||||||
|
|
||||||
|
if _, ok := b.avatarMap[strconv.Itoa(userid)]; !ok {
|
||||||
|
photos, err := b.c.GetUserProfilePhotos(tgbotapi.UserProfilePhotosConfig{UserID: userid, Limit: 1})
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("Userprofile download failed for %#v %s", userid, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(photos.Photos) > 0 {
|
||||||
|
photo := photos.Photos[0][0]
|
||||||
|
url := b.getFileDirectURL(photo.FileID)
|
||||||
|
name := strconv.Itoa(userid) + ".png"
|
||||||
|
b.Log.Debugf("trying to download %#v fileid %#v with size %#v", name, photo.FileID, photo.FileSize)
|
||||||
|
|
||||||
|
err := helper.HandleDownloadSize(b.Log, &rmsg, name, int64(photo.FileSize), b.General)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Error(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
data, err := helper.DownloadFile(url)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("download %s failed %#v", url, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
helper.HandleDownloadData(b.Log, &rmsg, name, rmsg.Text, "", data, b.General)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleDownloadFile handles file download
|
||||||
|
func (b *Btelegram) handleDownload(rmsg *config.Message, message *tgbotapi.Message) error {
|
||||||
|
size := 0
|
||||||
|
var url, name, text string
|
||||||
|
switch {
|
||||||
|
case message.Sticker != nil:
|
||||||
|
text, name, url = b.getDownloadInfo(message.Sticker.FileID, ".webp", true)
|
||||||
|
size = message.Sticker.FileSize
|
||||||
|
case message.Voice != nil:
|
||||||
|
text, name, url = b.getDownloadInfo(message.Voice.FileID, ".ogg", true)
|
||||||
|
size = message.Voice.FileSize
|
||||||
|
case message.Video != nil:
|
||||||
|
text, name, url = b.getDownloadInfo(message.Video.FileID, "", true)
|
||||||
|
size = message.Video.FileSize
|
||||||
|
case message.Audio != nil:
|
||||||
|
text, name, url = b.getDownloadInfo(message.Audio.FileID, "", true)
|
||||||
|
size = message.Audio.FileSize
|
||||||
|
case message.Document != nil:
|
||||||
|
_, _, url = b.getDownloadInfo(message.Document.FileID, "", false)
|
||||||
|
size = message.Document.FileSize
|
||||||
|
name = message.Document.FileName
|
||||||
|
text = " " + message.Document.FileName + " : " + url
|
||||||
|
case message.Photo != nil:
|
||||||
|
photos := *message.Photo
|
||||||
|
size = photos[len(photos)-1].FileSize
|
||||||
|
text, name, url = b.getDownloadInfo(photos[len(photos)-1].FileID, "", true)
|
||||||
|
}
|
||||||
|
|
||||||
|
// if name is empty we didn't match a thing to download
|
||||||
|
if name == "" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
// use the URL instead of native upload
|
||||||
|
if b.GetBool("UseInsecureURL") {
|
||||||
|
b.Log.Debugf("Setting message text to :%s", text)
|
||||||
|
rmsg.Text += text
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
// if we have a file attached, download it (in memory) and put a pointer to it in msg.Extra
|
||||||
|
err := helper.HandleDownloadSize(b.Log, rmsg, name, int64(size), b.General)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
data, err := helper.DownloadFile(url)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if strings.HasSuffix(name, ".webp") && b.GetBool("MediaConvertWebPToPNG") {
|
||||||
|
b.Log.Debugf("WebP to PNG conversion enabled, converting %s", name)
|
||||||
|
err := helper.ConvertWebPToPNG(data)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("conversion failed: %s", err)
|
||||||
|
} else {
|
||||||
|
name = strings.Replace(name, ".webp", ".png", 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
helper.HandleDownloadData(b.Log, rmsg, name, message.Caption, "", data, b.General)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Btelegram) getDownloadInfo(id string, suffix string, urlpart bool) (string, string, string) {
|
||||||
|
url := b.getFileDirectURL(id)
|
||||||
|
name := ""
|
||||||
|
if urlpart {
|
||||||
|
urlPart := strings.Split(url, "/")
|
||||||
|
name = urlPart[len(urlPart)-1]
|
||||||
|
}
|
||||||
|
if suffix != "" && !strings.HasSuffix(name, suffix) {
|
||||||
|
name += suffix
|
||||||
|
}
|
||||||
|
text := " " + url
|
||||||
|
return text, name, url
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleDelete handles message deleting
|
||||||
|
func (b *Btelegram) handleDelete(msg *config.Message, chatid int64) (string, error) {
|
||||||
|
if msg.ID == "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
msgid, err := strconv.Atoi(msg.ID)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
_, err = b.c.DeleteMessage(tgbotapi.DeleteMessageConfig{ChatID: chatid, MessageID: msgid})
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleEdit handles message editing.
|
||||||
|
func (b *Btelegram) handleEdit(msg *config.Message, chatid int64) (string, error) {
|
||||||
|
msgid, err := strconv.Atoi(msg.ID)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if strings.ToLower(b.GetString("MessageFormat")) == HTMLNick {
|
||||||
|
b.Log.Debug("Using mode HTML - nick only")
|
||||||
|
msg.Text = html.EscapeString(msg.Text)
|
||||||
|
}
|
||||||
|
m := tgbotapi.NewEditMessageText(chatid, msgid, msg.Username+msg.Text)
|
||||||
|
switch b.GetString("MessageFormat") {
|
||||||
|
case HTMLFormat:
|
||||||
|
b.Log.Debug("Using mode HTML")
|
||||||
|
m.ParseMode = tgbotapi.ModeHTML
|
||||||
|
case "Markdown":
|
||||||
|
b.Log.Debug("Using mode markdown")
|
||||||
|
m.ParseMode = tgbotapi.ModeMarkdown
|
||||||
|
}
|
||||||
|
if strings.ToLower(b.GetString("MessageFormat")) == HTMLNick {
|
||||||
|
b.Log.Debug("Using mode HTML - nick only")
|
||||||
|
m.ParseMode = tgbotapi.ModeHTML
|
||||||
|
}
|
||||||
|
_, err = b.c.Send(m)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleUploadFile handles native upload of files
|
||||||
|
func (b *Btelegram) handleUploadFile(msg *config.Message, chatid int64) string {
|
||||||
|
var c tgbotapi.Chattable
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
file := tgbotapi.FileBytes{
|
||||||
|
Name: fi.Name,
|
||||||
|
Bytes: *fi.Data,
|
||||||
|
}
|
||||||
|
re := regexp.MustCompile(".(jpg|png)$")
|
||||||
|
if re.MatchString(fi.Name) {
|
||||||
|
c = tgbotapi.NewPhotoUpload(chatid, file)
|
||||||
|
} else {
|
||||||
|
c = tgbotapi.NewDocumentUpload(chatid, file)
|
||||||
|
}
|
||||||
|
_, err := b.c.Send(c)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("file upload failed: %#v", err)
|
||||||
|
}
|
||||||
|
if fi.Comment != "" {
|
||||||
|
if _, err := b.sendMessage(chatid, msg.Username, fi.Comment); err != nil {
|
||||||
|
b.Log.Errorf("posting file comment %s failed: %s", fi.Comment, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Btelegram) handleQuote(message, quoteNick, quoteMessage string) string {
|
||||||
|
format := b.GetString("quoteformat")
|
||||||
|
if format == "" {
|
||||||
|
format = "{MESSAGE} (re @{QUOTENICK}: {QUOTEMESSAGE})"
|
||||||
|
}
|
||||||
|
format = strings.Replace(format, "{MESSAGE}", message, -1)
|
||||||
|
format = strings.Replace(format, "{QUOTENICK}", quoteNick, -1)
|
||||||
|
format = strings.Replace(format, "{QUOTEMESSAGE}", quoteMessage, -1)
|
||||||
|
return format
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleEntities handles messageEntities
|
||||||
|
func (b *Btelegram) handleEntities(rmsg *config.Message, message *tgbotapi.Message) {
|
||||||
|
if message.Entities == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// for now only do URL replacements
|
||||||
|
for _, e := range *message.Entities {
|
||||||
|
if e.Type == "text_link" {
|
||||||
|
url, err := e.ParseURL()
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("entity text_link url parse failed: %s", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
link := rmsg.Text[e.Offset : e.Offset+e.Length]
|
||||||
|
rmsg.Text = strings.Replace(rmsg.Text, link, url.String(), 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
65
bridge/telegram/html.go
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
package btelegram
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"html"
|
||||||
|
|
||||||
|
"github.com/russross/blackfriday"
|
||||||
|
)
|
||||||
|
|
||||||
|
type customHTML struct {
|
||||||
|
blackfriday.Renderer
|
||||||
|
}
|
||||||
|
|
||||||
|
func (options *customHTML) Paragraph(out *bytes.Buffer, text func() bool) {
|
||||||
|
marker := out.Len()
|
||||||
|
|
||||||
|
if !text() {
|
||||||
|
out.Truncate(marker)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
out.WriteString("\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (options *customHTML) BlockCode(out *bytes.Buffer, text []byte, lang string) {
|
||||||
|
out.WriteString("<pre>")
|
||||||
|
|
||||||
|
out.WriteString(html.EscapeString(string(text)))
|
||||||
|
out.WriteString("</pre>\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (options *customHTML) Header(out *bytes.Buffer, text func() bool, level int, id string) {
|
||||||
|
options.Paragraph(out, text)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (options *customHTML) HRule(out *bytes.Buffer) {
|
||||||
|
out.WriteByte('\n') //nolint:errcheck
|
||||||
|
}
|
||||||
|
|
||||||
|
func (options *customHTML) BlockQuote(out *bytes.Buffer, text []byte) {
|
||||||
|
out.WriteString("> ")
|
||||||
|
out.Write(text)
|
||||||
|
out.WriteByte('\n')
|
||||||
|
}
|
||||||
|
|
||||||
|
func (options *customHTML) List(out *bytes.Buffer, text func() bool, flags int) {
|
||||||
|
options.Paragraph(out, text)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (options *customHTML) ListItem(out *bytes.Buffer, text []byte, flags int) {
|
||||||
|
out.WriteString("- ")
|
||||||
|
out.Write(text)
|
||||||
|
out.WriteByte('\n')
|
||||||
|
}
|
||||||
|
|
||||||
|
func makeHTML(input string) string {
|
||||||
|
return string(blackfriday.Markdown([]byte(input),
|
||||||
|
&customHTML{blackfriday.HtmlRenderer(blackfriday.HTML_USE_XHTML|blackfriday.HTML_SKIP_IMAGES, "", "")},
|
||||||
|
blackfriday.EXTENSION_NO_INTRA_EMPHASIS|
|
||||||
|
blackfriday.EXTENSION_FENCED_CODE|
|
||||||
|
blackfriday.EXTENSION_AUTOLINK|
|
||||||
|
blackfriday.EXTENSION_SPACE_HEADERS|
|
||||||
|
blackfriday.EXTENSION_HEADER_IDS|
|
||||||
|
blackfriday.EXTENSION_BACKSLASH_LINE_BREAK|
|
||||||
|
blackfriday.EXTENSION_DEFINITION_LISTS))
|
||||||
|
}
|
@ -1,161 +1,143 @@
|
|||||||
package btelegram
|
package btelegram
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"html"
|
"html"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
log "github.com/Sirupsen/logrus"
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
"github.com/go-telegram-bot-api/telegram-bot-api"
|
"github.com/go-telegram-bot-api/telegram-bot-api"
|
||||||
"github.com/russross/blackfriday"
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
unknownUser = "unknown"
|
||||||
|
HTMLFormat = "HTML"
|
||||||
|
HTMLNick = "htmlnick"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Btelegram struct {
|
type Btelegram struct {
|
||||||
c *tgbotapi.BotAPI
|
c *tgbotapi.BotAPI
|
||||||
Config *config.Protocol
|
*bridge.Config
|
||||||
Remote chan config.Message
|
avatarMap map[string]string // keep cache of userid and avatar sha
|
||||||
Account string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var flog *log.Entry
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
var protocol = "telegram"
|
return &Btelegram{Config: cfg, avatarMap: make(map[string]string)}
|
||||||
|
|
||||||
func init() {
|
|
||||||
flog = log.WithFields(log.Fields{"module": protocol})
|
|
||||||
}
|
|
||||||
|
|
||||||
func New(cfg config.Protocol, account string, c chan config.Message) *Btelegram {
|
|
||||||
b := &Btelegram{}
|
|
||||||
b.Config = &cfg
|
|
||||||
b.Remote = c
|
|
||||||
b.Account = account
|
|
||||||
return b
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Btelegram) Connect() error {
|
func (b *Btelegram) Connect() error {
|
||||||
var err error
|
var err error
|
||||||
flog.Info("Connecting")
|
b.Log.Info("Connecting")
|
||||||
b.c, err = tgbotapi.NewBotAPI(b.Config.Token)
|
b.c, err = tgbotapi.NewBotAPI(b.GetString("Token"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
b.Log.Debugf("%#v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
updates, err := b.c.GetUpdatesChan(tgbotapi.NewUpdate(0))
|
u := tgbotapi.NewUpdate(0)
|
||||||
|
u.Timeout = 60
|
||||||
|
updates, err := b.c.GetUpdatesChan(u)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
b.Log.Debugf("%#v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
flog.Info("Connection succeeded")
|
b.Log.Info("Connection succeeded")
|
||||||
go b.handleRecv(updates)
|
go b.handleRecv(updates)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Btelegram) JoinChannel(channel string) error {
|
func (b *Btelegram) Disconnect() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type customHtml struct {
|
func (b *Btelegram) JoinChannel(channel config.ChannelInfo) error {
|
||||||
blackfriday.Renderer
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (options *customHtml) Paragraph(out *bytes.Buffer, text func() bool) {
|
func (b *Btelegram) Send(msg config.Message) (string, error) {
|
||||||
marker := out.Len()
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
|
||||||
if !text() {
|
// get the chatid
|
||||||
out.Truncate(marker)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
out.WriteString("\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (options *customHtml) BlockCode(out *bytes.Buffer, text []byte, lang string) {
|
|
||||||
out.WriteString("<pre>")
|
|
||||||
|
|
||||||
out.WriteString(html.EscapeString(string(text)))
|
|
||||||
out.WriteString("</pre>\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (options *customHtml) Header(out *bytes.Buffer, text func() bool, level int, id string) {
|
|
||||||
options.Paragraph(out, text)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (options *customHtml) HRule(out *bytes.Buffer) {
|
|
||||||
out.WriteByte('\n')
|
|
||||||
}
|
|
||||||
|
|
||||||
func (options *customHtml) BlockQuote(out *bytes.Buffer, text []byte) {
|
|
||||||
out.WriteString("> ")
|
|
||||||
out.Write(text)
|
|
||||||
out.WriteByte('\n')
|
|
||||||
}
|
|
||||||
|
|
||||||
func (options *customHtml) List(out *bytes.Buffer, text func() bool, flags int) {
|
|
||||||
options.Paragraph(out, text)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (options *customHtml) ListItem(out *bytes.Buffer, text []byte, flags int) {
|
|
||||||
out.WriteString("- ")
|
|
||||||
out.Write(text)
|
|
||||||
out.WriteByte('\n')
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Btelegram) Send(msg config.Message) error {
|
|
||||||
flog.Debugf("Receiving %#v", msg)
|
|
||||||
chatid, err := strconv.ParseInt(msg.Channel, 10, 64)
|
chatid, err := strconv.ParseInt(msg.Channel, 10, 64)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
parsed := blackfriday.Markdown([]byte(msg.Text),
|
// map the file SHA to our user (caches the avatar)
|
||||||
&customHtml{blackfriday.HtmlRenderer(blackfriday.HTML_USE_XHTML|blackfriday.HTML_SKIP_IMAGES, "", "")},
|
if msg.Event == config.EventAvatarDownload {
|
||||||
blackfriday.EXTENSION_NO_INTRA_EMPHASIS|
|
return b.cacheAvatar(&msg)
|
||||||
blackfriday.EXTENSION_FENCED_CODE|
|
}
|
||||||
blackfriday.EXTENSION_AUTOLINK|
|
|
||||||
blackfriday.EXTENSION_SPACE_HEADERS|
|
|
||||||
blackfriday.EXTENSION_HEADER_IDS|
|
|
||||||
blackfriday.EXTENSION_BACKSLASH_LINE_BREAK|
|
|
||||||
blackfriday.EXTENSION_DEFINITION_LISTS)
|
|
||||||
|
|
||||||
m := tgbotapi.NewMessage(chatid, html.EscapeString(msg.Username)+string(parsed))
|
if b.GetString("MessageFormat") == HTMLFormat {
|
||||||
m.ParseMode = "HTML"
|
msg.Text = makeHTML(msg.Text)
|
||||||
_, err = b.c.Send(m)
|
}
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Btelegram) handleRecv(updates <-chan tgbotapi.Update) {
|
// Delete message
|
||||||
username := ""
|
if msg.Event == config.EventMsgDelete {
|
||||||
text := ""
|
return b.handleDelete(&msg, chatid)
|
||||||
channel := ""
|
}
|
||||||
for update := range updates {
|
|
||||||
// handle channels
|
// Upload a file if it exists
|
||||||
if update.ChannelPost != nil {
|
if msg.Extra != nil {
|
||||||
if update.ChannelPost.From != nil {
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
username = update.ChannelPost.From.FirstName
|
if _, err := b.sendMessage(chatid, rmsg.Username, rmsg.Text); err != nil {
|
||||||
if username == "" {
|
b.Log.Errorf("sendMessage failed: %s", err)
|
||||||
username = update.ChannelPost.From.UserName
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
text = update.ChannelPost.Text
|
|
||||||
channel = strconv.FormatInt(update.ChannelPost.Chat.ID, 10)
|
|
||||||
}
|
}
|
||||||
// handle groups
|
// check if we have files to upload (from slack, telegram or mattermost)
|
||||||
if update.Message != nil {
|
if len(msg.Extra["file"]) > 0 {
|
||||||
if update.Message.From != nil {
|
b.handleUploadFile(&msg, chatid)
|
||||||
username = update.Message.From.FirstName
|
|
||||||
if username == "" {
|
|
||||||
username = update.Message.From.UserName
|
|
||||||
}
|
|
||||||
}
|
|
||||||
text = update.Message.Text
|
|
||||||
channel = strconv.FormatInt(update.Message.Chat.ID, 10)
|
|
||||||
}
|
|
||||||
if username == "" {
|
|
||||||
username = "unknown"
|
|
||||||
}
|
|
||||||
if text != "" {
|
|
||||||
flog.Debugf("Sending message from %s on %s to gateway", username, b.Account)
|
|
||||||
b.Remote <- config.Message{Username: username, Text: text, Channel: channel, Account: b.Account}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// edit the message if we have a msg ID
|
||||||
|
if msg.ID != "" {
|
||||||
|
return b.handleEdit(&msg, chatid)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Post normal message
|
||||||
|
return b.sendMessage(chatid, msg.Username, msg.Text)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Btelegram) getFileDirectURL(id string) string {
|
||||||
|
res, err := b.c.GetFileDirectURL(id)
|
||||||
|
if err != nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Btelegram) sendMessage(chatid int64, username, text string) (string, error) {
|
||||||
|
m := tgbotapi.NewMessage(chatid, "")
|
||||||
|
m.Text = username + text
|
||||||
|
if b.GetString("MessageFormat") == HTMLFormat {
|
||||||
|
b.Log.Debug("Using mode HTML")
|
||||||
|
m.ParseMode = tgbotapi.ModeHTML
|
||||||
|
}
|
||||||
|
if b.GetString("MessageFormat") == "Markdown" {
|
||||||
|
b.Log.Debug("Using mode markdown")
|
||||||
|
m.ParseMode = tgbotapi.ModeMarkdown
|
||||||
|
}
|
||||||
|
if strings.ToLower(b.GetString("MessageFormat")) == HTMLNick {
|
||||||
|
b.Log.Debug("Using mode HTML - nick only")
|
||||||
|
m.Text = username + html.EscapeString(text)
|
||||||
|
m.ParseMode = tgbotapi.ModeHTML
|
||||||
|
}
|
||||||
|
res, err := b.c.Send(m)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return strconv.Itoa(res.MessageID), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Btelegram) cacheAvatar(msg *config.Message) (string, error) {
|
||||||
|
fi := msg.Extra["file"][0].(config.FileInfo)
|
||||||
|
/* if we have a sha we have successfully uploaded the file to the media server,
|
||||||
|
so we can now cache the sha */
|
||||||
|
if fi.SHA != "" {
|
||||||
|
b.Log.Debugf("Added %s to %s in avatarMap", fi.SHA, msg.UserID)
|
||||||
|
b.avatarMap[msg.UserID] = fi.SHA
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
}
|
}
|
||||||
|
104
bridge/whatsapp/handlers.go
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
package bwhatsapp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
|
||||||
|
"github.com/matterbridge/go-whatsapp"
|
||||||
|
|
||||||
|
whatsappExt "github.com/matterbridge/mautrix-whatsapp/whatsapp-ext"
|
||||||
|
)
|
||||||
|
|
||||||
|
/*
|
||||||
|
Implement handling messages coming from WhatsApp
|
||||||
|
Check:
|
||||||
|
- https://github.com/Rhymen/go-whatsapp#add-message-handlers
|
||||||
|
- https://github.com/Rhymen/go-whatsapp/blob/master/handler.go
|
||||||
|
- https://github.com/tulir/mautrix-whatsapp/tree/master/whatsapp-ext for more advanced command handling
|
||||||
|
*/
|
||||||
|
|
||||||
|
// HandleError received from WhatsApp
|
||||||
|
func (b *Bwhatsapp) HandleError(err error) {
|
||||||
|
b.Log.Errorf("%v", err) // TODO implement proper handling? at least respond to different error types
|
||||||
|
}
|
||||||
|
|
||||||
|
// HandleTextMessage sent from WhatsApp, relay it to the brige
|
||||||
|
func (b *Bwhatsapp) HandleTextMessage(message whatsapp.TextMessage) {
|
||||||
|
if message.Info.FromMe { // || !strings.Contains(strings.ToLower(message.Text), "@echo") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// whatsapp sends last messages to show context , cut them
|
||||||
|
if message.Info.Timestamp < b.startedAt {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
messageTime := time.Unix(int64(message.Info.Timestamp), 0) // TODO check how behaves between timezones
|
||||||
|
groupJid := message.Info.RemoteJid
|
||||||
|
|
||||||
|
senderJid := message.Info.SenderJid
|
||||||
|
if len(senderJid) == 0 {
|
||||||
|
// TODO workaround till https://github.com/Rhymen/go-whatsapp/issues/86 resolved
|
||||||
|
senderJid = *message.Info.Source.Participant
|
||||||
|
}
|
||||||
|
|
||||||
|
// translate sender's Jid to the nicest username we can get
|
||||||
|
senderName := b.getSenderName(senderJid)
|
||||||
|
if senderName == "" {
|
||||||
|
senderName = "Someone" // don't expose telephone number
|
||||||
|
}
|
||||||
|
|
||||||
|
extText := message.Info.Source.Message.ExtendedTextMessage
|
||||||
|
if extText != nil && extText.ContextInfo != nil && extText.ContextInfo.MentionedJid != nil {
|
||||||
|
// handle user mentions
|
||||||
|
for _, mentionedJid := range extText.ContextInfo.MentionedJid {
|
||||||
|
numberAndSuffix := strings.SplitN(mentionedJid, "@", 2)
|
||||||
|
|
||||||
|
// mentions comes as telephone numbers and we don't want to expose it to other bridges
|
||||||
|
// replace it with something more meaninful to others
|
||||||
|
mention := b.getSenderNotify(numberAndSuffix[0] + whatsappExt.NewUserSuffix)
|
||||||
|
if mention == "" {
|
||||||
|
mention = "someone"
|
||||||
|
}
|
||||||
|
message.Text = strings.Replace(message.Text, "@"+numberAndSuffix[0], "@"+mention, 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", senderJid, b.Account)
|
||||||
|
rmsg := config.Message{
|
||||||
|
UserID: senderJid,
|
||||||
|
Username: senderName,
|
||||||
|
Text: message.Text,
|
||||||
|
Timestamp: messageTime,
|
||||||
|
Channel: groupJid,
|
||||||
|
Account: b.Account,
|
||||||
|
Protocol: b.Protocol,
|
||||||
|
Extra: make(map[string][]interface{}),
|
||||||
|
// ParentID: TODO, // TODO handle thread replies // map from Info.QuotedMessageID string
|
||||||
|
// Event string `json:"event"`
|
||||||
|
// Gateway string // will be added during message processing
|
||||||
|
ID: message.Info.Id}
|
||||||
|
|
||||||
|
if avatarURL, exists := b.userAvatars[senderJid]; exists {
|
||||||
|
rmsg.Avatar = avatarURL
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
//func (b *Bwhatsapp) HandleImageMessage(message whatsapp.ImageMessage) {
|
||||||
|
// fmt.Println(message) // TODO implement
|
||||||
|
//}
|
||||||
|
//
|
||||||
|
//func (b *Bwhatsapp) HandleVideoMessage(message whatsapp.VideoMessage) {
|
||||||
|
// fmt.Println(message) // TODO implement
|
||||||
|
//}
|
||||||
|
//
|
||||||
|
//func (b *Bwhatsapp) HandleJsonMessage(message string) {
|
||||||
|
// fmt.Println(message) // TODO implement
|
||||||
|
//}
|
||||||
|
// TODO HandleRawMessage
|
||||||
|
// TODO HandleAudioMessage
|
84
bridge/whatsapp/helpers.go
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
package bwhatsapp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/gob"
|
||||||
|
"errors"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
qrcodeTerminal "github.com/Baozisoftware/qrcode-terminal-go"
|
||||||
|
"github.com/matterbridge/go-whatsapp"
|
||||||
|
)
|
||||||
|
|
||||||
|
func qrFromTerminal(invert bool) chan string {
|
||||||
|
qr := make(chan string)
|
||||||
|
go func() {
|
||||||
|
terminal := qrcodeTerminal.New()
|
||||||
|
if invert {
|
||||||
|
terminal = qrcodeTerminal.New2(qrcodeTerminal.ConsoleColors.BrightWhite, qrcodeTerminal.ConsoleColors.BrightBlack, qrcodeTerminal.QRCodeRecoveryLevels.Medium)
|
||||||
|
}
|
||||||
|
|
||||||
|
terminal.Get(<-qr).Print()
|
||||||
|
}()
|
||||||
|
|
||||||
|
return qr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bwhatsapp) readSession() (whatsapp.Session, error) {
|
||||||
|
session := whatsapp.Session{}
|
||||||
|
sessionFile := b.Config.GetString(sessionFile)
|
||||||
|
|
||||||
|
if sessionFile == "" {
|
||||||
|
return session, errors.New("if you won't set SessionFile then you will need to scan QR code on every restart")
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Open(sessionFile)
|
||||||
|
if err != nil {
|
||||||
|
return session, err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
decoder := gob.NewDecoder(file)
|
||||||
|
err = decoder.Decode(&session)
|
||||||
|
if err != nil {
|
||||||
|
return session, err
|
||||||
|
}
|
||||||
|
return session, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bwhatsapp) writeSession(session whatsapp.Session) error {
|
||||||
|
sessionFile := b.Config.GetString(sessionFile)
|
||||||
|
|
||||||
|
if sessionFile == "" {
|
||||||
|
// we already sent a warning while starting the bridge, so let's be quiet here
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Create(sessionFile)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
encoder := gob.NewEncoder(file)
|
||||||
|
err = encoder.Encode(session)
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bwhatsapp) getSenderName(senderJid string) string {
|
||||||
|
if sender, exists := b.users[senderJid]; exists {
|
||||||
|
if sender.Name != "" {
|
||||||
|
return sender.Name
|
||||||
|
}
|
||||||
|
// if user is not in phone contacts
|
||||||
|
// it is the most obvious scenario unless you sync your phone contacts with some remote updated source
|
||||||
|
// users can change it in their WhatsApp settings -> profile -> click on Avatar
|
||||||
|
return sender.Notify
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bwhatsapp) getSenderNotify(senderJid string) string {
|
||||||
|
if sender, exists := b.users[senderJid]; exists {
|
||||||
|
return sender.Notify
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
305
bridge/whatsapp/whatsapp.go
Normal file
@ -0,0 +1,305 @@
|
|||||||
|
package bwhatsapp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/rand"
|
||||||
|
"encoding/hex"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
|
||||||
|
"github.com/matterbridge/go-whatsapp"
|
||||||
|
|
||||||
|
whatsappExt "github.com/matterbridge/mautrix-whatsapp/whatsapp-ext"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Account config parameters
|
||||||
|
cfgNumber = "Number"
|
||||||
|
qrOnWhiteTerminal = "QrOnWhiteTerminal"
|
||||||
|
sessionFile = "SessionFile"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Bwhatsapp Bridge structure keeping all the information needed for relying
|
||||||
|
type Bwhatsapp struct {
|
||||||
|
*bridge.Config
|
||||||
|
|
||||||
|
// https://github.com/Rhymen/go-whatsapp/blob/c31092027237441cffba1b9cb148eadf7c83c3d2/session.go#L18-L21
|
||||||
|
session *whatsapp.Session
|
||||||
|
conn *whatsapp.Conn
|
||||||
|
// https://github.com/tulir/mautrix-whatsapp/blob/master/whatsapp-ext/whatsapp.go
|
||||||
|
connExt *whatsappExt.ExtendedConn
|
||||||
|
startedAt uint64
|
||||||
|
|
||||||
|
users map[string]whatsapp.Contact
|
||||||
|
userAvatars map[string]string
|
||||||
|
}
|
||||||
|
|
||||||
|
// New Create a new WhatsApp bridge. This will be called for each [whatsapp.<server>] entry you have in the config file
|
||||||
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
number := cfg.GetString(cfgNumber)
|
||||||
|
if number == "" {
|
||||||
|
cfg.Log.Fatalf("Missing configuration for WhatsApp bridge: Number")
|
||||||
|
}
|
||||||
|
|
||||||
|
b := &Bwhatsapp{
|
||||||
|
Config: cfg,
|
||||||
|
|
||||||
|
users: make(map[string]whatsapp.Contact),
|
||||||
|
userAvatars: make(map[string]string),
|
||||||
|
}
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
// Connect to WhatsApp. Required implementation of the Bridger interface
|
||||||
|
// https://github.com/42wim/matterbridge/blob/2cfd880cdb0df29771bf8f31df8d990ab897889d/bridge/bridge.go#L11-L16
|
||||||
|
func (b *Bwhatsapp) Connect() error {
|
||||||
|
b.RLock() // TODO do we need locking for Whatsapp?
|
||||||
|
defer b.RUnlock()
|
||||||
|
|
||||||
|
number := b.GetString(cfgNumber)
|
||||||
|
if number == "" {
|
||||||
|
return errors.New("WhatsApp's telephone Number need to be configured")
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://github.com/Rhymen/go-whatsapp#creating-a-connection
|
||||||
|
b.Log.Debugln("Connecting to WhatsApp..")
|
||||||
|
conn, err := whatsapp.NewConn(20 * time.Second)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New("failed to connect to WhatsApp: " + err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
b.conn = conn
|
||||||
|
b.connExt = whatsappExt.ExtendConn(b.conn)
|
||||||
|
// TODO do we want to use it? b.connExt.SetClientName("Matterbridge WhatsApp bridge", "mb-wa")
|
||||||
|
|
||||||
|
b.conn.AddHandler(b)
|
||||||
|
b.Log.Debugln("WhatsApp connection successful")
|
||||||
|
|
||||||
|
// load existing session in order to keep it between restarts
|
||||||
|
if b.session == nil {
|
||||||
|
var session whatsapp.Session
|
||||||
|
session, err = b.readSession()
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
b.Log.Debugln("Restoring WhatsApp session..")
|
||||||
|
|
||||||
|
// https://github.com/Rhymen/go-whatsapp#restore
|
||||||
|
session, err = b.conn.RestoreSession(session)
|
||||||
|
if err != nil {
|
||||||
|
// TODO return or continue to normal login?
|
||||||
|
// restore session connection timed out (I couldn't get over it without logging in again)
|
||||||
|
return errors.New("failed to restore session: " + err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
b.session = &session
|
||||||
|
b.Log.Debugln("Session restored successfully!")
|
||||||
|
} else {
|
||||||
|
b.Log.Warn(err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// login to a new session
|
||||||
|
if b.session == nil {
|
||||||
|
err = b.Login()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.startedAt = uint64(time.Now().Unix())
|
||||||
|
|
||||||
|
_, err = b.conn.Contacts()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error on update of contacts: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// map all the users
|
||||||
|
for id, contact := range b.conn.Store.Contacts {
|
||||||
|
if !isGroupJid(id) && id != "status@broadcast" {
|
||||||
|
// it is user
|
||||||
|
b.users[id] = contact
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// get user avatar asynchronously
|
||||||
|
go func() {
|
||||||
|
b.Log.Debug("Getting user avatars..")
|
||||||
|
|
||||||
|
for jid := range b.users {
|
||||||
|
info, err := b.connExt.GetProfilePicThumb(jid)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Warnf("Could not get profile photo of %s: %v", jid, err)
|
||||||
|
|
||||||
|
} else {
|
||||||
|
// TODO any race conditions here?
|
||||||
|
b.userAvatars[jid] = info.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b.Log.Debug("Finished getting avatars..")
|
||||||
|
}()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Login to WhatsApp creating a new session. This will require to scan a QR code on your mobile device
|
||||||
|
func (b *Bwhatsapp) Login() error {
|
||||||
|
b.Log.Debugln("Logging in..")
|
||||||
|
|
||||||
|
invert := b.GetBool(qrOnWhiteTerminal) // false is the default
|
||||||
|
qrChan := qrFromTerminal(invert)
|
||||||
|
|
||||||
|
session, err := b.conn.Login(qrChan)
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Warnln("Failed to log in:", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.session = &session
|
||||||
|
|
||||||
|
b.Log.Infof("Logged into session: %#v", session)
|
||||||
|
b.Log.Infof("Connection: %#v", b.conn)
|
||||||
|
|
||||||
|
err = b.writeSession(session)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "error saving session: %v\n", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO change connection strings to configured ones longClientName:"github.com/rhymen/go-whatsapp", shortClientName:"go-whatsapp"}" prefix=whatsapp
|
||||||
|
// TODO get also a nice logo
|
||||||
|
|
||||||
|
// TODO notification about unplugged and dead battery
|
||||||
|
// conn.Info: Wid, Pushname, Connected, Battery, Plugged
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Disconnect is called while reconnecting to the bridge
|
||||||
|
// TODO 42wim Documentation would be helpful on when reconnects happen and what should be done in this function
|
||||||
|
// Required implementation of the Bridger interface
|
||||||
|
// https://github.com/42wim/matterbridge/blob/2cfd880cdb0df29771bf8f31df8d990ab897889d/bridge/bridge.go#L11-L16
|
||||||
|
func (b *Bwhatsapp) Disconnect() error {
|
||||||
|
// We could Logout, but that would close the session completely and would require a new QR code scan
|
||||||
|
// https://github.com/Rhymen/go-whatsapp/blob/c31092027237441cffba1b9cb148eadf7c83c3d2/session.go#L377-L381
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func isGroupJid(identifier string) bool {
|
||||||
|
return strings.HasSuffix(identifier, "@g.us") || strings.HasSuffix(identifier, "@temp")
|
||||||
|
}
|
||||||
|
|
||||||
|
// JoinChannel Join a WhatsApp group specified in gateway config as channel='number-id@g.us' or channel='Channel name'
|
||||||
|
// Required implementation of the Bridger interface
|
||||||
|
// https://github.com/42wim/matterbridge/blob/2cfd880cdb0df29771bf8f31df8d990ab897889d/bridge/bridge.go#L11-L16
|
||||||
|
func (b *Bwhatsapp) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
byJid := isGroupJid(channel.Name)
|
||||||
|
|
||||||
|
// verify if we are member of the given group
|
||||||
|
if byJid {
|
||||||
|
// channel.Name specifies static group jID, not the name
|
||||||
|
if _, exists := b.conn.Store.Contacts[channel.Name]; !exists {
|
||||||
|
return fmt.Errorf("account doesn't belong to group with jid %s", channel.Name)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// channel.Name specifies group name that might change, warn about it
|
||||||
|
var jids []string
|
||||||
|
for id, contact := range b.conn.Store.Contacts {
|
||||||
|
if isGroupJid(id) && contact.Name == channel.Name {
|
||||||
|
jids = append(jids, id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
switch len(jids) {
|
||||||
|
case 0:
|
||||||
|
// didn't match any group - print out possibilites
|
||||||
|
// TODO sort
|
||||||
|
// copy b;
|
||||||
|
//sort.Slice(people, func(i, j int) bool {
|
||||||
|
// return people[i].Age > people[j].Age
|
||||||
|
//})
|
||||||
|
for id, contact := range b.conn.Store.Contacts {
|
||||||
|
if isGroupJid(id) {
|
||||||
|
b.Log.Infof("%s %s", contact.Jid, contact.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return fmt.Errorf("please specify group's JID from the list above instead of the name '%s'", channel.Name)
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
return fmt.Errorf("group name might change. Please configure gateway with channel=\"%v\" instead of channel=\"%v\"", jids[0], channel.Name)
|
||||||
|
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("there is more than one group with name '%s'. Please specify one of JIDs as channel name: %v", channel.Name, jids)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send a message from the bridge to WhatsApp
|
||||||
|
// Required implementation of the Bridger interface
|
||||||
|
// https://github.com/42wim/matterbridge/blob/2cfd880cdb0df29771bf8f31df8d990ab897889d/bridge/bridge.go#L11-L16
|
||||||
|
func (b *Bwhatsapp) Send(msg config.Message) (string, error) {
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
|
||||||
|
// Delete message
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
if msg.ID == "" {
|
||||||
|
// No message ID in case action is executed on a message sent before the bridge was started
|
||||||
|
// and then the bridge cache doesn't have this message ID mapped
|
||||||
|
|
||||||
|
// TODO 42wim Doesn't the app get clogged with a ton of IDs after some time of running?
|
||||||
|
// WhatsApp allows to set any ID so in that case we could use external IDs and don't do mapping
|
||||||
|
// but external IDs are not set
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
// TODO delete message on WhatsApp https://github.com/Rhymen/go-whatsapp/issues/100
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Edit message
|
||||||
|
if msg.ID != "" {
|
||||||
|
b.Log.Debugf("updating message with id %s", msg.ID)
|
||||||
|
|
||||||
|
msg.Text += " (edited)"
|
||||||
|
// TODO handle edit as a message reply with updated text
|
||||||
|
}
|
||||||
|
|
||||||
|
//// TODO Handle Upload a file
|
||||||
|
//if msg.Extra != nil {
|
||||||
|
// for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
// b.c.SendMessage(roomID, rmsg.Username+rmsg.Text)
|
||||||
|
// }
|
||||||
|
// if len(msg.Extra["file"]) > 0 {
|
||||||
|
// return b.handleUploadFile(&msg, roomID)
|
||||||
|
// }
|
||||||
|
//}
|
||||||
|
|
||||||
|
// Post text message
|
||||||
|
text := whatsapp.TextMessage{
|
||||||
|
Info: whatsapp.MessageInfo{
|
||||||
|
RemoteJid: msg.Channel, // which equals to group id
|
||||||
|
},
|
||||||
|
Text: msg.Username + msg.Text,
|
||||||
|
}
|
||||||
|
|
||||||
|
b.Log.Debugf("=> Sending %#v", msg)
|
||||||
|
|
||||||
|
// create message ID
|
||||||
|
// TODO follow and act if https://github.com/Rhymen/go-whatsapp/issues/101 implemented
|
||||||
|
bytes := make([]byte, 10)
|
||||||
|
if _, err := rand.Read(bytes); err != nil {
|
||||||
|
b.Log.Warn(err.Error())
|
||||||
|
}
|
||||||
|
text.Info.Id = strings.ToUpper(hex.EncodeToString(bytes))
|
||||||
|
|
||||||
|
err := b.conn.Send(text)
|
||||||
|
|
||||||
|
return text.Info.Id, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO do we want that? to allow login with QR code from a bridged channel? https://github.com/tulir/mautrix-whatsapp/blob/513eb18e2d59bada0dd515ee1abaaf38a3bfe3d5/commands.go#L76
|
||||||
|
//func (b *Bwhatsapp) Command(cmd string) string {
|
||||||
|
// return ""
|
||||||
|
//}
|
@ -1,83 +1,129 @@
|
|||||||
package bxmpp
|
package bxmpp
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
|
||||||
log "github.com/Sirupsen/logrus"
|
|
||||||
"github.com/mattn/go-xmpp"
|
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
|
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
"github.com/jpillora/backoff"
|
||||||
|
"github.com/matterbridge/go-xmpp"
|
||||||
|
"github.com/rs/xid"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Bxmpp struct {
|
type Bxmpp struct {
|
||||||
xc *xmpp.Client
|
xc *xmpp.Client
|
||||||
xmppMap map[string]string
|
xmppMap map[string]string
|
||||||
Config *config.Protocol
|
*bridge.Config
|
||||||
Remote chan config.Message
|
startTime time.Time
|
||||||
Account string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var flog *log.Entry
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
var protocol = "xmpp"
|
b := &Bxmpp{Config: cfg}
|
||||||
|
|
||||||
func init() {
|
|
||||||
flog = log.WithFields(log.Fields{"module": protocol})
|
|
||||||
}
|
|
||||||
|
|
||||||
func New(cfg config.Protocol, account string, c chan config.Message) *Bxmpp {
|
|
||||||
b := &Bxmpp{}
|
|
||||||
b.xmppMap = make(map[string]string)
|
b.xmppMap = make(map[string]string)
|
||||||
b.Config = &cfg
|
|
||||||
b.Account = account
|
|
||||||
b.Remote = c
|
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bxmpp) Connect() error {
|
func (b *Bxmpp) Connect() error {
|
||||||
var err error
|
var err error
|
||||||
flog.Infof("Connecting %s", b.Config.Server)
|
b.Log.Infof("Connecting %s", b.GetString("Server"))
|
||||||
b.xc, err = b.createXMPP()
|
b.xc, err = b.createXMPP()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
flog.Debugf("%#v", err)
|
b.Log.Debugf("%#v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
flog.Info("Connection succeeded")
|
b.Log.Info("Connection succeeded")
|
||||||
go b.handleXmpp()
|
go func() {
|
||||||
|
initial := true
|
||||||
|
bf := &backoff.Backoff{
|
||||||
|
Min: time.Second,
|
||||||
|
Max: 5 * time.Minute,
|
||||||
|
Jitter: true,
|
||||||
|
}
|
||||||
|
for {
|
||||||
|
if initial {
|
||||||
|
b.handleXMPP()
|
||||||
|
initial = false
|
||||||
|
}
|
||||||
|
d := bf.Duration()
|
||||||
|
b.Log.Infof("Disconnected. Reconnecting in %s", d)
|
||||||
|
time.Sleep(d)
|
||||||
|
b.xc, err = b.createXMPP()
|
||||||
|
if err == nil {
|
||||||
|
b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: "", Account: b.Account, Event: config.EventRejoinChannels}
|
||||||
|
b.handleXMPP()
|
||||||
|
bf.Reset()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bxmpp) JoinChannel(channel string) error {
|
func (b *Bxmpp) Disconnect() error {
|
||||||
b.xc.JoinMUCNoHistory(channel+"@"+b.Config.Muc, b.Config.Nick)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bxmpp) Send(msg config.Message) error {
|
func (b *Bxmpp) JoinChannel(channel config.ChannelInfo) error {
|
||||||
flog.Debugf("Receiving %#v", msg)
|
if channel.Options.Key != "" {
|
||||||
b.xc.Send(xmpp.Chat{Type: "groupchat", Remote: msg.Channel + "@" + b.Config.Muc, Text: msg.Username + msg.Text})
|
b.Log.Debugf("using key %s for channel %s", channel.Options.Key, channel.Name)
|
||||||
|
b.xc.JoinProtectedMUC(channel.Name+"@"+b.GetString("Muc"), b.GetString("Nick"), channel.Options.Key, xmpp.NoHistory, 0, nil)
|
||||||
|
} else {
|
||||||
|
b.xc.JoinMUCNoHistory(channel.Name+"@"+b.GetString("Muc"), b.GetString("Nick"))
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Bxmpp) Send(msg config.Message) (string, error) {
|
||||||
|
// ignore delete messages
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
|
||||||
|
// Upload a file (in xmpp case send the upload URL because xmpp has no native upload support)
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
b.xc.Send(xmpp.Chat{Type: "groupchat", Remote: rmsg.Channel + "@" + b.GetString("Muc"), Text: rmsg.Username + rmsg.Text})
|
||||||
|
}
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
return b.handleUploadFile(&msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var msgreplaceid string
|
||||||
|
msgid := xid.New().String()
|
||||||
|
if msg.ID != "" {
|
||||||
|
msgid = msg.ID
|
||||||
|
msgreplaceid = msg.ID
|
||||||
|
}
|
||||||
|
// Post normal message
|
||||||
|
_, err := b.xc.Send(xmpp.Chat{Type: "groupchat", Remote: msg.Channel + "@" + b.GetString("Muc"), Text: msg.Username + msg.Text, ID: msgid, ReplaceID: msgreplaceid})
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return msgid, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (b *Bxmpp) createXMPP() (*xmpp.Client, error) {
|
func (b *Bxmpp) createXMPP() (*xmpp.Client, error) {
|
||||||
tc := new(tls.Config)
|
tc := new(tls.Config)
|
||||||
tc.InsecureSkipVerify = b.Config.SkipTLSVerify
|
tc.InsecureSkipVerify = b.GetBool("SkipTLSVerify")
|
||||||
tc.ServerName = strings.Split(b.Config.Server, ":")[0]
|
tc.ServerName = strings.Split(b.GetString("Server"), ":")[0]
|
||||||
options := xmpp.Options{
|
options := xmpp.Options{
|
||||||
Host: b.Config.Server,
|
Host: b.GetString("Server"),
|
||||||
User: b.Config.Jid,
|
User: b.GetString("Jid"),
|
||||||
Password: b.Config.Password,
|
Password: b.GetString("Password"),
|
||||||
NoTLS: true,
|
NoTLS: true,
|
||||||
StartTLS: true,
|
StartTLS: true,
|
||||||
TLSConfig: tc,
|
TLSConfig: tc,
|
||||||
|
Debug: b.GetBool("debug"),
|
||||||
//StartTLS: false,
|
Logger: b.Log.Writer(),
|
||||||
Debug: true,
|
|
||||||
Session: true,
|
Session: true,
|
||||||
Status: "",
|
Status: "",
|
||||||
StatusMessage: "",
|
StatusMessage: "",
|
||||||
Resource: "",
|
Resource: "",
|
||||||
InsecureAllowUnencryptedAuth: false,
|
InsecureAllowUnencryptedAuth: false,
|
||||||
//InsecureAllowUnencryptedAuth: true,
|
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
b.xc, err = options.NewClient()
|
b.xc, err = options.NewClient()
|
||||||
@ -92,7 +138,11 @@ func (b *Bxmpp) xmppKeepAlive() chan bool {
|
|||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case <-ticker.C:
|
case <-ticker.C:
|
||||||
b.xc.PingC2S("", "")
|
b.Log.Debugf("PING")
|
||||||
|
err := b.xc.PingC2S("", "")
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Debugf("PING failed %#v", err)
|
||||||
|
}
|
||||||
case <-done:
|
case <-done:
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -101,10 +151,12 @@ func (b *Bxmpp) xmppKeepAlive() chan bool {
|
|||||||
return done
|
return done
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Bxmpp) handleXmpp() error {
|
func (b *Bxmpp) handleXMPP() error {
|
||||||
|
var ok bool
|
||||||
|
var msgid string
|
||||||
|
b.startTime = time.Now()
|
||||||
done := b.xmppKeepAlive()
|
done := b.xmppKeepAlive()
|
||||||
defer close(done)
|
defer close(done)
|
||||||
nodelay := time.Time{}
|
|
||||||
for {
|
for {
|
||||||
m, err := b.xc.Recv()
|
m, err := b.xc.Recv()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -112,23 +164,121 @@ func (b *Bxmpp) handleXmpp() error {
|
|||||||
}
|
}
|
||||||
switch v := m.(type) {
|
switch v := m.(type) {
|
||||||
case xmpp.Chat:
|
case xmpp.Chat:
|
||||||
var channel, nick string
|
|
||||||
if v.Type == "groupchat" {
|
if v.Type == "groupchat" {
|
||||||
s := strings.Split(v.Remote, "@")
|
b.Log.Debugf("== Receiving %#v", v)
|
||||||
if len(s) == 2 {
|
event := ""
|
||||||
channel = s[0]
|
// skip invalid messages
|
||||||
|
if b.skipMessage(v) {
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
s = strings.Split(s[1], "/")
|
if strings.Contains(v.Text, "has set the subject to:") {
|
||||||
if len(s) == 2 {
|
event = config.EventTopicChange
|
||||||
nick = s[1]
|
|
||||||
}
|
}
|
||||||
if nick != b.Config.Nick && v.Stamp == nodelay && v.Text != "" {
|
msgid = v.ID
|
||||||
flog.Debugf("Sending message from %s on %s to gateway", nick, b.Account)
|
if v.ReplaceID != "" {
|
||||||
b.Remote <- config.Message{Username: nick, Text: v.Text, Channel: channel, Account: b.Account}
|
msgid = v.ReplaceID
|
||||||
}
|
}
|
||||||
|
rmsg := config.Message{
|
||||||
|
Username: b.parseNick(v.Remote),
|
||||||
|
Text: v.Text,
|
||||||
|
Channel: b.parseChannel(v.Remote),
|
||||||
|
Account: b.Account,
|
||||||
|
UserID: v.Remote,
|
||||||
|
ID: msgid,
|
||||||
|
Event: event,
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if we have an action event
|
||||||
|
rmsg.Text, ok = b.replaceAction(rmsg.Text)
|
||||||
|
if ok {
|
||||||
|
rmsg.Event = config.EventUserAction
|
||||||
|
}
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", rmsg.Username, b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
}
|
}
|
||||||
case xmpp.Presence:
|
case xmpp.Presence:
|
||||||
// do nothing
|
// do nothing
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *Bxmpp) replaceAction(text string) (string, bool) {
|
||||||
|
if strings.HasPrefix(text, "/me ") {
|
||||||
|
return strings.Replace(text, "/me ", "", -1), true
|
||||||
|
}
|
||||||
|
return text, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleUploadFile handles native upload of files
|
||||||
|
func (b *Bxmpp) handleUploadFile(msg *config.Message) (string, error) {
|
||||||
|
var urldesc = ""
|
||||||
|
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text += fi.Comment + ": "
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text = fi.URL
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text = fi.Comment + ": " + fi.URL
|
||||||
|
urldesc = fi.Comment
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_, err := b.xc.Send(xmpp.Chat{Type: "groupchat", Remote: msg.Channel + "@" + b.GetString("Muc"), Text: msg.Username + msg.Text})
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
b.xc.SendOOB(xmpp.Chat{Type: "groupchat", Remote: msg.Channel + "@" + b.GetString("Muc"), Ooburl: fi.URL, Oobdesc: urldesc})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bxmpp) parseNick(remote string) string {
|
||||||
|
s := strings.Split(remote, "@")
|
||||||
|
if len(s) > 0 {
|
||||||
|
s = strings.Split(s[1], "/")
|
||||||
|
if len(s) == 2 {
|
||||||
|
return s[1] // nick
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bxmpp) parseChannel(remote string) string {
|
||||||
|
s := strings.Split(remote, "@")
|
||||||
|
if len(s) >= 2 {
|
||||||
|
return s[0] // channel
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// skipMessage skips messages that need to be skipped
|
||||||
|
func (b *Bxmpp) skipMessage(message xmpp.Chat) bool {
|
||||||
|
// skip messages from ourselves
|
||||||
|
if b.parseNick(message.Remote) == b.GetString("Nick") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// skip empty messages
|
||||||
|
if message.Text == "" {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// skip subject messages
|
||||||
|
if strings.Contains(message.Text, "</subject>") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// do not show subjects on connect #732
|
||||||
|
if strings.Contains(message.Text, "has set the subject to:") && time.Since(b.startTime) < time.Second*5 {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// skip delayed messages
|
||||||
|
t := time.Time{}
|
||||||
|
return message.Stamp != t
|
||||||
|
}
|
||||||
|
207
bridge/zulip/zulip.go
Normal file
@ -0,0 +1,207 @@
|
|||||||
|
package bzulip
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"io/ioutil"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/bridge/helper"
|
||||||
|
gzb "github.com/matterbridge/gozulipbot"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Bzulip struct {
|
||||||
|
q *gzb.Queue
|
||||||
|
bot *gzb.Bot
|
||||||
|
streams map[int]string
|
||||||
|
*bridge.Config
|
||||||
|
sync.RWMutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
return &Bzulip{Config: cfg, streams: make(map[int]string)}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bzulip) Connect() error {
|
||||||
|
bot := gzb.Bot{APIKey: b.GetString("token"), APIURL: b.GetString("server") + "/api/v1/", Email: b.GetString("login")}
|
||||||
|
bot.Init()
|
||||||
|
q, err := bot.RegisterAll()
|
||||||
|
b.q = q
|
||||||
|
b.bot = &bot
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("Connect() %#v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// init stream
|
||||||
|
b.getChannel(0)
|
||||||
|
b.Log.Info("Connection succeeded")
|
||||||
|
go b.handleQueue()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bzulip) Disconnect() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bzulip) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bzulip) Send(msg config.Message) (string, error) {
|
||||||
|
b.Log.Debugf("=> Receiving %#v", msg)
|
||||||
|
|
||||||
|
// Delete message
|
||||||
|
if msg.Event == config.EventMsgDelete {
|
||||||
|
if msg.ID == "" {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
_, err := b.bot.UpdateMessage(msg.ID, "")
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload a file if it exists
|
||||||
|
if msg.Extra != nil {
|
||||||
|
for _, rmsg := range helper.HandleExtra(&msg, b.General) {
|
||||||
|
b.sendMessage(rmsg)
|
||||||
|
}
|
||||||
|
if len(msg.Extra["file"]) > 0 {
|
||||||
|
return b.handleUploadFile(&msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// edit the message if we have a msg ID
|
||||||
|
if msg.ID != "" {
|
||||||
|
_, err := b.bot.UpdateMessage(msg.ID, msg.Username+msg.Text)
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Post normal message
|
||||||
|
return b.sendMessage(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bzulip) getChannel(id int) string {
|
||||||
|
if name, ok := b.streams[id]; ok {
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
streams, err := b.bot.GetRawStreams()
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("getChannel: %#v", err)
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
for _, stream := range streams.Streams {
|
||||||
|
b.streams[stream.StreamID] = stream.Name
|
||||||
|
}
|
||||||
|
if name, ok := b.streams[id]; ok {
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bzulip) handleQueue() error {
|
||||||
|
for {
|
||||||
|
messages, err := b.q.GetEvents()
|
||||||
|
switch err {
|
||||||
|
case gzb.BackoffError:
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
case gzb.NoJSONError:
|
||||||
|
b.Log.Error("Response wasn't JSON, server down or restarting? sleeping 10 seconds")
|
||||||
|
time.Sleep(time.Second * 10)
|
||||||
|
case gzb.BadEventQueueError:
|
||||||
|
b.Log.Info("got a bad event queue id error, reconnecting")
|
||||||
|
b.bot.Queues = nil
|
||||||
|
for {
|
||||||
|
b.q, err = b.bot.RegisterAll()
|
||||||
|
if err != nil {
|
||||||
|
b.Log.Errorf("reconnecting failed: %s. Sleeping 10 seconds", err)
|
||||||
|
time.Sleep(time.Second * 10)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
case gzb.HeartbeatError:
|
||||||
|
b.Log.Debug("heartbeat received.")
|
||||||
|
default:
|
||||||
|
b.Log.Debugf("receiving error: %#v", err)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
for _, m := range messages {
|
||||||
|
b.Log.Debugf("== Receiving %#v", m)
|
||||||
|
// ignore our own messages
|
||||||
|
if m.SenderEmail == b.GetString("login") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
rmsg := config.Message{
|
||||||
|
Username: m.SenderFullName,
|
||||||
|
Text: m.Content,
|
||||||
|
Channel: b.getChannel(m.StreamID) + "/topic:" + m.Subject,
|
||||||
|
Account: b.Account,
|
||||||
|
UserID: strconv.Itoa(m.SenderID),
|
||||||
|
Avatar: m.AvatarURL,
|
||||||
|
}
|
||||||
|
b.Log.Debugf("<= Sending message from %s on %s to gateway", rmsg.Username, b.Account)
|
||||||
|
b.Log.Debugf("<= Message is %#v", rmsg)
|
||||||
|
b.Remote <- rmsg
|
||||||
|
b.q.LastEventID = m.ID
|
||||||
|
}
|
||||||
|
time.Sleep(time.Second * 3)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bzulip) sendMessage(msg config.Message) (string, error) {
|
||||||
|
topic := ""
|
||||||
|
if strings.Contains(msg.Channel, "/topic:") {
|
||||||
|
res := strings.Split(msg.Channel, "/topic:")
|
||||||
|
topic = res[1]
|
||||||
|
msg.Channel = res[0]
|
||||||
|
}
|
||||||
|
m := gzb.Message{
|
||||||
|
Stream: msg.Channel,
|
||||||
|
Topic: topic,
|
||||||
|
Content: msg.Username + msg.Text,
|
||||||
|
}
|
||||||
|
resp, err := b.bot.Message(m)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if resp != nil {
|
||||||
|
defer resp.Body.Close()
|
||||||
|
res, err := ioutil.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
var jr struct {
|
||||||
|
ID int `json:"id"`
|
||||||
|
}
|
||||||
|
err = json.Unmarshal(res, &jr)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return strconv.Itoa(jr.ID), nil
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bzulip) handleUploadFile(msg *config.Message) (string, error) {
|
||||||
|
for _, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text += fi.Comment + ": "
|
||||||
|
}
|
||||||
|
if fi.URL != "" {
|
||||||
|
msg.Text = fi.URL
|
||||||
|
if fi.Comment != "" {
|
||||||
|
msg.Text = fi.Comment + ": " + fi.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_, err := b.sendMessage(*msg)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
690
changelog.md
@ -1,4 +1,691 @@
|
|||||||
|
# v1.14.1
|
||||||
|
## Bugfix
|
||||||
|
* slack: Fix crash double unlock (slack) (#771)
|
||||||
|
|
||||||
|
# v1.14.0
|
||||||
|
|
||||||
|
## Breaking
|
||||||
|
* zulip: Need to specify /topic:mytopic for channel configuration (zulip). (#751)
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* whatsapp: new protocol added. Add initial WhatsApp support (#711) Thanks to @KrzysztofMadejski
|
||||||
|
* facebook messenger: new protocol via matterbridge api. See https://github.com/VictorNine/fbridge/ for more information.
|
||||||
|
* general: Add scripting (tengo) support for every incoming message (#731). See `TengoModifyMessage`
|
||||||
|
* general: Allow regexs in ignoreNicks. Closes #690 (#720)
|
||||||
|
* general: Support rewriting messages from relaybots using ExtractNicks. Fixes #466 (#730). See `ExtractNicks` in matterbridge.toml.sample
|
||||||
|
* general: refactor Make all loggers derive from non-default instance (#728). Thanks to @Helcaraxan
|
||||||
|
* rocketchat: add support for the rocketchat API. Sending to rocketchat now supports uploading of files, editing and deleting of messages.
|
||||||
|
* discord: Support join/leaves from discord. Closes #654 (#721)
|
||||||
|
* discord: Allow sending discriminator with Discord username (#726). See `UseDiscriminator` in matterbridge.toml.sample
|
||||||
|
* slack: Add extra debug option (slack). See `Debug` in the slack section in matterbridge.toml.sample
|
||||||
|
* telegram: Add support for URL in messageEntities (telegram). Fixes #735 (#736)
|
||||||
|
* telegram: Add MediaConvertWebPToPNG option (telegram). (#741). See `MediaConvertWebPToPNG` in matterbridge.toml.sample
|
||||||
|
|
||||||
|
## Enhancements
|
||||||
|
* general: Fail gracefully on incorrect human input. Fixes #739 (#740)
|
||||||
|
* matrix: Detect html nicks in RemoteNickFormat (matrix). Fixes #696 (#719)
|
||||||
|
* matrix: Send notices on join/parts (matrix). Fixes #712 (#716)
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* general: Handle file upload/download only once for each message (#742)
|
||||||
|
* zulip: Fix error handling on bad event queue id (zulip). Closes #694
|
||||||
|
* zulip: Keep reconnecting until succeed (zulip) (#737)
|
||||||
|
* irc: add support for (older) unrealircd versions. #708
|
||||||
|
* irc: Support quits from irc correctly. Fixes #722 (#724)
|
||||||
|
* matrix: Send username when uploading video/images (matrix). Fixes #715 (#717)
|
||||||
|
* matrix: Trim <p> and </p> tags (matrix). Closes #686 (#753)
|
||||||
|
* slack: Hint at thread replies when messages are unthreaded (slack) (#684)
|
||||||
|
* slack: Fix race-condition in populateUser() (#767)
|
||||||
|
* xmpp: Do not send topic changes on connect (xmpp). Fixes #732 (#733)
|
||||||
|
* telegram: Fix regression in HTML handling (telegram). Closes #734
|
||||||
|
* discord: Do not relay any bot messages (discord) (#743)
|
||||||
|
* rocketchat: Do not send duplicate messages (rocketchat). Fixes #745 (#752)
|
||||||
|
|
||||||
|
## Contributors
|
||||||
|
This release couldn't exist without the following contributors:
|
||||||
|
@Helcaraxan, @KrzysztofMadejski, @AJolly, @DeclanHoare
|
||||||
|
|
||||||
|
# v1.13.1
|
||||||
|
|
||||||
|
This release fixes go modules issues because of https://github.com/labstack/echo/issues/1272
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* general: fixes Unable to build 1.13.0 #698
|
||||||
|
* api: move to labstack/echo/v4 fixes #698
|
||||||
|
|
||||||
|
# v1.13.0
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* general: refactors of telegram, irc, mattermost, matrix, discord, sshchat bridges and the gateway.
|
||||||
|
* irc: Add option to send RAW commands after connection (irc) #490. See `RunCommands` in matterbridge.toml.sample
|
||||||
|
* mattermost: 3.x support dropped
|
||||||
|
* mattermost: Add support for mattermost threading (#627)
|
||||||
|
* slack: Sync channel topics between Slack bridges #585. See `SyncTopic` in matterbridge.toml.sample
|
||||||
|
* matrix: Add support for markdown to HTML conversion (matrix). Closes #663 (#670)
|
||||||
|
* discord: Improve error reporting on failure to join Discord. Fixes #672 (#680)
|
||||||
|
* discord: Use only one webhook if possible (discord) (#681)
|
||||||
|
* discord: Allow to bridge non-bot Discord users (discord) (#689) If you prefix a token with `User ` it'll treat is as a user token.
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: Try downloading files again if slack is too slow (slack). Closes #655 (#656)
|
||||||
|
* slack: Ignore LatencyReport event (slack)
|
||||||
|
* slack: Fix #668 strip lang in code fences sent to Slack (#673)
|
||||||
|
* sshchat: Fix sshchat connection logic (#661)
|
||||||
|
* sshchat: set quiet mode to filter joins/quits
|
||||||
|
* sshchat: Trim newlines in the end of relayed messages
|
||||||
|
* sshchat: fix media links
|
||||||
|
* sshchat: do not relay "Rate limiting is in effect" message
|
||||||
|
* mattermost: Fail if channel starts with hashtag (mattermost). Closes #625
|
||||||
|
* discord: Add file comment to webhook messages (discord). Fixes #358
|
||||||
|
* matrix: Fix displaying usernames for plain text clients. (matrix) (#685)
|
||||||
|
* irc: Fix possible data race (irc). Closes #693
|
||||||
|
* irc: Handle servers without MOTD (irc). Closes #692
|
||||||
|
|
||||||
|
# v1.12.3
|
||||||
|
## Bugfix
|
||||||
|
* slack: Fix bot (legacy token) messages not being send. Closes #571
|
||||||
|
* slack: Populate user on channel join (slack) (#644)
|
||||||
|
* slack: Add wait option for populateUsers/Channels (slack) Fixes #579 (#653)
|
||||||
|
|
||||||
|
# v1.12.2
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* irc: Fix multiple channel join regression. Closes #639
|
||||||
|
* slack: Make slack-legacy change less restrictive (#626)
|
||||||
|
|
||||||
|
# v1.12.1
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* discord: fix regression on server ID connection #619 #617
|
||||||
|
* discord: Limit discord username via webhook to 32 chars
|
||||||
|
* slack: Make sure threaded files stay in thread (slack). Fixes #590
|
||||||
|
* slack: Do not post empty messages (slack). Fixes #574
|
||||||
|
* slack: Handle deleted/edited thread starting messages (slack). Fixes #600 (#605)
|
||||||
|
* irc: Rework connection logic (irc)
|
||||||
|
* irc: Fix Nickserv logic (irc) #602
|
||||||
|
|
||||||
|
# v1.12.0
|
||||||
|
|
||||||
|
## Breaking changes
|
||||||
|
The slack bridge has been split in a `slack-legacy` and `slack` bridge.
|
||||||
|
If you're still using `legacy tokens` and want to keep using them you'll have to rename `slack` to `slack-legacy` in your configuration. See [wiki](https://github.com/42wim/matterbridge/wiki/Section-Slack-(basic)#legacy-configuration) for more information.
|
||||||
|
|
||||||
|
To migrate to the new bot-token based setup you can follow the instructions [here](https://github.com/42wim/matterbridge/wiki/Slack-bot-setup).
|
||||||
|
|
||||||
|
Slack legacy tokens may be deprecated by Slack at short notice, so it is STRONGLY recommended to use a proper bot-token instead.
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* general: New {GATEWAY} variable for `RemoteNickFormat` #501. See `RemoteNickFormat` in matterbridge.toml.sample.
|
||||||
|
* general: New {CHANNEL} variable for `RemoteNickFormat` #515. See `RemoteNickFormat` in matterbridge.toml.sample.
|
||||||
|
* general: Remove hyphens when auto-loading envvars from viper config #545
|
||||||
|
* discord: You can mention discord-users from other bridges.
|
||||||
|
* slack: Preserve threading between Slack instances #529. See `PreserveThreading` in matterbridge.toml.sample.
|
||||||
|
* slack: Add ability to show when user is typing across Slack bridges #559
|
||||||
|
* slack: Add rate-limiting
|
||||||
|
* mattermost: Add support for mattermost [matterbridge plugin](https://github.com/matterbridge/mattermost-plugin)
|
||||||
|
* api: Respond with message on connect. #550
|
||||||
|
* api: Add a health endpoint to API #554
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: Refactoring and making it better.
|
||||||
|
* slack: Restore file comments coming from Slack. #583
|
||||||
|
* irc: Fix IRC line splitting. #587
|
||||||
|
* mattermost: Fix cookie and personal token behaviour. #530
|
||||||
|
* mattermost: Check for expiring sessions and reconnect.
|
||||||
|
|
||||||
|
|
||||||
|
## Contributors
|
||||||
|
This release couldn't exist without the following contributors:
|
||||||
|
@jheiselman, @NikkyAI, @dajohi, @NetwideRogue, @patcon and @Helcaraxan
|
||||||
|
|
||||||
|
Special thanks to @Helcaraxan and @patcon for their work on improving/refactoring slack.
|
||||||
|
|
||||||
|
# v1.11.3
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* mattermost: fix panic when using webhooks #491
|
||||||
|
* slack: fix issues regarding API changes and lots of channels #489
|
||||||
|
* irc: fix rejoin on kick problem #488
|
||||||
|
|
||||||
|
# v1.11.2
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: fix slack API changes regarding to files/images
|
||||||
|
|
||||||
|
# v1.11.1
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* slack: Add support for slack channels by ID. Closes #436
|
||||||
|
* discord: Clip too long messages sent to discord (discord). Closes #440
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* general: fix possible panic on downloads that are too big #448
|
||||||
|
* general: Fix avatar uploads to work with MediaDownloadPath. Closes #454
|
||||||
|
* discord: allow receiving of topic changes/channel leave/joins from other bridges through the webhook
|
||||||
|
* discord: Add a space before url in file uploads (discord). Closes #461
|
||||||
|
* discord: Skip empty messages being sent with the webhook (discord). #469
|
||||||
|
* mattermost: Use nickname instead of username if defined (mattermost). Closes #452
|
||||||
|
* irc: Stop numbers being stripped after non-color control codes (irc) (#465)
|
||||||
|
* slack: Use UserID to look for avatar instead of username (slack). Closes #472
|
||||||
|
|
||||||
|
# v1.11.0
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* general: Add config option MediaDownloadPath (#443). See `MediaDownloadPath` in matterbridge.toml.sample
|
||||||
|
* general: Add MediaDownloadBlacklist option. Closes #442. See `MediaDownloadBlacklist` in matterbridge.toml.sample
|
||||||
|
* xmpp: Add channel password support for XMPP (#451)
|
||||||
|
* xmpp: Add message correction support for XMPP (#437)
|
||||||
|
* telegram: Add support for MessageFormat=htmlnick (telegram). #444
|
||||||
|
* mattermost: Add support for mattermost 5.x
|
||||||
|
|
||||||
|
## Enhancements
|
||||||
|
* slack: Add Title from attachment slack message (#446)
|
||||||
|
* irc: Prevent white or black color codes (irc) (#434)
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: Fix regexp in replaceMention (slack). (#435)
|
||||||
|
* irc: Reconnect on quit. (irc) See #431 (#445)
|
||||||
|
* sshchat: Ignore messages from ourself. (sshchat) Closes #439
|
||||||
|
|
||||||
|
# v1.10.1
|
||||||
|
## New features
|
||||||
|
* irc: Colorize username sent to IRC using its crc32 IEEE checksum (#423). See `ColorNicks` in matterbridge.toml.sample
|
||||||
|
* irc: Add support for CJK to/from utf-8 (irc). #400
|
||||||
|
* telegram: Add QuoteFormat option (telegram). Closes #413. See `QuoteFormat` in matterbridge.toml.sample
|
||||||
|
* xmpp: Send attached files to XMPP in different message with OOB data and without body (#421)
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* general: updated irc/xmpp/telegram libraries
|
||||||
|
* mattermost/slack/rocketchat: Fix iconurl regression. Closes #430
|
||||||
|
* mattermost/slack: Use uuid instead of userid. Fixes #429
|
||||||
|
* slack: Avatar spoofing from Slack to Discord with uppercase in nick doesn't work (#433)
|
||||||
|
* irc: Fix format string bug (irc) (#428)
|
||||||
|
|
||||||
|
# v1.10.0
|
||||||
|
## New features
|
||||||
|
* general: Add support for reloading all settings automatically after changing config except connection and gateway configuration. Closes #373
|
||||||
|
* zulip: New protocol support added (https://zulipchat.com)
|
||||||
|
|
||||||
|
## Enhancements
|
||||||
|
* general: Handle file comment better
|
||||||
|
* steam: Handle file uploads to mediaserver (steam)
|
||||||
|
* slack: Properly set Slack user who initiated slash command (#394)
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* general: Use only alphanumeric for file uploads to mediaserver. Closes #416
|
||||||
|
* general: Fix crash on invalid filenames
|
||||||
|
* general: Fix regression in ReplaceMessages and ReplaceNicks. Closes #407
|
||||||
|
* telegram: Fix possible nil when using channels (telegram). #410
|
||||||
|
* telegram: Fix panic (telegram). Closes #410
|
||||||
|
* telegram: Handle channel posts correctly
|
||||||
|
* mattermost: Update GetFileLinks to API_V4
|
||||||
|
|
||||||
|
# v1.9.1
|
||||||
|
## New features
|
||||||
|
* telegram: Add QuoteDisable option (telegram). Closes #399. See QuoteDisable in matterbridge.toml.sample
|
||||||
|
## Enhancements
|
||||||
|
* discord: Send mediaserver link to Discord in Webhook mode (discord) (#405)
|
||||||
|
* mattermost: Print list of valid team names when team not found (#390)
|
||||||
|
* slack: Strip markdown URLs with blank text (slack) (#392)
|
||||||
|
## Bugfix
|
||||||
|
* slack/mattermost: Make our callbackid more unique. Fixes issue with running multiple matterbridge on the same channel (slack,mattermost)
|
||||||
|
* telegram: fix newlines in multiline messages #399
|
||||||
|
* telegram: Revert #378
|
||||||
|
|
||||||
|
# v1.9.0 (the refactor release)
|
||||||
|
## New features
|
||||||
|
* general: better debug messages
|
||||||
|
* general: better support for environment variables override
|
||||||
|
* general: Ability to disable sending join/leave messages to other gateways. #382
|
||||||
|
* slack: Allow Slack @usergroups to be parsed as human-friendly names #379
|
||||||
|
* slack: Provide better context for shared posts from Slack<=>Slack enhancement #369
|
||||||
|
* telegram: Convert nicks automatically into HTML when MessageFormat is set to HTML #378
|
||||||
|
* irc: Add DebugLevel option
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: Ignore restricted_action on channel join (slack). Closes #387
|
||||||
|
* slack: Add slack attachment support to matterhook
|
||||||
|
* slack: Update userlist on join (slack). Closes #372
|
||||||
|
|
||||||
|
# v1.8.0
|
||||||
|
## New features
|
||||||
|
* general: Send chat notification if media is too big to be re-uploaded to MediaServer. See #359
|
||||||
|
* general: Download (and upload) avatar images from mattermost and telegram when mediaserver is configured. Closes #362
|
||||||
|
* general: Add label support in RemoteNickFormat
|
||||||
|
* general: Prettier info/debug log output
|
||||||
|
* mattermost: Download files and reupload to supported bridges (mattermost). Closes #357
|
||||||
|
* slack: Add ShowTopicChange option. Allow/disable topic change messages (currently only from slack). Closes #353
|
||||||
|
* slack: Add support for file comments (slack). Closes #346
|
||||||
|
* telegram: Add comment to file upload from telegram. Show comments on all bridges. Closes #358
|
||||||
|
* telegram: Add markdown support (telegram). #355
|
||||||
|
* api: Give api access to whole config.Message (and events). Closes #374
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* discord: Check for a valid WebhookURL (discord). Closes #367
|
||||||
|
* discord: Fix role mention replace issues
|
||||||
|
* irc: Truncate messages sent to IRC based on byte count (#368)
|
||||||
|
* mattermost: Add file download urls also to mattermost webhooks #356
|
||||||
|
* telegram: Fix panic on nil messages (telegram). Closes #366
|
||||||
|
* telegram: Fix the UseInsecureURL text (telegram). Closes #184
|
||||||
|
|
||||||
|
# v1.7.1
|
||||||
|
## Bugfix
|
||||||
|
* telegram: Enable Long Polling for Telegram. Reduces bandwidth consumption. (#350)
|
||||||
|
|
||||||
|
# v1.7.0
|
||||||
|
## New features
|
||||||
|
* matrix: Add support for deleting messages from/to matrix (matrix). Closes #320
|
||||||
|
* xmpp: Ignore <subject> messages (xmpp). #272
|
||||||
|
* irc: Add twitch support (irc) to README / wiki
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* general: Change RemoteNickFormat replacement order. Closes #336
|
||||||
|
* general: Make edits/delete work for bridges that gets reused. Closes #342
|
||||||
|
* general: Lowercase irc channels in config. Closes #348
|
||||||
|
* matrix: Fix possible panics (matrix). Closes #333
|
||||||
|
* matrix: Add an extension to images without one (matrix). #331
|
||||||
|
* api: Obey the Gateway value from the json (api). Closes #344
|
||||||
|
* xmpp: Print only debug messages when specified (xmpp). Closes #345
|
||||||
|
* xmpp: Allow xmpp to receive the extra messages (file uploads) when text is empty. #295
|
||||||
|
|
||||||
|
# v1.6.3
|
||||||
|
## Bugfix
|
||||||
|
* slack: Fix connection issues
|
||||||
|
* slack: Add more debug messages
|
||||||
|
* irc: Convert received IRC channel names to lowercase. Fixes #329 (#330)
|
||||||
|
|
||||||
|
# v1.6.2
|
||||||
|
## Bugfix
|
||||||
|
* mattermost: Crashes while connecting to Mattermost (regression). Closes #327
|
||||||
|
|
||||||
|
# v1.6.1
|
||||||
|
## Bugfix
|
||||||
|
* general: Display of nicks not longer working (regression). Closes #323
|
||||||
|
|
||||||
|
# v1.6.0
|
||||||
|
## New features
|
||||||
|
* sshchat: New protocol support added (https://github.com/shazow/ssh-chat)
|
||||||
|
* general: Allow specifying maximum download size of media using MediaDownloadSize (slack,telegram,matrix)
|
||||||
|
* api: Add (simple, one listener) long-polling support (api). Closes #307
|
||||||
|
* telegram: Add support for forwarded messages. Closes #313
|
||||||
|
* telegram: Add support for Audio/Voice files (telegram). Closes #314
|
||||||
|
* irc: Add RejoinDelay option. Delay to rejoin after channel kick (irc). Closes #322
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* telegram: Also use HTML in edited messages (telegram). Closes #315
|
||||||
|
* matrix: Fix panic (matrix). Closes #316
|
||||||
|
|
||||||
|
# v1.5.1
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* irc: Fix irc ACTION regression (irc). Closes #306
|
||||||
|
* irc: Split on UTF-8 for MessageSplit (irc). Closes #308
|
||||||
|
|
||||||
|
# v1.5.0
|
||||||
|
## New features
|
||||||
|
* general: remote mediaserver support. See MediaServerDownload and MediaServerUpload in matterbridge.toml.sample
|
||||||
|
more information on https://github.com/42wim/matterbridge/wiki/Mediaserver-setup-%5Badvanced%5D
|
||||||
|
* general: Add support for ReplaceNicks using regexp to replace nicks. Closes #269 (see matterbridge.toml.sample)
|
||||||
|
* general: Add support for ReplaceMessages using regexp to replace messages. #269 (see matterbridge.toml.sample)
|
||||||
|
* irc: Add MessageSplit option to split messages on MessageLength (irc). Closes #281
|
||||||
|
* matrix: Add support for uploading images/video (matrix). Closes #302
|
||||||
|
* matrix: Add support for uploaded images/video (matrix)
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* telegram: Add webp extension to stickers if necessary (telegram)
|
||||||
|
* mattermost: Break when re-login fails (mattermost)
|
||||||
|
|
||||||
|
# v1.4.1
|
||||||
|
## Bugfix
|
||||||
|
* telegram: fix issue with uploading for images/documents/stickers
|
||||||
|
* slack: remove double messages sent to other bridges when uploading files
|
||||||
|
* irc: Fix strict user handling of girc (irc). Closes #298
|
||||||
|
|
||||||
|
# v1.4.0
|
||||||
|
## Breaking changes
|
||||||
|
* general: `[general]` settings don't override the specific bridge settings
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* irc: Replace sorcix/irc and go-ircevent with girc, this should be give better reconnects
|
||||||
|
* steam: Add support for bridging to individual steam chats. (steam) (#294)
|
||||||
|
* telegram: Download files from telegram and reupload to supported bridges (telegram). #278
|
||||||
|
* slack: Add support to upload files to slack, from bridges with private urls like slack/mattermost/telegram. (slack)
|
||||||
|
* discord: Add support to upload files to discord, from bridges with private urls like slack/mattermost/telegram. (discord)
|
||||||
|
* general: Add systemd service file (#291)
|
||||||
|
* general: Add support for DEBUG=1 envvar to enable debug. Closes #283
|
||||||
|
* general: Add StripNick option, only allow alphanumerical nicks. Closes #285
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* gitter: Use room.URI instead of room.Name. (gitter) (#293)
|
||||||
|
* slack: Allow slack messages with variables (eg. @here) to be formatted correctly. (slack) (#288)
|
||||||
|
* slack: Resolve slack channel to human-readable name. (slack) (#282)
|
||||||
|
* slack: Use DisplayName instead of deprecated username (slack). Closes #276
|
||||||
|
* slack: Allowed Slack bridge to extract simpler link format. (#287)
|
||||||
|
* irc: Strip irc colors correct, strip also ctrl chars (irc)
|
||||||
|
|
||||||
|
# v1.3.1
|
||||||
|
## New features
|
||||||
|
* Support mattermost 4.3.0 and every other 4.x as api4 should be stable (mattermost)
|
||||||
|
## Bugfix
|
||||||
|
* Use bot username if specified (slack). Closes #273
|
||||||
|
|
||||||
|
# v1.3.0
|
||||||
|
## New features
|
||||||
|
* Relay slack_attachments from mattermost to slack (slack). Closes #260
|
||||||
|
* Add support for quoting previous message when replying (telegram). #237
|
||||||
|
* Add support for Quakenet auth (irc). Closes #263
|
||||||
|
* Download files (max size 1MB) from slack and reupload to mattermost (slack/mattermost). Closes #255
|
||||||
|
|
||||||
|
## Enhancements
|
||||||
|
* Backoff for 60 seconds when reconnecting too fast (irc) #267
|
||||||
|
* Use override username if specified (mattermost). #260
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* Try to not forward slack unfurls. Closes #266
|
||||||
|
|
||||||
|
# v1.2.0
|
||||||
|
## Breaking changes
|
||||||
|
* If you're running a discord bridge, update to this release before 16 october otherwise
|
||||||
|
it will stop working. (see https://discordapp.com/developers/docs/reference)
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* general: Add delete support. (actually delete the messages on bridges that support it)
|
||||||
|
(mattermost,discord,gitter,slack,telegram)
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* Do not break messages on newline (slack). Closes #258
|
||||||
|
* Update telegram library
|
||||||
|
* Update discord library (supports v6 API now). Old API is deprecated on 16 October
|
||||||
|
|
||||||
|
# v1.1.2
|
||||||
|
## New features
|
||||||
|
* general: also build darwin binaries
|
||||||
|
* mattermost: add support for mattermost 4.2.x
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* mattermost: Send images when text is empty regression. (mattermost). Closes #254
|
||||||
|
* slack: also send the first messsage after connect. #252
|
||||||
|
|
||||||
|
# v1.1.1
|
||||||
|
## Bugfix
|
||||||
|
* mattermost: fix public links
|
||||||
|
|
||||||
|
# v1.1.0
|
||||||
|
## New features
|
||||||
|
* general: Add better editing support. (actually edit the messages on bridges that support it)
|
||||||
|
(mattermost,discord,gitter,slack,telegram)
|
||||||
|
* mattermost: use API v4 (removes support for mattermost < 3.8)
|
||||||
|
* mattermost: add support for personal access tokens (since mattermost 4.1)
|
||||||
|
Use ```Token="yourtoken"``` in mattermost config
|
||||||
|
See https://docs.mattermost.com/developer/personal-access-tokens.html for more info
|
||||||
|
* matrix: Relay notices (matrix). Closes #243
|
||||||
|
* irc: Add a charset option. Closes #247
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: Handle leave/join events (slack). Closes #246
|
||||||
|
* slack: Replace mentions from other bridges. (slack). Closes #233
|
||||||
|
* gitter: remove ZWSP after messages
|
||||||
|
|
||||||
|
# v1.0.1
|
||||||
|
## New features
|
||||||
|
* mattermost: add support for mattermost 4.1.x
|
||||||
|
* discord: allow a webhookURL per channel #239
|
||||||
|
|
||||||
|
# v1.0.0
|
||||||
|
## New features
|
||||||
|
* general: Add action support for slack,mattermost,irc,gitter,matrix,xmpp,discord. #199
|
||||||
|
* discord: Shows the username instead of the server nickname #234
|
||||||
|
|
||||||
|
# v1.0.0-rc1
|
||||||
|
## New features
|
||||||
|
* general: Add action support for slack,mattermost,irc,gitter,matrix,xmpp,discord. #199
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* general: Handle same account in multiple gateways better
|
||||||
|
* mattermost: ignore edited messages with reactions
|
||||||
|
* mattermost: Fix double posting of edited messages by using lru cache
|
||||||
|
* irc: update vendor
|
||||||
|
|
||||||
|
# v0.16.3
|
||||||
|
## Bugfix
|
||||||
|
* general: Fix in/out logic. Closes #224
|
||||||
|
* general: Fix message modification
|
||||||
|
* slack: Disable message from other bots when using webhooks (slack)
|
||||||
|
* mattermost: Return better error messages on mattermost connect
|
||||||
|
|
||||||
|
# v0.16.2
|
||||||
|
## New features
|
||||||
|
* general: binary builds against latest commit are now available on https://bintray.com/42wim/nightly/Matterbridge/_latestVersion
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: fix loop introduced by relaying message of other bots #219
|
||||||
|
* slack: Suppress parent message when child message is received #218
|
||||||
|
* mattermost: fix regression when using webhookurl and webhookbindaddress #221
|
||||||
|
|
||||||
|
# v0.16.1
|
||||||
|
## New features
|
||||||
|
* slack: also relay messages of other bots #213
|
||||||
|
* mattermost: show also links if public links have not been enabled.
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* mattermost, slack: fix connecting logic #216
|
||||||
|
|
||||||
|
# v0.16.0
|
||||||
|
## Breaking Changes
|
||||||
|
* URL,UseAPI,BindAddress is deprecated. Your config has to be updated.
|
||||||
|
* URL => WebhookURL
|
||||||
|
* BindAddress => WebhookBindAddress
|
||||||
|
* UseAPI => removed
|
||||||
|
This change allows you to specify a WebhookURL and a token (slack,discord), so that
|
||||||
|
messages will be sent with the webhook, but received via the token (API)
|
||||||
|
If you have not specified WebhookURL and WebhookBindAddress the API (login or token)
|
||||||
|
will be used automatically. (no need for UseAPI)
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* mattermost: add support for mattermost 4.0
|
||||||
|
* steam: New protocol support added (http://store.steampowered.com/)
|
||||||
|
* discord: Support for embedded messages (sent by other bots)
|
||||||
|
Shows title, description and URL of embedded messages (sent by other bots)
|
||||||
|
To enable add ```ShowEmbeds=true``` to your discord config
|
||||||
|
* discord: ```WebhookURL``` posting support added (thanks @saury07) #204
|
||||||
|
Discord API does not allow to change the name of the user posting, but webhooks does.
|
||||||
|
|
||||||
|
## Changes
|
||||||
|
* general: all :emoji: will be converted to unicode, providing consistent emojis across all bridges
|
||||||
|
* telegram: Add ```UseInsecureURL``` option for telegram (default false)
|
||||||
|
WARNING! If enabled this will relay GIF/stickers/documents and other attachments as URLs
|
||||||
|
Those URLs will contain your bot-token. This may not be what you want.
|
||||||
|
For now there is no secure way to relay GIF/stickers/documents without seeing your token.
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* irc: detect charset and try to convert it to utf-8 before sending it to other bridges. #209 #210
|
||||||
|
* slack: Remove label from URLs (slack). #205
|
||||||
|
* slack: Relay <>& correctly to other bridges #215
|
||||||
|
* steam: Fix channel id bug in steam (channels are off by 0x18000000000000)
|
||||||
|
* general: various improvements
|
||||||
|
* general: samechannelgateway now relays messages correct again #207
|
||||||
|
|
||||||
|
|
||||||
|
# v0.16.0-rc2
|
||||||
|
## Breaking Changes
|
||||||
|
* URL,UseAPI,BindAddress is deprecated. Your config has to be updated.
|
||||||
|
* URL => WebhookURL
|
||||||
|
* BindAddress => WebhookBindAddress
|
||||||
|
* UseAPI => removed
|
||||||
|
This change allows you to specify a WebhookURL and a token (slack,discord), so that
|
||||||
|
messages will be sent with the webhook, but received via the token (API)
|
||||||
|
If you have not specified WebhookURL and WebhookBindAddress the API (login or token)
|
||||||
|
will be used automatically. (no need for UseAPI)
|
||||||
|
|
||||||
|
## Bugfix since rc1
|
||||||
|
* steam: Fix channel id bug in steam (channels are off by 0x18000000000000)
|
||||||
|
* telegram: Add UseInsecureURL option for telegram (default false)
|
||||||
|
WARNING! If enabled this will relay GIF/stickers/documents and other attachments as URLs
|
||||||
|
Those URLs will contain your bot-token. This may not be what you want.
|
||||||
|
For now there is no secure way to relay GIF/stickers/documents without seeing your token.
|
||||||
|
* irc: detect charset and try to convert it to utf-8 before sending it to other bridges. #209 #210
|
||||||
|
* general: various improvements
|
||||||
|
|
||||||
|
|
||||||
|
# v0.16.0-rc1
|
||||||
|
## Breaking Changes
|
||||||
|
* URL,UseAPI,BindAddress is deprecated. Your config has to be updated.
|
||||||
|
* URL => WebhookURL
|
||||||
|
* BindAddress => WebhookBindAddress
|
||||||
|
* UseAPI => removed
|
||||||
|
This change allows you to specify a WebhookURL and a token (slack,discord), so that
|
||||||
|
messages will be sent with the webhook, but received via the token (API)
|
||||||
|
If you have not specified WebhookURL and WebhookBindAddress the API (login or token)
|
||||||
|
will be used automatically. (no need for UseAPI)
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* steam: New protocol support added (http://store.steampowered.com/)
|
||||||
|
* discord: WebhookURL posting support added (thanks @saury07) #204
|
||||||
|
Discord API does not allow to change the name of the user posting, but webhooks does.
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* general: samechannelgateway now relays messages correct again #207
|
||||||
|
* slack: Remove label from URLs (slack). #205
|
||||||
|
|
||||||
|
# v0.15.0
|
||||||
|
## New features
|
||||||
|
* general: add option IgnoreMessages for all protocols (see mattebridge.toml.sample)
|
||||||
|
Messages matching these regexp will be ignored and not sent to other bridges
|
||||||
|
e.g. IgnoreMessages="^~~ badword"
|
||||||
|
* telegram: add support for sticker/video/photo/document #184
|
||||||
|
|
||||||
|
## Changes
|
||||||
|
* api: add userid to each message #200
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* discord: fix crash in memberupdate #198
|
||||||
|
* mattermost: Fix incorrect behaviour of EditDisable (mattermost). Fixes #197
|
||||||
|
* irc: Do not relay join/part of ourselves (irc). Closes #190
|
||||||
|
* irc: make reconnections more robust. #153
|
||||||
|
* gitter: update library, fixes possible crash
|
||||||
|
|
||||||
|
# v0.14.0
|
||||||
|
## New features
|
||||||
|
* api: add token authentication
|
||||||
|
* mattermost: add support for mattermost 3.10.0
|
||||||
|
|
||||||
|
## Changes
|
||||||
|
* api: gateway name is added in JSON messages
|
||||||
|
* api: lowercase JSON keys
|
||||||
|
* api: channel name isn't needed in config #195
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* discord: Add hashtag to channelname (when translating from id) (discord)
|
||||||
|
* mattermost: Fix a panic. #186
|
||||||
|
* mattermost: use teamid cache if possible. Fixes a panic
|
||||||
|
* api: post valid json. #185
|
||||||
|
* api: allow reuse of api in different gateways. #189
|
||||||
|
* general: Fix utf-8 issues for {NOPINGNICK}. #193
|
||||||
|
|
||||||
|
# v0.13.0
|
||||||
|
## New features
|
||||||
|
* irc: Limit message length. ```MessageLength=400```
|
||||||
|
Maximum length of message sent to irc server. If it exceeds <message clipped> will be add to the message.
|
||||||
|
* irc: Add NOPINGNICK option.
|
||||||
|
The string "{NOPINGNICK}" (case sensitive) will be replaced by the actual nick / username, but with a ZWSP inside the nick, so the irc user with the same nick won't get pinged.
|
||||||
|
See https://github.com/42wim/matterbridge/issues/175 for more information
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: Fix sending to different channels on same account (slack). Closes #177
|
||||||
|
* telegram: Fix incorrect usernames being sent. Closes #181
|
||||||
|
|
||||||
|
|
||||||
|
# v0.12.1
|
||||||
|
## New features
|
||||||
|
* telegram: Add UseFirstName option (telegram). Closes #144
|
||||||
|
* matrix: Add NoHomeServerSuffix. Option to disable homeserver on username (matrix). Closes #160.
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* xmpp: Add Compatibility for Cisco Jabber (xmpp) (#166)
|
||||||
|
* irc: Fix JoinChannel argument to use IRC channel key (#172)
|
||||||
|
* discord: Fix possible crash on nil (discord)
|
||||||
|
* discord: Replace long ids in channel metions (discord). Fixes #174
|
||||||
|
|
||||||
|
# v0.12.0
|
||||||
|
## Changes
|
||||||
|
* general: edited messages are now being sent by default on discord/mattermost/telegram/slack. See "New Features"
|
||||||
|
|
||||||
|
## New features
|
||||||
|
* general: add support for edited messages.
|
||||||
|
Add new keyword EditDisable (false/true), default false. Which means by default edited messages will be sent to other bridges.
|
||||||
|
Add new keyword EditSuffix , default "". You can change this eg to "(edited)", this will be appended to every edit message.
|
||||||
|
* mattermost: support mattermost v3.9.x
|
||||||
|
* general: Add support for HTTP{S}_PROXY env variables (#162)
|
||||||
|
* discord: Strip custom emoji metadata (discord). Closes #148
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: Ignore error on private channel join (slack) Fixes #150
|
||||||
|
* mattermost: fix crash on reconnects when server is down. Closes #163
|
||||||
|
* irc: Relay messages starting with ! (irc). Closes #164
|
||||||
|
|
||||||
|
# v0.11.0
|
||||||
|
## New features
|
||||||
|
* general: reusing the same account on multiple gateways now also reuses the connection.
|
||||||
|
This is particuarly useful for irc. See #87
|
||||||
|
* general: the Name is now REQUIRED and needs to be UNIQUE for each gateway configuration
|
||||||
|
* telegram: Support edited messages (telegram). See #141
|
||||||
|
* mattermost: Add support for showing/hiding join/leave messages from mattermost. Closes #147
|
||||||
|
* mattermost: Reconnect on session removal/timeout (mattermost)
|
||||||
|
* mattermost: Support mattermost v3.8.x
|
||||||
|
* irc: Rejoin channel when kicked (irc).
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* mattermost: Remove space after nick (mattermost). Closes #142
|
||||||
|
* mattermost: Modify iconurl correctly (mattermost).
|
||||||
|
* irc: Fix join/leave regression (irc)
|
||||||
|
|
||||||
|
# v0.10.3
|
||||||
|
## Bugfix
|
||||||
|
* slack: Allow bot tokens for now without warning (slack). Closes #140 (fixes user_is_bot message on channel join)
|
||||||
|
|
||||||
|
# v0.10.2
|
||||||
|
## New features
|
||||||
|
* general: gops agent added. Allows for more debugging. See #134
|
||||||
|
* general: toml inline table support added for config file
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* all: vendored libs updated
|
||||||
|
|
||||||
|
## Changes
|
||||||
|
* general: add more informative messages on startup
|
||||||
|
|
||||||
|
# v0.10.1
|
||||||
|
## Bugfix
|
||||||
|
* gitter: Fix sending messages on new channel join.
|
||||||
|
|
||||||
|
# v0.10.0
|
||||||
|
## New features
|
||||||
|
* matrix: New protocol support added (https://matrix.org)
|
||||||
|
* mattermost: works with mattermost release v3.7.0
|
||||||
|
* discord: Replace role ids in mentions to role names (discord). Closes #133
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* mattermost: Add ReadTimeout to close lingering connections (mattermost). See #125
|
||||||
|
* gitter: Join rooms not already joined by the bot (gitter). See #135
|
||||||
|
* general: Fail when bridge is unable to join a channel (general)
|
||||||
|
|
||||||
|
## Changes
|
||||||
|
* telegram: Do not use HTML parsemode by default. Set ```MessageFormat="HTML"``` to use it. Closes #126
|
||||||
|
|
||||||
|
# v0.9.3
|
||||||
|
## New features
|
||||||
|
* API: rest interface to read / post messages (see API section in matterbridge.toml.sample)
|
||||||
|
|
||||||
|
## Bugfix
|
||||||
|
* slack: fix receiving messages from private channels #118
|
||||||
|
* slack: fix echo when using webhooks #119
|
||||||
|
* mattermost: reconnecting should work better now
|
||||||
|
* irc: keeps reconnecting (every 60 seconds) now after ping timeout/disconnects.
|
||||||
|
|
||||||
# v0.9.2
|
# v0.9.2
|
||||||
|
## New features
|
||||||
|
* slack: support private channels #118
|
||||||
|
|
||||||
## Bugfix
|
## Bugfix
|
||||||
* general: make ignorenicks work again #115
|
* general: make ignorenicks work again #115
|
||||||
* telegram: fix receiving from channels and groups #112
|
* telegram: fix receiving from channels and groups #112
|
||||||
@ -7,9 +694,6 @@
|
|||||||
* irc: update vendor (fixes some crashes) #117
|
* irc: update vendor (fixes some crashes) #117
|
||||||
* xmpp: fix tls by setting ServerName #114
|
* xmpp: fix tls by setting ServerName #114
|
||||||
|
|
||||||
## New features
|
|
||||||
* slack: support private channels #118
|
|
||||||
|
|
||||||
# v0.9.1
|
# v0.9.1
|
||||||
## New features
|
## New features
|
||||||
* Rocket.Chat: New protocol support added (https://rocket.chat)
|
* Rocket.Chat: New protocol support added (https://rocket.chat)
|
||||||
|
27
ci/bintray.sh
Executable file
@ -0,0 +1,27 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
go version | grep go1.11 || exit
|
||||||
|
VERSION=$(git describe --tags)
|
||||||
|
mkdir ci/binaries
|
||||||
|
GOOS=windows GOARCH=amd64 go build -ldflags "-s -w -X main.githash=$(git log --pretty=format:'%h' -n 1)" -o ci/binaries/matterbridge-$VERSION-windows-amd64.exe
|
||||||
|
GOOS=linux GOARCH=amd64 go build -ldflags "-s -w -X main.githash=$(git log --pretty=format:'%h' -n 1)" -o ci/binaries/matterbridge-$VERSION-linux-amd64
|
||||||
|
GOOS=linux GOARCH=arm go build -ldflags "-s -w -X main.githash=$(git log --pretty=format:'%h' -n 1)" -o ci/binaries/matterbridge-$VERSION-linux-arm
|
||||||
|
GOOS=darwin GOARCH=amd64 go build -ldflags "-s -w -X main.githash=$(git log --pretty=format:'%h' -n 1)" -o ci/binaries/matterbridge-$VERSION-darwin-amd64
|
||||||
|
cd ci
|
||||||
|
cat > deploy.json <<EOF
|
||||||
|
{
|
||||||
|
"package": {
|
||||||
|
"name": "Matterbridge",
|
||||||
|
"repo": "nightly",
|
||||||
|
"subject": "42wim"
|
||||||
|
},
|
||||||
|
"version": {
|
||||||
|
"name": "$VERSION"
|
||||||
|
},
|
||||||
|
"files":
|
||||||
|
[
|
||||||
|
{"includePattern": "ci/binaries/(.*)", "uploadPattern":"\$1"}
|
||||||
|
],
|
||||||
|
"publish": true
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
|
210
contrib/api.yaml
Normal file
@ -0,0 +1,210 @@
|
|||||||
|
openapi: 3.0.0
|
||||||
|
info:
|
||||||
|
contact: {}
|
||||||
|
description: A read/write API for the Matterbridge chat bridge.
|
||||||
|
license:
|
||||||
|
name: Apache 2.0
|
||||||
|
url: 'https://github.com/42wim/matterbridge/blob/master/LICENSE'
|
||||||
|
title: Matterbridge API
|
||||||
|
version: "0.1.0-oas3"
|
||||||
|
paths:
|
||||||
|
/health:
|
||||||
|
get:
|
||||||
|
responses:
|
||||||
|
'200':
|
||||||
|
description: OK
|
||||||
|
content:
|
||||||
|
'*/*':
|
||||||
|
schema:
|
||||||
|
type: string
|
||||||
|
summary: Checks if the server is alive.
|
||||||
|
/message:
|
||||||
|
post:
|
||||||
|
responses:
|
||||||
|
'200':
|
||||||
|
description: OK
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
$ref: '#/components/schemas/config.OutgoingMessageResponse'
|
||||||
|
summary: Create a message
|
||||||
|
requestBody:
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
$ref: '#/components/schemas/config.OutgoingMessage'
|
||||||
|
description: Message object to create
|
||||||
|
required: true
|
||||||
|
/messages:
|
||||||
|
get:
|
||||||
|
responses:
|
||||||
|
'200':
|
||||||
|
description: OK
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
items:
|
||||||
|
$ref: '#/components/schemas/config.IncomingMessage'
|
||||||
|
type: array
|
||||||
|
security:
|
||||||
|
- ApiKeyAuth: []
|
||||||
|
summary: List new messages
|
||||||
|
/stream:
|
||||||
|
get:
|
||||||
|
responses:
|
||||||
|
'200':
|
||||||
|
description: OK
|
||||||
|
content:
|
||||||
|
application/x-json-stream:
|
||||||
|
schema:
|
||||||
|
$ref: '#/components/schemas/config.IncomingMessage'
|
||||||
|
summary: Stream realtime messages
|
||||||
|
servers:
|
||||||
|
- url: /api
|
||||||
|
components:
|
||||||
|
securitySchemes:
|
||||||
|
bearerAuth:
|
||||||
|
type: http
|
||||||
|
scheme: bearer
|
||||||
|
schemas:
|
||||||
|
config.IncomingMessage:
|
||||||
|
properties:
|
||||||
|
avatar:
|
||||||
|
description: URL to an avatar image
|
||||||
|
example: >-
|
||||||
|
https://secure.gravatar.com/avatar/1234567890abcdef1234567890abcdef.jpg
|
||||||
|
type: string
|
||||||
|
event:
|
||||||
|
description: >-
|
||||||
|
A specific matterbridge event. (see
|
||||||
|
https://github.com/42wim/matterbridge/blob/master/bridge/config/config.go#L16)
|
||||||
|
type: string
|
||||||
|
gateway:
|
||||||
|
description: Name of the gateway as configured in matterbridge.toml
|
||||||
|
example: mygateway
|
||||||
|
type: string
|
||||||
|
text:
|
||||||
|
description: Content of the message
|
||||||
|
example: 'Testing, testing, 1-2-3.'
|
||||||
|
type: string
|
||||||
|
username:
|
||||||
|
description: Human-readable username
|
||||||
|
example: alice
|
||||||
|
type: string
|
||||||
|
account:
|
||||||
|
description: Unique account name of format "[protocol].[slug]" as defined in matterbridge.toml
|
||||||
|
example: slack.myteam
|
||||||
|
type: string
|
||||||
|
channel:
|
||||||
|
description: Human-readable channel name of sending bridge
|
||||||
|
example: test-channel
|
||||||
|
type: string
|
||||||
|
id:
|
||||||
|
description: Unique ID of message on the gateway
|
||||||
|
example: slack 1541361213.030700
|
||||||
|
type: string
|
||||||
|
parent_id:
|
||||||
|
description: Unique ID of a parent message, if threaded
|
||||||
|
example: slack 1541361213.030700
|
||||||
|
type: string
|
||||||
|
protocol:
|
||||||
|
description: Chat protocol of the sending bridge
|
||||||
|
example: slack
|
||||||
|
type: string
|
||||||
|
timestamp:
|
||||||
|
description: Timestamp of the message
|
||||||
|
example: "1541361213.030700"
|
||||||
|
type: string
|
||||||
|
userid:
|
||||||
|
description: Userid on the sending bridge
|
||||||
|
example: U4MCXJKNC
|
||||||
|
type: string
|
||||||
|
extra:
|
||||||
|
description: Extra data that doesn't fit in other fields (eg base64 encoded files)
|
||||||
|
type: object
|
||||||
|
config.OutgoingMessage:
|
||||||
|
properties:
|
||||||
|
avatar:
|
||||||
|
description: URL to an avatar image
|
||||||
|
example: >-
|
||||||
|
https://secure.gravatar.com/avatar/1234567890abcdef1234567890abcdef.jpg
|
||||||
|
type: string
|
||||||
|
event:
|
||||||
|
description: >-
|
||||||
|
A specific matterbridge event. (see
|
||||||
|
https://github.com/42wim/matterbridge/blob/master/bridge/config/config.go#L16)
|
||||||
|
example: ""
|
||||||
|
type: string
|
||||||
|
gateway:
|
||||||
|
description: Name of the gateway as configured in matterbridge.toml
|
||||||
|
example: mygateway
|
||||||
|
type: string
|
||||||
|
text:
|
||||||
|
description: Content of the message
|
||||||
|
example: 'Testing, testing, 1-2-3.'
|
||||||
|
type: string
|
||||||
|
username:
|
||||||
|
description: Human-readable username
|
||||||
|
example: alice
|
||||||
|
type: string
|
||||||
|
type: object
|
||||||
|
required:
|
||||||
|
- gateway
|
||||||
|
- text
|
||||||
|
- username
|
||||||
|
config.OutgoingMessageResponse:
|
||||||
|
properties:
|
||||||
|
avatar:
|
||||||
|
description: URL to an avatar image
|
||||||
|
example: >-
|
||||||
|
https://secure.gravatar.com/avatar/1234567890abcdef1234567890abcdef.jpg
|
||||||
|
type: string
|
||||||
|
event:
|
||||||
|
description: >-
|
||||||
|
A specific matterbridge event. (see
|
||||||
|
https://github.com/42wim/matterbridge/blob/master/bridge/config/config.go#L16)
|
||||||
|
example: ""
|
||||||
|
type: string
|
||||||
|
gateway:
|
||||||
|
description: Name of the gateway as configured in matterbridge.toml
|
||||||
|
example: mygateway
|
||||||
|
type: string
|
||||||
|
text:
|
||||||
|
description: Content of the message
|
||||||
|
example: 'Testing, testing, 1-2-3.'
|
||||||
|
type: string
|
||||||
|
username:
|
||||||
|
description: Human-readable username
|
||||||
|
example: alice
|
||||||
|
type: string
|
||||||
|
account:
|
||||||
|
description: fixed api account
|
||||||
|
example: api.local
|
||||||
|
type: string
|
||||||
|
channel:
|
||||||
|
description: fixed api channel
|
||||||
|
example: api
|
||||||
|
type: string
|
||||||
|
id:
|
||||||
|
example: ""
|
||||||
|
type: string
|
||||||
|
parent_id:
|
||||||
|
example: ""
|
||||||
|
type: string
|
||||||
|
protocol:
|
||||||
|
description: fixed api protocol
|
||||||
|
example: api
|
||||||
|
type: string
|
||||||
|
timestamp:
|
||||||
|
description: Timestamp of the message
|
||||||
|
example: "1541361213.030700"
|
||||||
|
type: string
|
||||||
|
userid:
|
||||||
|
example: ""
|
||||||
|
type: string
|
||||||
|
extra:
|
||||||
|
example: null
|
||||||
|
type: object
|
||||||
|
type: object
|
||||||
|
security:
|
||||||
|
- bearerAuth: []
|
2
contrib/example.tengo
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
text := import("text")
|
||||||
|
msgText=text.re_replace("matterbridge",msgText,"matterbridge (https://github.com/42wim/matterbridge)")
|
11
contrib/matterbridge.service
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=matterbridge
|
||||||
|
After=network.target
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
ExecStart=/usr/bin/matterbridge -conf /etc/matterbridge/bridge.toml
|
||||||
|
User=matterbridge
|
||||||
|
Group=matterbridge
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
9
docker/arm/Dockerfile
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
FROM alpine:edge as certs
|
||||||
|
RUN apk --update add ca-certificates
|
||||||
|
|
||||||
|
FROM scratch
|
||||||
|
ARG VERSION=1.12.3
|
||||||
|
COPY --from=certs /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/ca-certificates.crt
|
||||||
|
ADD https://github.com/42wim/matterbridge/releases/download/v${VERSION}/matterbridge-linux-arm /bin/matterbridge
|
||||||
|
RUN chmod +x /bin/matterbridge
|
||||||
|
ENTRYPOINT ["/bin/matterbridge"]
|
5
gateway/bench.tengo
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
text := import("text")
|
||||||
|
if text.re_match("blah",msgText) {
|
||||||
|
msgText="replaced by this"
|
||||||
|
msgUsername="fakeuser"
|
||||||
|
}
|
37
gateway/bridgemap/bridgemap.go
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
package bridgemap
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/api"
|
||||||
|
"github.com/42wim/matterbridge/bridge/discord"
|
||||||
|
"github.com/42wim/matterbridge/bridge/gitter"
|
||||||
|
"github.com/42wim/matterbridge/bridge/irc"
|
||||||
|
"github.com/42wim/matterbridge/bridge/matrix"
|
||||||
|
"github.com/42wim/matterbridge/bridge/mattermost"
|
||||||
|
"github.com/42wim/matterbridge/bridge/rocketchat"
|
||||||
|
"github.com/42wim/matterbridge/bridge/slack"
|
||||||
|
"github.com/42wim/matterbridge/bridge/sshchat"
|
||||||
|
"github.com/42wim/matterbridge/bridge/steam"
|
||||||
|
"github.com/42wim/matterbridge/bridge/telegram"
|
||||||
|
"github.com/42wim/matterbridge/bridge/whatsapp"
|
||||||
|
"github.com/42wim/matterbridge/bridge/xmpp"
|
||||||
|
"github.com/42wim/matterbridge/bridge/zulip"
|
||||||
|
)
|
||||||
|
|
||||||
|
var FullMap = map[string]bridge.Factory{
|
||||||
|
"api": api.New,
|
||||||
|
"discord": bdiscord.New,
|
||||||
|
"gitter": bgitter.New,
|
||||||
|
"irc": birc.New,
|
||||||
|
"mattermost": bmattermost.New,
|
||||||
|
"matrix": bmatrix.New,
|
||||||
|
"rocketchat": brocketchat.New,
|
||||||
|
"slack-legacy": bslack.NewLegacy,
|
||||||
|
"slack": bslack.New,
|
||||||
|
"sshchat": bsshchat.New,
|
||||||
|
"steam": bsteam.New,
|
||||||
|
"telegram": btelegram.New,
|
||||||
|
"whatsapp": bwhatsapp.New,
|
||||||
|
"xmpp": bxmpp.New,
|
||||||
|
"zulip": bzulip.New,
|
||||||
|
}
|
@ -1,187 +1,520 @@
|
|||||||
package gateway
|
package gateway
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/42wim/matterbridge/bridge"
|
"github.com/42wim/matterbridge/bridge"
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
log "github.com/Sirupsen/logrus"
|
"github.com/d5/tengo/script"
|
||||||
"reflect"
|
lru "github.com/hashicorp/golang-lru"
|
||||||
"strings"
|
"github.com/peterhellberg/emojilib"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Gateway struct {
|
type Gateway struct {
|
||||||
*config.Config
|
config.Config
|
||||||
MyConfig *config.Gateway
|
|
||||||
//Bridges []*bridge.Bridge
|
Router *Router
|
||||||
|
MyConfig *config.Gateway
|
||||||
Bridges map[string]*bridge.Bridge
|
Bridges map[string]*bridge.Bridge
|
||||||
ChannelsOut map[string][]string
|
Channels map[string]*config.ChannelInfo
|
||||||
ChannelsIn map[string][]string
|
|
||||||
ChannelOptions map[string]config.ChannelOptions
|
ChannelOptions map[string]config.ChannelOptions
|
||||||
Name string
|
|
||||||
Message chan config.Message
|
Message chan config.Message
|
||||||
|
Name string
|
||||||
|
Messages *lru.Cache
|
||||||
|
|
||||||
|
logger *logrus.Entry
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(cfg *config.Config, gateway *config.Gateway) *Gateway {
|
type BrMsgID struct {
|
||||||
gw := &Gateway{}
|
br *bridge.Bridge
|
||||||
gw.Name = gateway.Name
|
ID string
|
||||||
gw.Config = cfg
|
ChannelID string
|
||||||
gw.MyConfig = gateway
|
}
|
||||||
gw.Message = make(chan config.Message)
|
|
||||||
gw.Bridges = make(map[string]*bridge.Bridge)
|
const apiProtocol = "api"
|
||||||
|
|
||||||
|
// New creates a new Gateway object associated with the specified router and
|
||||||
|
// following the given configuration.
|
||||||
|
func New(rootLogger *logrus.Logger, cfg *config.Gateway, r *Router) *Gateway {
|
||||||
|
logger := rootLogger.WithFields(logrus.Fields{"prefix": "gateway"})
|
||||||
|
|
||||||
|
cache, _ := lru.New(5000)
|
||||||
|
gw := &Gateway{
|
||||||
|
Channels: make(map[string]*config.ChannelInfo),
|
||||||
|
Message: r.Message,
|
||||||
|
Router: r,
|
||||||
|
Bridges: make(map[string]*bridge.Bridge),
|
||||||
|
Config: r.Config,
|
||||||
|
Messages: cache,
|
||||||
|
logger: logger,
|
||||||
|
}
|
||||||
|
if err := gw.AddConfig(cfg); err != nil {
|
||||||
|
logger.Errorf("Failed to add configuration to gateway: %#v", err)
|
||||||
|
}
|
||||||
return gw
|
return gw
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *Gateway) AddBridge(cfg *config.Bridge) error {
|
// FindCanonicalMsgID returns the ID under which a message was stored in the cache.
|
||||||
for _, br := range gw.Bridges {
|
func (gw *Gateway) FindCanonicalMsgID(protocol string, mID string) string {
|
||||||
if br.Account == cfg.Account {
|
ID := protocol + " " + mID
|
||||||
return nil
|
if gw.Messages.Contains(ID) {
|
||||||
}
|
return mID
|
||||||
}
|
}
|
||||||
log.Infof("Starting bridge: %s ", cfg.Account)
|
|
||||||
br := bridge.New(gw.Config, cfg, gw.Message)
|
// If not keyed, iterate through cache for downstream, and infer upstream.
|
||||||
gw.Bridges[cfg.Account] = br
|
for _, mid := range gw.Messages.Keys() {
|
||||||
err := br.Connect()
|
v, _ := gw.Messages.Peek(mid)
|
||||||
if err != nil {
|
ids := v.([]*BrMsgID)
|
||||||
return fmt.Errorf("Bridge %s failed to start: %v", br.Account, err)
|
for _, downstreamMsgObj := range ids {
|
||||||
}
|
if ID == downstreamMsgObj.ID {
|
||||||
exists := make(map[string]bool)
|
return strings.Replace(mid.(string), protocol+" ", "", 1)
|
||||||
for _, channel := range append(gw.ChannelsOut[br.Account], gw.ChannelsIn[br.Account]...) {
|
|
||||||
if !exists[br.Account+channel] {
|
|
||||||
mychannel := channel
|
|
||||||
log.Infof("%s: joining %s", br.Account, channel)
|
|
||||||
if br.Protocol == "irc" && gw.ChannelOptions[br.Account+channel].Key != "" {
|
|
||||||
log.Debugf("using key %s for channel %s", gw.ChannelOptions[br.Account+channel].Key, channel)
|
|
||||||
mychannel = mychannel + " " + gw.ChannelOptions[br.Account+channel].Key
|
|
||||||
}
|
}
|
||||||
br.JoinChannel(mychannel)
|
|
||||||
exists[br.Account+channel] = true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// AddBridge sets up a new bridge in the gateway object with the specified configuration.
|
||||||
|
func (gw *Gateway) AddBridge(cfg *config.Bridge) error {
|
||||||
|
br := gw.Router.getBridge(cfg.Account)
|
||||||
|
if br == nil {
|
||||||
|
br = bridge.New(cfg)
|
||||||
|
br.Config = gw.Router.Config
|
||||||
|
br.General = &gw.BridgeValues().General
|
||||||
|
br.Log = gw.logger.WithFields(logrus.Fields{"prefix": br.Protocol})
|
||||||
|
brconfig := &bridge.Config{
|
||||||
|
Remote: gw.Message,
|
||||||
|
Bridge: br,
|
||||||
|
}
|
||||||
|
// add the actual bridger for this protocol to this bridge using the bridgeMap
|
||||||
|
if _, ok := gw.Router.BridgeMap[br.Protocol]; !ok {
|
||||||
|
gw.logger.Fatalf("Incorrect protocol %s specified in gateway configuration %s, exiting.", br.Protocol, cfg.Account)
|
||||||
|
}
|
||||||
|
br.Bridger = gw.Router.BridgeMap[br.Protocol](brconfig)
|
||||||
|
}
|
||||||
|
gw.mapChannelsToBridge(br)
|
||||||
|
gw.Bridges[cfg.Account] = br
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *Gateway) Start() error {
|
// AddConfig associates a new configuration with the gateway object.
|
||||||
gw.mapChannels()
|
func (gw *Gateway) AddConfig(cfg *config.Gateway) error {
|
||||||
|
gw.Name = cfg.Name
|
||||||
|
gw.MyConfig = cfg
|
||||||
|
if err := gw.mapChannels(); err != nil {
|
||||||
|
gw.logger.Errorf("mapChannels() failed: %s", err)
|
||||||
|
}
|
||||||
for _, br := range append(gw.MyConfig.In, append(gw.MyConfig.InOut, gw.MyConfig.Out...)...) {
|
for _, br := range append(gw.MyConfig.In, append(gw.MyConfig.InOut, gw.MyConfig.Out...)...) {
|
||||||
|
br := br //scopelint
|
||||||
err := gw.AddBridge(&br)
|
err := gw.AddBridge(&br)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
go gw.handleReceive()
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *Gateway) handleReceive() {
|
func (gw *Gateway) mapChannelsToBridge(br *bridge.Bridge) {
|
||||||
for {
|
for ID, channel := range gw.Channels {
|
||||||
select {
|
if br.Account == channel.Account {
|
||||||
case msg := <-gw.Message:
|
br.Channels[ID] = *channel
|
||||||
if !gw.ignoreMessage(&msg) {
|
}
|
||||||
for _, br := range gw.Bridges {
|
}
|
||||||
gw.handleMessage(msg, br)
|
}
|
||||||
}
|
|
||||||
|
func (gw *Gateway) reconnectBridge(br *bridge.Bridge) {
|
||||||
|
if err := br.Disconnect(); err != nil {
|
||||||
|
gw.logger.Errorf("Disconnect() %s failed: %s", br.Account, err)
|
||||||
|
}
|
||||||
|
time.Sleep(time.Second * 5)
|
||||||
|
RECONNECT:
|
||||||
|
gw.logger.Infof("Reconnecting %s", br.Account)
|
||||||
|
err := br.Connect()
|
||||||
|
if err != nil {
|
||||||
|
gw.logger.Errorf("Reconnection failed: %s. Trying again in 60 seconds", err)
|
||||||
|
time.Sleep(time.Second * 60)
|
||||||
|
goto RECONNECT
|
||||||
|
}
|
||||||
|
br.Joined = make(map[string]bool)
|
||||||
|
if err := br.JoinChannels(); err != nil {
|
||||||
|
gw.logger.Errorf("JoinChannels() %s failed: %s", br.Account, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gw *Gateway) mapChannelConfig(cfg []config.Bridge, direction string) {
|
||||||
|
for _, br := range cfg {
|
||||||
|
if isAPI(br.Account) {
|
||||||
|
br.Channel = apiProtocol
|
||||||
|
}
|
||||||
|
// make sure to lowercase irc channels in config #348
|
||||||
|
if strings.HasPrefix(br.Account, "irc.") {
|
||||||
|
br.Channel = strings.ToLower(br.Channel)
|
||||||
|
}
|
||||||
|
if strings.HasPrefix(br.Account, "mattermost.") && strings.HasPrefix(br.Channel, "#") {
|
||||||
|
gw.logger.Errorf("Mattermost channels do not start with a #: remove the # in %s", br.Channel)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
if strings.HasPrefix(br.Account, "zulip.") && !strings.Contains(br.Channel, "/topic:") {
|
||||||
|
gw.logger.Errorf("Breaking change, since matterbridge 1.14.0 zulip channels need to specify the topic with channel/topic:mytopic in %s of %s", br.Channel, br.Account)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
ID := br.Channel + br.Account
|
||||||
|
if _, ok := gw.Channels[ID]; !ok {
|
||||||
|
channel := &config.ChannelInfo{
|
||||||
|
Name: br.Channel,
|
||||||
|
Direction: direction,
|
||||||
|
ID: ID,
|
||||||
|
Options: br.Options,
|
||||||
|
Account: br.Account,
|
||||||
|
SameChannel: make(map[string]bool),
|
||||||
|
}
|
||||||
|
channel.SameChannel[gw.Name] = br.SameChannel
|
||||||
|
gw.Channels[channel.ID] = channel
|
||||||
|
} else {
|
||||||
|
// if we already have a key and it's not our current direction it means we have a bidirectional inout
|
||||||
|
if gw.Channels[ID].Direction != direction {
|
||||||
|
gw.Channels[ID].Direction = "inout"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
gw.Channels[ID].SameChannel[gw.Name] = br.SameChannel
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *Gateway) mapChannels() error {
|
func (gw *Gateway) mapChannels() error {
|
||||||
options := make(map[string]config.ChannelOptions)
|
gw.mapChannelConfig(gw.MyConfig.In, "in")
|
||||||
m := make(map[string][]string)
|
gw.mapChannelConfig(gw.MyConfig.Out, "out")
|
||||||
for _, br := range gw.MyConfig.Out {
|
gw.mapChannelConfig(gw.MyConfig.InOut, "inout")
|
||||||
m[br.Account] = append(m[br.Account], br.Channel)
|
|
||||||
options[br.Account+br.Channel] = br.Options
|
|
||||||
}
|
|
||||||
gw.ChannelsOut = m
|
|
||||||
m = nil
|
|
||||||
m = make(map[string][]string)
|
|
||||||
for _, br := range gw.MyConfig.In {
|
|
||||||
m[br.Account] = append(m[br.Account], br.Channel)
|
|
||||||
options[br.Account+br.Channel] = br.Options
|
|
||||||
}
|
|
||||||
gw.ChannelsIn = m
|
|
||||||
for _, br := range gw.MyConfig.InOut {
|
|
||||||
gw.ChannelsIn[br.Account] = append(gw.ChannelsIn[br.Account], br.Channel)
|
|
||||||
gw.ChannelsOut[br.Account] = append(gw.ChannelsOut[br.Account], br.Channel)
|
|
||||||
options[br.Account+br.Channel] = br.Options
|
|
||||||
}
|
|
||||||
gw.ChannelOptions = options
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *Gateway) getDestChannel(msg *config.Message, dest string) []string {
|
func (gw *Gateway) getDestChannel(msg *config.Message, dest bridge.Bridge) []config.ChannelInfo {
|
||||||
channels := gw.ChannelsIn[msg.Account]
|
var channels []config.ChannelInfo
|
||||||
// broadcast to every out channel (irc QUIT)
|
|
||||||
if msg.Event == config.EVENT_JOIN_LEAVE && msg.Channel == "" {
|
|
||||||
return gw.ChannelsOut[dest]
|
|
||||||
}
|
|
||||||
for _, channel := range channels {
|
|
||||||
if channel == msg.Channel {
|
|
||||||
return gw.ChannelsOut[dest]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return []string{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (gw *Gateway) handleMessage(msg config.Message, dest *bridge.Bridge) {
|
// for messages received from the api check that the gateway is the specified one
|
||||||
// only relay join/part when configged
|
if msg.Protocol == apiProtocol && gw.Name != msg.Gateway {
|
||||||
if msg.Event == config.EVENT_JOIN_LEAVE && !gw.Bridges[dest.Account].Config.ShowJoinPart {
|
return channels
|
||||||
return
|
|
||||||
}
|
}
|
||||||
originchannel := msg.Channel
|
|
||||||
channels := gw.getDestChannel(&msg, dest.Account)
|
// discord join/leave is for the whole bridge, isn't a per channel join/leave
|
||||||
for _, channel := range channels {
|
if msg.Event == config.EventJoinLeave && getProtocol(msg) == "discord" && msg.Channel == "" {
|
||||||
// do not send the message to the bridge we come from if also the channel is the same
|
for _, channel := range gw.Channels {
|
||||||
if msg.Account == dest.Account && channel == originchannel {
|
if channel.Account == dest.Account && strings.Contains(channel.Direction, "out") &&
|
||||||
|
gw.validGatewayDest(msg) {
|
||||||
|
channels = append(channels, *channel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return channels
|
||||||
|
}
|
||||||
|
|
||||||
|
// irc quit is for the whole bridge, isn't a per channel quit.
|
||||||
|
// channel is empty when we quit
|
||||||
|
if msg.Event == config.EventJoinLeave && getProtocol(msg) == "irc" && msg.Channel == "" {
|
||||||
|
// if we only have one channel on this irc bridge it's got to be the sending one.
|
||||||
|
// don't send it back
|
||||||
|
if dest.Account == msg.Account && len(dest.Channels) == 1 && dest.Protocol == "irc" {
|
||||||
|
return channels
|
||||||
|
}
|
||||||
|
for _, channel := range gw.Channels {
|
||||||
|
if channel.Account == dest.Account && strings.Contains(channel.Direction, "out") &&
|
||||||
|
gw.validGatewayDest(msg) {
|
||||||
|
channels = append(channels, *channel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return channels
|
||||||
|
}
|
||||||
|
|
||||||
|
// if source channel is in only, do nothing
|
||||||
|
for _, channel := range gw.Channels {
|
||||||
|
// lookup the channel from the message
|
||||||
|
if channel.ID == getChannelID(msg) {
|
||||||
|
// we only have destinations if the original message is from an "in" (sending) channel
|
||||||
|
if !strings.Contains(channel.Direction, "in") {
|
||||||
|
return channels
|
||||||
|
}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
msg.Channel = channel
|
}
|
||||||
if msg.Channel == "" {
|
for _, channel := range gw.Channels {
|
||||||
log.Debug("empty channel")
|
if _, ok := gw.Channels[getChannelID(msg)]; !ok {
|
||||||
return
|
continue
|
||||||
}
|
}
|
||||||
log.Debugf("Sending %#v from %s (%s) to %s (%s)", msg, msg.Account, originchannel, dest.Account, channel)
|
|
||||||
gw.modifyUsername(&msg, dest)
|
// do samechannelgateway logic
|
||||||
err := dest.Send(msg)
|
if channel.SameChannel[msg.Gateway] {
|
||||||
if err != nil {
|
if msg.Channel == channel.Name && msg.Account != dest.Account {
|
||||||
fmt.Println(err)
|
channels = append(channels, *channel)
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if strings.Contains(channel.Direction, "out") && channel.Account == dest.Account && gw.validGatewayDest(msg) {
|
||||||
|
channels = append(channels, *channel)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return channels
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gw *Gateway) getDestMsgID(msgID string, dest *bridge.Bridge, channel *config.ChannelInfo) string {
|
||||||
|
if res, ok := gw.Messages.Get(msgID); ok {
|
||||||
|
IDs := res.([]*BrMsgID)
|
||||||
|
for _, id := range IDs {
|
||||||
|
// check protocol, bridge name and channelname
|
||||||
|
// for people that reuse the same bridge multiple times. see #342
|
||||||
|
if dest.Protocol == id.br.Protocol && dest.Name == id.br.Name && channel.ID == id.ChannelID {
|
||||||
|
return strings.Replace(id.ID, dest.Protocol+" ", "", 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// ignoreTextEmpty returns true if we need to ignore a message with an empty text.
|
||||||
|
func (gw *Gateway) ignoreTextEmpty(msg *config.Message) bool {
|
||||||
|
if msg.Text != "" {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if msg.Event == config.EventUserTyping {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
// we have an attachment or actual bytes, do not ignore
|
||||||
|
if msg.Extra != nil &&
|
||||||
|
(msg.Extra["attachments"] != nil ||
|
||||||
|
len(msg.Extra["file"]) > 0 ||
|
||||||
|
len(msg.Extra[config.EventFileFailureSize]) > 0) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
gw.logger.Debugf("ignoring empty message %#v from %s", msg, msg.Account)
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *Gateway) ignoreMessage(msg *config.Message) bool {
|
func (gw *Gateway) ignoreMessage(msg *config.Message) bool {
|
||||||
for _, entry := range strings.Fields(gw.Bridges[msg.Account].Config.IgnoreNicks) {
|
// if we don't have the bridge, ignore it
|
||||||
if msg.Username == entry {
|
if _, ok := gw.Bridges[msg.Account]; !ok {
|
||||||
log.Debugf("ignoring %s from %s", msg.Username, msg.Account)
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
igNicks := strings.Fields(gw.Bridges[msg.Account].GetString("IgnoreNicks"))
|
||||||
|
igMessages := strings.Fields(gw.Bridges[msg.Account].GetString("IgnoreMessages"))
|
||||||
|
if gw.ignoreTextEmpty(msg) || gw.ignoreText(msg.Username, igNicks) || gw.ignoreText(msg.Text, igMessages) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gw *Gateway) modifyUsername(msg *config.Message, dest *bridge.Bridge) string {
|
||||||
|
br := gw.Bridges[msg.Account]
|
||||||
|
msg.Protocol = br.Protocol
|
||||||
|
if dest.GetBool("StripNick") {
|
||||||
|
re := regexp.MustCompile("[^a-zA-Z0-9]+")
|
||||||
|
msg.Username = re.ReplaceAllString(msg.Username, "")
|
||||||
|
}
|
||||||
|
nick := dest.GetString("RemoteNickFormat")
|
||||||
|
|
||||||
|
// loop to replace nicks
|
||||||
|
for _, outer := range br.GetStringSlice2D("ReplaceNicks") {
|
||||||
|
search := outer[0]
|
||||||
|
replace := outer[1]
|
||||||
|
// TODO move compile to bridge init somewhere
|
||||||
|
re, err := regexp.Compile(search)
|
||||||
|
if err != nil {
|
||||||
|
gw.logger.Errorf("regexp in %s failed: %s", msg.Account, err)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
msg.Username = re.ReplaceAllString(msg.Username, replace)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(msg.Username) > 0 {
|
||||||
|
// fix utf-8 issue #193
|
||||||
|
i := 0
|
||||||
|
for index := range msg.Username {
|
||||||
|
if i == 1 {
|
||||||
|
i = index
|
||||||
|
break
|
||||||
|
}
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
nick = strings.Replace(nick, "{NOPINGNICK}", msg.Username[:i]+""+msg.Username[i:], -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
nick = strings.Replace(nick, "{BRIDGE}", br.Name, -1)
|
||||||
|
nick = strings.Replace(nick, "{PROTOCOL}", br.Protocol, -1)
|
||||||
|
nick = strings.Replace(nick, "{GATEWAY}", gw.Name, -1)
|
||||||
|
nick = strings.Replace(nick, "{LABEL}", br.GetString("Label"), -1)
|
||||||
|
nick = strings.Replace(nick, "{NICK}", msg.Username, -1)
|
||||||
|
nick = strings.Replace(nick, "{CHANNEL}", msg.Channel, -1)
|
||||||
|
return nick
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gw *Gateway) modifyAvatar(msg *config.Message, dest *bridge.Bridge) string {
|
||||||
|
iconurl := dest.GetString("IconURL")
|
||||||
|
iconurl = strings.Replace(iconurl, "{NICK}", msg.Username, -1)
|
||||||
|
if msg.Avatar == "" {
|
||||||
|
msg.Avatar = iconurl
|
||||||
|
}
|
||||||
|
return msg.Avatar
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gw *Gateway) modifyMessage(msg *config.Message) {
|
||||||
|
if err := modifyMessageTengo(gw.BridgeValues().General.TengoModifyMessage, msg); err != nil {
|
||||||
|
gw.logger.Errorf("TengoModifyMessage failed: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// replace :emoji: to unicode
|
||||||
|
msg.Text = emojilib.Replace(msg.Text)
|
||||||
|
|
||||||
|
br := gw.Bridges[msg.Account]
|
||||||
|
// loop to replace messages
|
||||||
|
for _, outer := range br.GetStringSlice2D("ReplaceMessages") {
|
||||||
|
search := outer[0]
|
||||||
|
replace := outer[1]
|
||||||
|
// TODO move compile to bridge init somewhere
|
||||||
|
re, err := regexp.Compile(search)
|
||||||
|
if err != nil {
|
||||||
|
gw.logger.Errorf("regexp in %s failed: %s", msg.Account, err)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
msg.Text = re.ReplaceAllString(msg.Text, replace)
|
||||||
|
}
|
||||||
|
|
||||||
|
gw.handleExtractNicks(msg)
|
||||||
|
|
||||||
|
// messages from api have Gateway specified, don't overwrite
|
||||||
|
if msg.Protocol != apiProtocol {
|
||||||
|
msg.Gateway = gw.Name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendMessage sends a message (with specified parentID) to the channel on the selected
|
||||||
|
// destination bridge and returns a message ID or an error.
|
||||||
|
func (gw *Gateway) SendMessage(
|
||||||
|
rmsg *config.Message,
|
||||||
|
dest *bridge.Bridge,
|
||||||
|
channel *config.ChannelInfo,
|
||||||
|
canonicalParentMsgID string,
|
||||||
|
) (string, error) {
|
||||||
|
msg := *rmsg
|
||||||
|
// Only send the avatar download event to ourselves.
|
||||||
|
if msg.Event == config.EventAvatarDownload {
|
||||||
|
if channel.ID != getChannelID(rmsg) {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// do not send to ourself for any other event
|
||||||
|
if channel.ID == getChannelID(rmsg) {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Too noisy to log like other events
|
||||||
|
if msg.Event != config.EventUserTyping {
|
||||||
|
gw.logger.Debugf("=> Sending %#v from %s (%s) to %s (%s)", msg, msg.Account, rmsg.Channel, dest.Account, channel.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
msg.Channel = channel.Name
|
||||||
|
msg.Avatar = gw.modifyAvatar(rmsg, dest)
|
||||||
|
msg.Username = gw.modifyUsername(rmsg, dest)
|
||||||
|
|
||||||
|
msg.ID = gw.getDestMsgID(rmsg.Protocol+" "+rmsg.ID, dest, channel)
|
||||||
|
|
||||||
|
// for api we need originchannel as channel
|
||||||
|
if dest.Protocol == apiProtocol {
|
||||||
|
msg.Channel = rmsg.Channel
|
||||||
|
}
|
||||||
|
|
||||||
|
msg.ParentID = gw.getDestMsgID(rmsg.Protocol+" "+canonicalParentMsgID, dest, channel)
|
||||||
|
if msg.ParentID == "" {
|
||||||
|
msg.ParentID = canonicalParentMsgID
|
||||||
|
}
|
||||||
|
|
||||||
|
// if the parentID is still empty and we have a parentID set in the original message
|
||||||
|
// this means that we didn't find it in the cache so set it "msg-parent-not-found"
|
||||||
|
if msg.ParentID == "" && rmsg.ParentID != "" {
|
||||||
|
msg.ParentID = "msg-parent-not-found"
|
||||||
|
}
|
||||||
|
|
||||||
|
// if we are using mattermost plugin account, send messages to MattermostPlugin channel
|
||||||
|
// that can be picked up by the mattermost matterbridge plugin
|
||||||
|
if dest.Account == "mattermost.plugin" {
|
||||||
|
gw.Router.MattermostPlugin <- msg
|
||||||
|
}
|
||||||
|
|
||||||
|
mID, err := dest.Send(msg)
|
||||||
|
if err != nil {
|
||||||
|
return mID, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// append the message ID (mID) from this bridge (dest) to our brMsgIDs slice
|
||||||
|
if mID != "" {
|
||||||
|
gw.logger.Debugf("mID %s: %s", dest.Account, mID)
|
||||||
|
return mID, nil
|
||||||
|
//brMsgIDs = append(brMsgIDs, &BrMsgID{dest, dest.Protocol + " " + mID, channel.ID})
|
||||||
|
}
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gw *Gateway) validGatewayDest(msg *config.Message) bool {
|
||||||
|
return msg.Gateway == gw.Name
|
||||||
|
}
|
||||||
|
|
||||||
|
func getChannelID(msg *config.Message) string {
|
||||||
|
return msg.Channel + msg.Account
|
||||||
|
}
|
||||||
|
|
||||||
|
func isAPI(account string) bool {
|
||||||
|
return strings.HasPrefix(account, "api.")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ignoreText returns true if text matches any of the input regexes.
|
||||||
|
func (gw *Gateway) ignoreText(text string, input []string) bool {
|
||||||
|
for _, entry := range input {
|
||||||
|
if entry == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// TODO do not compile regexps everytime
|
||||||
|
re, err := regexp.Compile(entry)
|
||||||
|
if err != nil {
|
||||||
|
gw.logger.Errorf("incorrect regexp %s", entry)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if re.MatchString(text) {
|
||||||
|
gw.logger.Debugf("matching %s. ignoring %s", entry, text)
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *Gateway) modifyMessage(msg *config.Message, dest *bridge.Bridge) {
|
func getProtocol(msg *config.Message) string {
|
||||||
val := reflect.ValueOf(gw.Config).Elem()
|
p := strings.Split(msg.Account, ".")
|
||||||
for i := 0; i < val.NumField(); i++ {
|
return p[0]
|
||||||
typeField := val.Type().Field(i)
|
|
||||||
// look for the protocol map (both lowercase)
|
|
||||||
if strings.ToLower(typeField.Name) == dest.Protocol {
|
|
||||||
// get the Protocol struct from the map
|
|
||||||
protoCfg := val.Field(i).MapIndex(reflect.ValueOf(dest.Name))
|
|
||||||
//config.SetNickFormat(msg, protoCfg.Interface().(config.Protocol))
|
|
||||||
val.Field(i).SetMapIndex(reflect.ValueOf(dest.Name), protoCfg)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *Gateway) modifyUsername(msg *config.Message, dest *bridge.Bridge) {
|
func modifyMessageTengo(filename string, msg *config.Message) error {
|
||||||
br := gw.Bridges[msg.Account]
|
if filename == "" {
|
||||||
nick := gw.Config.General.RemoteNickFormat
|
return nil
|
||||||
if nick == "" {
|
|
||||||
nick = dest.Config.RemoteNickFormat
|
|
||||||
}
|
}
|
||||||
nick = strings.Replace(nick, "{NICK}", msg.Username, -1)
|
res, err := ioutil.ReadFile(filename)
|
||||||
nick = strings.Replace(nick, "{BRIDGE}", br.Name, -1)
|
if err != nil {
|
||||||
nick = strings.Replace(nick, "{PROTOCOL}", br.Protocol, -1)
|
return err
|
||||||
msg.Username = nick
|
}
|
||||||
|
s := script.New(res)
|
||||||
|
_ = s.Add("msgText", msg.Text)
|
||||||
|
_ = s.Add("msgUsername", msg.Username)
|
||||||
|
_ = s.Add("msgAccount", msg.Account)
|
||||||
|
_ = s.Add("msgChannel", msg.Channel)
|
||||||
|
c, err := s.Compile()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := c.Run(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
msg.Text = c.Get("msgText").String()
|
||||||
|
msg.Username = c.Get("msgUsername").String()
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
529
gateway/gateway_test.go
Normal file
@ -0,0 +1,529 @@
|
|||||||
|
package gateway
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"strconv"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/gateway/bridgemap"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/suite"
|
||||||
|
)
|
||||||
|
|
||||||
|
var testconfig = []byte(`
|
||||||
|
[irc.freenode]
|
||||||
|
[mattermost.test]
|
||||||
|
[gitter.42wim]
|
||||||
|
[discord.test]
|
||||||
|
[slack.test]
|
||||||
|
|
||||||
|
[[gateway]]
|
||||||
|
name = "bridge1"
|
||||||
|
enable=true
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account = "irc.freenode"
|
||||||
|
channel = "#wimtesting"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="gitter.42wim"
|
||||||
|
channel="42wim/testroom"
|
||||||
|
#channel="matterbridge/Lobby"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account = "discord.test"
|
||||||
|
channel = "general"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="slack.test"
|
||||||
|
channel="testing"
|
||||||
|
`)
|
||||||
|
|
||||||
|
var testconfig2 = []byte(`
|
||||||
|
[irc.freenode]
|
||||||
|
[mattermost.test]
|
||||||
|
[gitter.42wim]
|
||||||
|
[discord.test]
|
||||||
|
[slack.test]
|
||||||
|
|
||||||
|
[[gateway]]
|
||||||
|
name = "bridge1"
|
||||||
|
enable=true
|
||||||
|
|
||||||
|
[[gateway.in]]
|
||||||
|
account = "irc.freenode"
|
||||||
|
channel = "#wimtesting"
|
||||||
|
|
||||||
|
[[gateway.in]]
|
||||||
|
account="gitter.42wim"
|
||||||
|
channel="42wim/testroom"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account = "discord.test"
|
||||||
|
channel = "general"
|
||||||
|
|
||||||
|
[[gateway.out]]
|
||||||
|
account="slack.test"
|
||||||
|
channel="testing"
|
||||||
|
[[gateway]]
|
||||||
|
name = "bridge2"
|
||||||
|
enable=true
|
||||||
|
|
||||||
|
[[gateway.in]]
|
||||||
|
account = "irc.freenode"
|
||||||
|
channel = "#wimtesting2"
|
||||||
|
|
||||||
|
[[gateway.out]]
|
||||||
|
account="gitter.42wim"
|
||||||
|
channel="42wim/testroom"
|
||||||
|
|
||||||
|
[[gateway.out]]
|
||||||
|
account = "discord.test"
|
||||||
|
channel = "general2"
|
||||||
|
`)
|
||||||
|
|
||||||
|
var testconfig3 = []byte(`
|
||||||
|
[irc.zzz]
|
||||||
|
[telegram.zzz]
|
||||||
|
[slack.zzz]
|
||||||
|
[[gateway]]
|
||||||
|
name="bridge"
|
||||||
|
enable=true
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="irc.zzz"
|
||||||
|
channel="#main"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="telegram.zzz"
|
||||||
|
channel="-1111111111111"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="slack.zzz"
|
||||||
|
channel="irc"
|
||||||
|
|
||||||
|
[[gateway]]
|
||||||
|
name="announcements"
|
||||||
|
enable=true
|
||||||
|
|
||||||
|
[[gateway.in]]
|
||||||
|
account="telegram.zzz"
|
||||||
|
channel="-2222222222222"
|
||||||
|
|
||||||
|
[[gateway.out]]
|
||||||
|
account="irc.zzz"
|
||||||
|
channel="#main"
|
||||||
|
|
||||||
|
[[gateway.out]]
|
||||||
|
account="irc.zzz"
|
||||||
|
channel="#main-help"
|
||||||
|
|
||||||
|
[[gateway.out]]
|
||||||
|
account="telegram.zzz"
|
||||||
|
channel="--333333333333"
|
||||||
|
|
||||||
|
[[gateway.out]]
|
||||||
|
account="slack.zzz"
|
||||||
|
channel="general"
|
||||||
|
|
||||||
|
[[gateway]]
|
||||||
|
name="bridge2"
|
||||||
|
enable=true
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="irc.zzz"
|
||||||
|
channel="#main-help"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="telegram.zzz"
|
||||||
|
channel="--444444444444"
|
||||||
|
|
||||||
|
|
||||||
|
[[gateway]]
|
||||||
|
name="bridge3"
|
||||||
|
enable=true
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="irc.zzz"
|
||||||
|
channel="#main-telegram"
|
||||||
|
|
||||||
|
[[gateway.inout]]
|
||||||
|
account="telegram.zzz"
|
||||||
|
channel="--333333333333"
|
||||||
|
`)
|
||||||
|
|
||||||
|
const (
|
||||||
|
ircTestAccount = "irc.zzz"
|
||||||
|
tgTestAccount = "telegram.zzz"
|
||||||
|
slackTestAccount = "slack.zzz"
|
||||||
|
)
|
||||||
|
|
||||||
|
func maketestRouter(input []byte) *Router {
|
||||||
|
logger := logrus.New()
|
||||||
|
logger.SetOutput(ioutil.Discard)
|
||||||
|
cfg := config.NewConfigFromString(logger, input)
|
||||||
|
r, err := NewRouter(logger, cfg, bridgemap.FullMap)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
func TestNewRouter(t *testing.T) {
|
||||||
|
r := maketestRouter(testconfig)
|
||||||
|
assert.Equal(t, 1, len(r.Gateways))
|
||||||
|
assert.Equal(t, 4, len(r.Gateways["bridge1"].Bridges))
|
||||||
|
assert.Equal(t, 4, len(r.Gateways["bridge1"].Channels))
|
||||||
|
|
||||||
|
r = maketestRouter(testconfig2)
|
||||||
|
assert.Equal(t, 2, len(r.Gateways))
|
||||||
|
assert.Equal(t, 4, len(r.Gateways["bridge1"].Bridges))
|
||||||
|
assert.Equal(t, 3, len(r.Gateways["bridge2"].Bridges))
|
||||||
|
assert.Equal(t, 4, len(r.Gateways["bridge1"].Channels))
|
||||||
|
assert.Equal(t, 3, len(r.Gateways["bridge2"].Channels))
|
||||||
|
assert.Equal(t, &config.ChannelInfo{
|
||||||
|
Name: "42wim/testroom",
|
||||||
|
Direction: "out",
|
||||||
|
ID: "42wim/testroomgitter.42wim",
|
||||||
|
Account: "gitter.42wim",
|
||||||
|
SameChannel: map[string]bool{"bridge2": false},
|
||||||
|
}, r.Gateways["bridge2"].Channels["42wim/testroomgitter.42wim"])
|
||||||
|
assert.Equal(t, &config.ChannelInfo{
|
||||||
|
Name: "42wim/testroom",
|
||||||
|
Direction: "in",
|
||||||
|
ID: "42wim/testroomgitter.42wim",
|
||||||
|
Account: "gitter.42wim",
|
||||||
|
SameChannel: map[string]bool{"bridge1": false},
|
||||||
|
}, r.Gateways["bridge1"].Channels["42wim/testroomgitter.42wim"])
|
||||||
|
assert.Equal(t, &config.ChannelInfo{
|
||||||
|
Name: "general",
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "generaldiscord.test",
|
||||||
|
Account: "discord.test",
|
||||||
|
SameChannel: map[string]bool{"bridge1": false},
|
||||||
|
}, r.Gateways["bridge1"].Channels["generaldiscord.test"])
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetDestChannel(t *testing.T) {
|
||||||
|
r := maketestRouter(testconfig2)
|
||||||
|
msg := &config.Message{Text: "test", Channel: "general", Account: "discord.test", Gateway: "bridge1", Protocol: "discord", Username: "test"}
|
||||||
|
for _, br := range r.Gateways["bridge1"].Bridges {
|
||||||
|
switch br.Account {
|
||||||
|
case "discord.test":
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "general",
|
||||||
|
Account: "discord.test",
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "generaldiscord.test",
|
||||||
|
SameChannel: map[string]bool{"bridge1": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, r.Gateways["bridge1"].getDestChannel(msg, *br))
|
||||||
|
case "slack.test":
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "testing",
|
||||||
|
Account: "slack.test",
|
||||||
|
Direction: "out",
|
||||||
|
ID: "testingslack.test",
|
||||||
|
SameChannel: map[string]bool{"bridge1": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, r.Gateways["bridge1"].getDestChannel(msg, *br))
|
||||||
|
case "gitter.42wim":
|
||||||
|
assert.Equal(t, []config.ChannelInfo(nil), r.Gateways["bridge1"].getDestChannel(msg, *br))
|
||||||
|
case "irc.freenode":
|
||||||
|
assert.Equal(t, []config.ChannelInfo(nil), r.Gateways["bridge1"].getDestChannel(msg, *br))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetDestChannelAdvanced(t *testing.T) {
|
||||||
|
r := maketestRouter(testconfig3)
|
||||||
|
var msgs []*config.Message
|
||||||
|
i := 0
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
for _, channel := range gw.Channels {
|
||||||
|
msgs = append(msgs, &config.Message{Text: "text" + strconv.Itoa(i), Channel: channel.Name, Account: channel.Account, Gateway: gw.Name, Username: "user" + strconv.Itoa(i)})
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
hits := make(map[string]int)
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
for _, br := range gw.Bridges {
|
||||||
|
for _, msg := range msgs {
|
||||||
|
channels := gw.getDestChannel(msg, *br)
|
||||||
|
if gw.Name != msg.Gateway {
|
||||||
|
assert.Equal(t, []config.ChannelInfo(nil), channels)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
switch gw.Name {
|
||||||
|
case "bridge":
|
||||||
|
if (msg.Channel == "#main" || msg.Channel == "-1111111111111" || msg.Channel == "irc") &&
|
||||||
|
(msg.Account == ircTestAccount || msg.Account == tgTestAccount || msg.Account == slackTestAccount) {
|
||||||
|
hits[gw.Name]++
|
||||||
|
switch br.Account {
|
||||||
|
case ircTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "#main",
|
||||||
|
Account: ircTestAccount,
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "#mainirc.zzz",
|
||||||
|
SameChannel: map[string]bool{"bridge": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
case tgTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "-1111111111111",
|
||||||
|
Account: tgTestAccount,
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "-1111111111111telegram.zzz",
|
||||||
|
SameChannel: map[string]bool{"bridge": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
case slackTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "irc",
|
||||||
|
Account: slackTestAccount,
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "ircslack.zzz",
|
||||||
|
SameChannel: map[string]bool{"bridge": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case "bridge2":
|
||||||
|
if (msg.Channel == "#main-help" || msg.Channel == "--444444444444") &&
|
||||||
|
(msg.Account == ircTestAccount || msg.Account == tgTestAccount) {
|
||||||
|
hits[gw.Name]++
|
||||||
|
switch br.Account {
|
||||||
|
case ircTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "#main-help",
|
||||||
|
Account: ircTestAccount,
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "#main-helpirc.zzz",
|
||||||
|
SameChannel: map[string]bool{"bridge2": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
case tgTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "--444444444444",
|
||||||
|
Account: tgTestAccount,
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "--444444444444telegram.zzz",
|
||||||
|
SameChannel: map[string]bool{"bridge2": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case "bridge3":
|
||||||
|
if (msg.Channel == "#main-telegram" || msg.Channel == "--333333333333") &&
|
||||||
|
(msg.Account == ircTestAccount || msg.Account == tgTestAccount) {
|
||||||
|
hits[gw.Name]++
|
||||||
|
switch br.Account {
|
||||||
|
case ircTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "#main-telegram",
|
||||||
|
Account: ircTestAccount,
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "#main-telegramirc.zzz",
|
||||||
|
SameChannel: map[string]bool{"bridge3": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
case tgTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "--333333333333",
|
||||||
|
Account: tgTestAccount,
|
||||||
|
Direction: "inout",
|
||||||
|
ID: "--333333333333telegram.zzz",
|
||||||
|
SameChannel: map[string]bool{"bridge3": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case "announcements":
|
||||||
|
if msg.Channel != "-2222222222222" && msg.Account != "telegram" {
|
||||||
|
assert.Equal(t, []config.ChannelInfo(nil), channels)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
hits[gw.Name]++
|
||||||
|
switch br.Account {
|
||||||
|
case ircTestAccount:
|
||||||
|
assert.Len(t, channels, 2)
|
||||||
|
assert.Contains(t, channels, config.ChannelInfo{
|
||||||
|
Name: "#main",
|
||||||
|
Account: ircTestAccount,
|
||||||
|
Direction: "out",
|
||||||
|
ID: "#mainirc.zzz",
|
||||||
|
SameChannel: map[string]bool{"announcements": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
})
|
||||||
|
assert.Contains(t, channels, config.ChannelInfo{
|
||||||
|
Name: "#main-help",
|
||||||
|
Account: ircTestAccount,
|
||||||
|
Direction: "out",
|
||||||
|
ID: "#main-helpirc.zzz",
|
||||||
|
SameChannel: map[string]bool{"announcements": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
})
|
||||||
|
case slackTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "general",
|
||||||
|
Account: slackTestAccount,
|
||||||
|
Direction: "out",
|
||||||
|
ID: "generalslack.zzz",
|
||||||
|
SameChannel: map[string]bool{"announcements": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
case tgTestAccount:
|
||||||
|
assert.Equal(t, []config.ChannelInfo{{
|
||||||
|
Name: "--333333333333",
|
||||||
|
Account: tgTestAccount,
|
||||||
|
Direction: "out",
|
||||||
|
ID: "--333333333333telegram.zzz",
|
||||||
|
SameChannel: map[string]bool{"announcements": false},
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
}}, channels)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
assert.Equal(t, map[string]int{"bridge3": 4, "bridge": 9, "announcements": 3, "bridge2": 4}, hits)
|
||||||
|
}
|
||||||
|
|
||||||
|
type ignoreTestSuite struct {
|
||||||
|
suite.Suite
|
||||||
|
|
||||||
|
gw *Gateway
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestIgnoreSuite(t *testing.T) {
|
||||||
|
s := &ignoreTestSuite{}
|
||||||
|
suite.Run(t, s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ignoreTestSuite) SetupSuite() {
|
||||||
|
logger := logrus.New()
|
||||||
|
logger.SetOutput(ioutil.Discard)
|
||||||
|
s.gw = &Gateway{logger: logrus.NewEntry(logger)}
|
||||||
|
}
|
||||||
|
func (s *ignoreTestSuite) TestIgnoreTextEmpty() {
|
||||||
|
extraFile := make(map[string][]interface{})
|
||||||
|
extraAttach := make(map[string][]interface{})
|
||||||
|
extraFailure := make(map[string][]interface{})
|
||||||
|
extraFile["file"] = append(extraFile["file"], config.FileInfo{})
|
||||||
|
extraAttach["attachments"] = append(extraAttach["attachments"], []string{})
|
||||||
|
extraFailure[config.EventFileFailureSize] = append(extraFailure[config.EventFileFailureSize], config.FileInfo{})
|
||||||
|
|
||||||
|
msgTests := map[string]struct {
|
||||||
|
input *config.Message
|
||||||
|
output bool
|
||||||
|
}{
|
||||||
|
"usertyping": {
|
||||||
|
input: &config.Message{Event: config.EventUserTyping},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
"file attach": {
|
||||||
|
input: &config.Message{Extra: extraFile},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
"attachments": {
|
||||||
|
input: &config.Message{Extra: extraAttach},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
config.EventFileFailureSize: {
|
||||||
|
input: &config.Message{Extra: extraFailure},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
"nil extra": {
|
||||||
|
input: &config.Message{Extra: nil},
|
||||||
|
output: true,
|
||||||
|
},
|
||||||
|
"empty": {
|
||||||
|
input: &config.Message{},
|
||||||
|
output: true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for testname, testcase := range msgTests {
|
||||||
|
output := s.gw.ignoreTextEmpty(testcase.input)
|
||||||
|
s.Assert().Equalf(testcase.output, output, "case '%s' failed", testname)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ignoreTestSuite) TestIgnoreTexts() {
|
||||||
|
msgTests := map[string]struct {
|
||||||
|
input string
|
||||||
|
re []string
|
||||||
|
output bool
|
||||||
|
}{
|
||||||
|
"no regex": {
|
||||||
|
input: "a text message",
|
||||||
|
re: []string{},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
"simple regex": {
|
||||||
|
input: "a text message",
|
||||||
|
re: []string{"text"},
|
||||||
|
output: true,
|
||||||
|
},
|
||||||
|
"multiple regex fail": {
|
||||||
|
input: "a text message",
|
||||||
|
re: []string{"abc", "123$"},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
"multiple regex pass": {
|
||||||
|
input: "a text message",
|
||||||
|
re: []string{"lala", "sage$"},
|
||||||
|
output: true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for testname, testcase := range msgTests {
|
||||||
|
output := s.gw.ignoreText(testcase.input, testcase.re)
|
||||||
|
s.Assert().Equalf(testcase.output, output, "case '%s' failed", testname)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ignoreTestSuite) TestIgnoreNicks() {
|
||||||
|
msgTests := map[string]struct {
|
||||||
|
input string
|
||||||
|
re []string
|
||||||
|
output bool
|
||||||
|
}{
|
||||||
|
"no entry": {
|
||||||
|
input: "user",
|
||||||
|
re: []string{},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
"one entry": {
|
||||||
|
input: "user",
|
||||||
|
re: []string{"user"},
|
||||||
|
output: true,
|
||||||
|
},
|
||||||
|
"multiple entries": {
|
||||||
|
input: "user",
|
||||||
|
re: []string{"abc", "user"},
|
||||||
|
output: true,
|
||||||
|
},
|
||||||
|
"multiple entries fail": {
|
||||||
|
input: "user",
|
||||||
|
re: []string{"abc", "def"},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for testname, testcase := range msgTests {
|
||||||
|
output := s.gw.ignoreText(testcase.input, testcase.re)
|
||||||
|
s.Assert().Equalf(testcase.output, output, "case '%s' failed", testname)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkTengo(b *testing.B) {
|
||||||
|
msg := &config.Message{Username: "user", Text: "blah testing", Account: "protocol.account", Channel: "mychannel"}
|
||||||
|
for n := 0; n < b.N; n++ {
|
||||||
|
err := modifyMessageTengo("bench.tengo", msg)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
266
gateway/handlers.go
Normal file
@ -0,0 +1,266 @@
|
|||||||
|
package gateway
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"crypto/sha1" //nolint:gosec
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
// handleEventFailure handles failures and reconnects bridges.
|
||||||
|
func (r *Router) handleEventFailure(msg *config.Message) {
|
||||||
|
if msg.Event != config.EventFailure {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
for _, br := range gw.Bridges {
|
||||||
|
if msg.Account == br.Account {
|
||||||
|
go gw.reconnectBridge(br)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleEventGetChannelMembers handles channel members
|
||||||
|
func (r *Router) handleEventGetChannelMembers(msg *config.Message) {
|
||||||
|
if msg.Event != config.EventGetChannelMembers {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
for _, br := range gw.Bridges {
|
||||||
|
if msg.Account == br.Account {
|
||||||
|
cMembers := msg.Extra[config.EventGetChannelMembers][0].(config.ChannelMembers)
|
||||||
|
r.logger.Debugf("Syncing channelmembers from %s", msg.Account)
|
||||||
|
br.SetChannelMembers(&cMembers)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleEventRejoinChannels handles rejoining of channels.
|
||||||
|
func (r *Router) handleEventRejoinChannels(msg *config.Message) {
|
||||||
|
if msg.Event != config.EventRejoinChannels {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
for _, br := range gw.Bridges {
|
||||||
|
if msg.Account == br.Account {
|
||||||
|
br.Joined = make(map[string]bool)
|
||||||
|
if err := br.JoinChannels(); err != nil {
|
||||||
|
r.logger.Errorf("channel join failed for %s: %s", msg.Account, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleFiles uploads or places all files on the given msg to the MediaServer and
|
||||||
|
// adds the new URL of the file on the MediaServer onto the given msg.
|
||||||
|
func (gw *Gateway) handleFiles(msg *config.Message) {
|
||||||
|
reg := regexp.MustCompile("[^a-zA-Z0-9]+")
|
||||||
|
|
||||||
|
// If we don't have a attachfield or we don't have a mediaserver configured return
|
||||||
|
if msg.Extra == nil ||
|
||||||
|
(gw.BridgeValues().General.MediaServerUpload == "" &&
|
||||||
|
gw.BridgeValues().General.MediaDownloadPath == "") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we don't have files, nothing to upload.
|
||||||
|
if len(msg.Extra["file"]) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, f := range msg.Extra["file"] {
|
||||||
|
fi := f.(config.FileInfo)
|
||||||
|
ext := filepath.Ext(fi.Name)
|
||||||
|
fi.Name = fi.Name[0 : len(fi.Name)-len(ext)]
|
||||||
|
fi.Name = reg.ReplaceAllString(fi.Name, "_")
|
||||||
|
fi.Name += ext
|
||||||
|
|
||||||
|
sha1sum := fmt.Sprintf("%x", sha1.Sum(*fi.Data))[:8] //nolint:gosec
|
||||||
|
|
||||||
|
if gw.BridgeValues().General.MediaServerUpload != "" {
|
||||||
|
// Use MediaServerUpload. Upload using a PUT HTTP request and basicauth.
|
||||||
|
if err := gw.handleFilesUpload(&fi); err != nil {
|
||||||
|
gw.logger.Error(err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Use MediaServerPath. Place the file on the current filesystem.
|
||||||
|
if err := gw.handleFilesLocal(&fi); err != nil {
|
||||||
|
gw.logger.Error(err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Download URL.
|
||||||
|
durl := gw.BridgeValues().General.MediaServerDownload + "/" + sha1sum + "/" + fi.Name
|
||||||
|
|
||||||
|
gw.logger.Debugf("mediaserver download URL = %s", durl)
|
||||||
|
|
||||||
|
// We uploaded/placed the file successfully. Add the SHA and URL.
|
||||||
|
extra := msg.Extra["file"][i].(config.FileInfo)
|
||||||
|
extra.URL = durl
|
||||||
|
extra.SHA = sha1sum
|
||||||
|
msg.Extra["file"][i] = extra
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleFilesUpload uses MediaServerUpload configuration to upload the file.
|
||||||
|
// Returns error on failure.
|
||||||
|
func (gw *Gateway) handleFilesUpload(fi *config.FileInfo) error {
|
||||||
|
client := &http.Client{
|
||||||
|
Timeout: time.Second * 5,
|
||||||
|
}
|
||||||
|
// Use MediaServerUpload. Upload using a PUT HTTP request and basicauth.
|
||||||
|
sha1sum := fmt.Sprintf("%x", sha1.Sum(*fi.Data))[:8] //nolint:gosec
|
||||||
|
url := gw.BridgeValues().General.MediaServerUpload + "/" + sha1sum + "/" + fi.Name
|
||||||
|
|
||||||
|
req, err := http.NewRequest("PUT", url, bytes.NewReader(*fi.Data))
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("mediaserver upload failed, could not create request: %#v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
gw.logger.Debugf("mediaserver upload url: %s", url)
|
||||||
|
|
||||||
|
req.Header.Set("Content-Type", "binary/octet-stream")
|
||||||
|
_, err = client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("mediaserver upload failed, could not Do request: %#v", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleFilesLocal use MediaServerPath configuration, places the file on the current filesystem.
|
||||||
|
// Returns error on failure.
|
||||||
|
func (gw *Gateway) handleFilesLocal(fi *config.FileInfo) error {
|
||||||
|
sha1sum := fmt.Sprintf("%x", sha1.Sum(*fi.Data))[:8] //nolint:gosec
|
||||||
|
dir := gw.BridgeValues().General.MediaDownloadPath + "/" + sha1sum
|
||||||
|
err := os.Mkdir(dir, os.ModePerm)
|
||||||
|
if err != nil && !os.IsExist(err) {
|
||||||
|
return fmt.Errorf("mediaserver path failed, could not mkdir: %s %#v", err, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
path := dir + "/" + fi.Name
|
||||||
|
gw.logger.Debugf("mediaserver path placing file: %s", path)
|
||||||
|
|
||||||
|
err = ioutil.WriteFile(path, *fi.Data, os.ModePerm)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("mediaserver path failed, could not writefile: %s %#v", err, err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ignoreEvent returns true if we need to ignore this event for the specified destination bridge.
|
||||||
|
func (gw *Gateway) ignoreEvent(event string, dest *bridge.Bridge) bool {
|
||||||
|
switch event {
|
||||||
|
case config.EventAvatarDownload:
|
||||||
|
// Avatar downloads are only relevant for telegram and mattermost for now
|
||||||
|
if dest.Protocol != "mattermost" && dest.Protocol != "telegram" {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
case config.EventJoinLeave:
|
||||||
|
// only relay join/part when configured
|
||||||
|
if !dest.GetBool("ShowJoinPart") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
case config.EventTopicChange:
|
||||||
|
// only relay topic change when used in some way on other side
|
||||||
|
if dest.GetBool("ShowTopicChange") && dest.GetBool("SyncTopic") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// handleMessage makes sure the message get sent to the correct bridge/channels.
|
||||||
|
// Returns an array of msg ID's
|
||||||
|
func (gw *Gateway) handleMessage(rmsg *config.Message, dest *bridge.Bridge) []*BrMsgID {
|
||||||
|
var brMsgIDs []*BrMsgID
|
||||||
|
|
||||||
|
// if we have an attached file, or other info
|
||||||
|
if rmsg.Extra != nil && len(rmsg.Extra[config.EventFileFailureSize]) != 0 && rmsg.Text == "" {
|
||||||
|
return brMsgIDs
|
||||||
|
}
|
||||||
|
|
||||||
|
if gw.ignoreEvent(rmsg.Event, dest) {
|
||||||
|
return brMsgIDs
|
||||||
|
}
|
||||||
|
|
||||||
|
// broadcast to every out channel (irc QUIT)
|
||||||
|
if rmsg.Channel == "" && rmsg.Event != config.EventJoinLeave {
|
||||||
|
gw.logger.Debug("empty channel")
|
||||||
|
return brMsgIDs
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the ID of the parent message in thread
|
||||||
|
var canonicalParentMsgID string
|
||||||
|
if rmsg.ParentID != "" && dest.GetBool("PreserveThreading") {
|
||||||
|
canonicalParentMsgID = gw.FindCanonicalMsgID(rmsg.Protocol, rmsg.ParentID)
|
||||||
|
}
|
||||||
|
|
||||||
|
channels := gw.getDestChannel(rmsg, *dest)
|
||||||
|
for idx := range channels {
|
||||||
|
channel := &channels[idx]
|
||||||
|
msgID, err := gw.SendMessage(rmsg, dest, channel, canonicalParentMsgID)
|
||||||
|
if err != nil {
|
||||||
|
gw.logger.Errorf("SendMessage failed: %s", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if msgID == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
brMsgIDs = append(brMsgIDs, &BrMsgID{dest, dest.Protocol + " " + msgID, channel.ID})
|
||||||
|
}
|
||||||
|
return brMsgIDs
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gw *Gateway) handleExtractNicks(msg *config.Message) {
|
||||||
|
var err error
|
||||||
|
br := gw.Bridges[msg.Account]
|
||||||
|
for _, outer := range br.GetStringSlice2D("ExtractNicks") {
|
||||||
|
search := outer[0]
|
||||||
|
replace := outer[1]
|
||||||
|
msg.Username, msg.Text, err = extractNick(search, replace, msg.Username, msg.Text)
|
||||||
|
if err != nil {
|
||||||
|
gw.logger.Errorf("regexp in %s failed: %s", msg.Account, err)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// extractNick searches for a username (based on "search" a regular expression).
|
||||||
|
// if this matches it extracts a nick (based on "extract" another regular expression) from text
|
||||||
|
// and replaces username with this result.
|
||||||
|
// returns error if the regexp doesn't compile.
|
||||||
|
func extractNick(search, extract, username, text string) (string, string, error) {
|
||||||
|
re, err := regexp.Compile(search)
|
||||||
|
if err != nil {
|
||||||
|
return username, text, err
|
||||||
|
}
|
||||||
|
if re.MatchString(username) {
|
||||||
|
re, err = regexp.Compile(extract)
|
||||||
|
if err != nil {
|
||||||
|
return username, text, err
|
||||||
|
}
|
||||||
|
res := re.FindAllStringSubmatch(text, 1)
|
||||||
|
// only replace if we have exactly 1 match
|
||||||
|
if len(res) > 0 && len(res[0]) == 2 {
|
||||||
|
username = res[0][1]
|
||||||
|
text = strings.Replace(text, res[0][0], "", 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return username, text, nil
|
||||||
|
}
|
75
gateway/handlers_test.go
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
package gateway
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestIgnoreEvent(t *testing.T) {
|
||||||
|
eventTests := map[string]struct {
|
||||||
|
input string
|
||||||
|
dest *bridge.Bridge
|
||||||
|
output bool
|
||||||
|
}{
|
||||||
|
"avatar mattermost": {
|
||||||
|
input: config.EventAvatarDownload,
|
||||||
|
dest: &bridge.Bridge{Protocol: "mattermost"},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
"avatar slack": {
|
||||||
|
input: config.EventAvatarDownload,
|
||||||
|
dest: &bridge.Bridge{Protocol: "slack"},
|
||||||
|
output: true,
|
||||||
|
},
|
||||||
|
"avatar telegram": {
|
||||||
|
input: config.EventAvatarDownload,
|
||||||
|
dest: &bridge.Bridge{Protocol: "telegram"},
|
||||||
|
output: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
gw := &Gateway{}
|
||||||
|
for testname, testcase := range eventTests {
|
||||||
|
output := gw.ignoreEvent(testcase.input, testcase.dest)
|
||||||
|
assert.Equalf(t, testcase.output, output, "case '%s' failed", testname)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestExtractNick(t *testing.T) {
|
||||||
|
eventTests := map[string]struct {
|
||||||
|
search string
|
||||||
|
extract string
|
||||||
|
username string
|
||||||
|
text string
|
||||||
|
resultUsername string
|
||||||
|
resultText string
|
||||||
|
}{
|
||||||
|
"test1": {
|
||||||
|
search: "fromgitter",
|
||||||
|
extract: "<(.*?)>\\s+",
|
||||||
|
username: "fromgitter",
|
||||||
|
text: "<userx> blahblah",
|
||||||
|
resultUsername: "userx",
|
||||||
|
resultText: "blahblah",
|
||||||
|
},
|
||||||
|
"test2": {
|
||||||
|
search: "<.*?bot>",
|
||||||
|
//extract: `\((.*?)\)\s+`,
|
||||||
|
extract: "\\((.*?)\\)\\s+",
|
||||||
|
username: "<matterbot>",
|
||||||
|
text: "(userx) blahblah (abc) test",
|
||||||
|
resultUsername: "userx",
|
||||||
|
resultText: "blahblah (abc) test",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
// gw := &Gateway{}
|
||||||
|
for testname, testcase := range eventTests {
|
||||||
|
resultUsername, resultText, _ := extractNick(testcase.search, testcase.extract, testcase.username, testcase.text)
|
||||||
|
assert.Equalf(t, testcase.resultUsername, resultUsername, "case '%s' failed", testname)
|
||||||
|
assert.Equalf(t, testcase.resultText, resultText, "case '%s' failed", testname)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
172
gateway/router.go
Normal file
@ -0,0 +1,172 @@
|
|||||||
|
package gateway
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/42wim/matterbridge/gateway/samechannel"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Router struct {
|
||||||
|
config.Config
|
||||||
|
sync.RWMutex
|
||||||
|
|
||||||
|
BridgeMap map[string]bridge.Factory
|
||||||
|
Gateways map[string]*Gateway
|
||||||
|
Message chan config.Message
|
||||||
|
MattermostPlugin chan config.Message
|
||||||
|
|
||||||
|
logger *logrus.Entry
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRouter initializes a new Matterbridge router for the specified configuration and
|
||||||
|
// sets up all required gateways.
|
||||||
|
func NewRouter(rootLogger *logrus.Logger, cfg config.Config, bridgeMap map[string]bridge.Factory) (*Router, error) {
|
||||||
|
logger := rootLogger.WithFields(logrus.Fields{"prefix": "router"})
|
||||||
|
|
||||||
|
r := &Router{
|
||||||
|
Config: cfg,
|
||||||
|
BridgeMap: bridgeMap,
|
||||||
|
Message: make(chan config.Message),
|
||||||
|
MattermostPlugin: make(chan config.Message),
|
||||||
|
Gateways: make(map[string]*Gateway),
|
||||||
|
logger: logger,
|
||||||
|
}
|
||||||
|
sgw := samechannel.New(cfg)
|
||||||
|
gwconfigs := append(sgw.GetConfig(), cfg.BridgeValues().Gateway...)
|
||||||
|
|
||||||
|
for idx := range gwconfigs {
|
||||||
|
entry := &gwconfigs[idx]
|
||||||
|
if !entry.Enable {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if entry.Name == "" {
|
||||||
|
return nil, fmt.Errorf("%s", "Gateway without name found")
|
||||||
|
}
|
||||||
|
if _, ok := r.Gateways[entry.Name]; ok {
|
||||||
|
return nil, fmt.Errorf("Gateway with name %s already exists", entry.Name)
|
||||||
|
}
|
||||||
|
r.Gateways[entry.Name] = New(rootLogger, entry, r)
|
||||||
|
}
|
||||||
|
return r, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start will connect all gateways belonging to this router and subsequently route messages
|
||||||
|
// between them.
|
||||||
|
func (r *Router) Start() error {
|
||||||
|
m := make(map[string]*bridge.Bridge)
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
r.logger.Infof("Parsing gateway %s", gw.Name)
|
||||||
|
for _, br := range gw.Bridges {
|
||||||
|
m[br.Account] = br
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, br := range m {
|
||||||
|
r.logger.Infof("Starting bridge: %s ", br.Account)
|
||||||
|
err := br.Connect()
|
||||||
|
if err != nil {
|
||||||
|
e := fmt.Errorf("Bridge %s failed to start: %v", br.Account, err)
|
||||||
|
if r.disableBridge(br, e) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return e
|
||||||
|
}
|
||||||
|
err = br.JoinChannels()
|
||||||
|
if err != nil {
|
||||||
|
e := fmt.Errorf("Bridge %s failed to join channel: %v", br.Account, err)
|
||||||
|
if r.disableBridge(br, e) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return e
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// remove unused bridges
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
for i, br := range gw.Bridges {
|
||||||
|
if br.Bridger == nil {
|
||||||
|
r.logger.Errorf("removing failed bridge %s", i)
|
||||||
|
delete(gw.Bridges, i)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
go r.handleReceive()
|
||||||
|
//go r.updateChannelMembers()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// disableBridge returns true and empties a bridge if we have IgnoreFailureOnStart configured
|
||||||
|
// otherwise returns false
|
||||||
|
func (r *Router) disableBridge(br *bridge.Bridge, err error) bool {
|
||||||
|
if r.BridgeValues().General.IgnoreFailureOnStart {
|
||||||
|
r.logger.Error(err)
|
||||||
|
// setting this bridge empty
|
||||||
|
*br = bridge.Bridge{}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Router) getBridge(account string) *bridge.Bridge {
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
if br, ok := gw.Bridges[account]; ok {
|
||||||
|
return br
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Router) handleReceive() {
|
||||||
|
for msg := range r.Message {
|
||||||
|
msg := msg // scopelint
|
||||||
|
r.handleEventGetChannelMembers(&msg)
|
||||||
|
r.handleEventFailure(&msg)
|
||||||
|
r.handleEventRejoinChannels(&msg)
|
||||||
|
idx := 0
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
// record all the message ID's of the different bridges
|
||||||
|
var msgIDs []*BrMsgID
|
||||||
|
if gw.ignoreMessage(&msg) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
msg.Timestamp = time.Now()
|
||||||
|
gw.modifyMessage(&msg)
|
||||||
|
if idx == 0 {
|
||||||
|
gw.handleFiles(&msg)
|
||||||
|
}
|
||||||
|
for _, br := range gw.Bridges {
|
||||||
|
msgIDs = append(msgIDs, gw.handleMessage(&msg, br)...)
|
||||||
|
}
|
||||||
|
// only add the message ID if it doesn't already exists
|
||||||
|
if _, ok := gw.Messages.Get(msg.Protocol + " " + msg.ID); !ok && msg.ID != "" {
|
||||||
|
gw.Messages.Add(msg.Protocol+" "+msg.ID, msgIDs)
|
||||||
|
}
|
||||||
|
idx++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// updateChannelMembers sends every minute an GetChannelMembers event to all bridges.
|
||||||
|
func (r *Router) updateChannelMembers() {
|
||||||
|
// TODO sleep a minute because slack can take a while
|
||||||
|
// fix this by having actually connectionDone events send to the router
|
||||||
|
time.Sleep(time.Minute)
|
||||||
|
for {
|
||||||
|
for _, gw := range r.Gateways {
|
||||||
|
for _, br := range gw.Bridges {
|
||||||
|
// only for slack now
|
||||||
|
if br.Protocol != "slack" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
r.logger.Debugf("sending %s to %s", config.EventGetChannelMembers, br.Account)
|
||||||
|
if _, err := br.Send(config.Message{Event: config.EventGetChannelMembers}); err != nil {
|
||||||
|
r.logger.Errorf("updateChannelMembers: %s", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
time.Sleep(time.Minute)
|
||||||
|
}
|
||||||
|
}
|
@ -1,105 +1,28 @@
|
|||||||
package samechannelgateway
|
package samechannel
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/42wim/matterbridge/bridge"
|
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
log "github.com/Sirupsen/logrus"
|
|
||||||
"strings"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type SameChannelGateway struct {
|
type SameChannelGateway struct {
|
||||||
*config.Config
|
config.Config
|
||||||
MyConfig *config.SameChannelGateway
|
|
||||||
Bridges map[string]*bridge.Bridge
|
|
||||||
Channels []string
|
|
||||||
ignoreNicks map[string][]string
|
|
||||||
Name string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(cfg *config.Config, gateway *config.SameChannelGateway) error {
|
func New(cfg config.Config) *SameChannelGateway {
|
||||||
c := make(chan config.Message)
|
return &SameChannelGateway{Config: cfg}
|
||||||
gw := &SameChannelGateway{}
|
|
||||||
gw.Bridges = make(map[string]*bridge.Bridge)
|
|
||||||
gw.Name = gateway.Name
|
|
||||||
gw.Config = cfg
|
|
||||||
gw.MyConfig = gateway
|
|
||||||
gw.Channels = gateway.Channels
|
|
||||||
for _, account := range gateway.Accounts {
|
|
||||||
br := config.Bridge{Account: account}
|
|
||||||
log.Infof("Starting bridge: %s", account)
|
|
||||||
gw.Bridges[account] = bridge.New(cfg, &br, c)
|
|
||||||
}
|
|
||||||
for _, br := range gw.Bridges {
|
|
||||||
err := br.Connect()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Bridge %s failed to start: %v", br.Account, err)
|
|
||||||
}
|
|
||||||
for _, channel := range gw.Channels {
|
|
||||||
log.Infof("%s: joining %s", br.Account, channel)
|
|
||||||
br.JoinChannel(channel)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
gw.handleReceive(c)
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gw *SameChannelGateway) handleReceive(c chan config.Message) {
|
func (sgw *SameChannelGateway) GetConfig() []config.Gateway {
|
||||||
for {
|
var gwconfigs []config.Gateway
|
||||||
select {
|
cfg := sgw.Config
|
||||||
case msg := <-c:
|
for _, gw := range cfg.BridgeValues().SameChannelGateway {
|
||||||
if !gw.ignoreMessage(&msg) {
|
gwconfig := config.Gateway{Name: gw.Name, Enable: gw.Enable}
|
||||||
for _, br := range gw.Bridges {
|
for _, account := range gw.Accounts {
|
||||||
gw.handleMessage(msg, br)
|
for _, channel := range gw.Channels {
|
||||||
}
|
gwconfig.InOut = append(gwconfig.InOut, config.Bridge{Account: account, Channel: channel, SameChannel: true})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
gwconfigs = append(gwconfigs, gwconfig)
|
||||||
}
|
}
|
||||||
}
|
return gwconfigs
|
||||||
|
|
||||||
func (gw *SameChannelGateway) handleMessage(msg config.Message, dest *bridge.Bridge) {
|
|
||||||
// is this a configured channel
|
|
||||||
if !gw.validChannel(msg.Channel) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// do not send the message to the bridge we come from if also the channel is the same
|
|
||||||
if msg.Account == dest.Account {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
gw.modifyUsername(&msg, dest)
|
|
||||||
log.Debugf("Sending %#v from %s (%s) to %s (%s)", msg, msg.Account, msg.Channel, dest.Account, msg.Channel)
|
|
||||||
err := dest.Send(msg)
|
|
||||||
if err != nil {
|
|
||||||
log.Error(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (gw *SameChannelGateway) ignoreMessage(msg *config.Message) bool {
|
|
||||||
for _, entry := range strings.Fields(gw.Bridges[msg.Account].Config.IgnoreNicks) {
|
|
||||||
if msg.Username == entry {
|
|
||||||
log.Debugf("ignoring %s from %s", msg.Username, msg.Account)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (gw *SameChannelGateway) modifyUsername(msg *config.Message, dest *bridge.Bridge) {
|
|
||||||
br := gw.Bridges[msg.Account]
|
|
||||||
nick := gw.Config.General.RemoteNickFormat
|
|
||||||
if nick == "" {
|
|
||||||
nick = dest.Config.RemoteNickFormat
|
|
||||||
}
|
|
||||||
nick = strings.Replace(nick, "{NICK}", msg.Username, -1)
|
|
||||||
nick = strings.Replace(nick, "{BRIDGE}", br.Name, -1)
|
|
||||||
nick = strings.Replace(nick, "{PROTOCOL}", br.Protocol, -1)
|
|
||||||
msg.Username = nick
|
|
||||||
}
|
|
||||||
|
|
||||||
func (gw *SameChannelGateway) validChannel(channel string) bool {
|
|
||||||
for _, c := range gw.Channels {
|
|
||||||
if c == channel {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
77
gateway/samechannel/samechannel_test.go
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
package samechannel
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
const testConfig = `
|
||||||
|
[mattermost.test]
|
||||||
|
[slack.test]
|
||||||
|
|
||||||
|
[[samechannelgateway]]
|
||||||
|
enable = true
|
||||||
|
name = "blah"
|
||||||
|
accounts = [ "mattermost.test","slack.test" ]
|
||||||
|
channels = [ "testing","testing2","testing10"]
|
||||||
|
`
|
||||||
|
|
||||||
|
var (
|
||||||
|
expectedConfig = config.Gateway{
|
||||||
|
Name: "blah",
|
||||||
|
Enable: true,
|
||||||
|
In: []config.Bridge(nil),
|
||||||
|
Out: []config.Bridge(nil),
|
||||||
|
InOut: []config.Bridge{
|
||||||
|
{
|
||||||
|
Account: "mattermost.test",
|
||||||
|
Channel: "testing",
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
SameChannel: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Account: "mattermost.test",
|
||||||
|
Channel: "testing2",
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
SameChannel: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Account: "mattermost.test",
|
||||||
|
Channel: "testing10",
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
SameChannel: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Account: "slack.test",
|
||||||
|
Channel: "testing",
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
SameChannel: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Account: "slack.test",
|
||||||
|
Channel: "testing2",
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
SameChannel: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Account: "slack.test",
|
||||||
|
Channel: "testing10",
|
||||||
|
Options: config.ChannelOptions{Key: ""},
|
||||||
|
SameChannel: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGetConfig(t *testing.T) {
|
||||||
|
logger := logrus.New()
|
||||||
|
logger.SetOutput(ioutil.Discard)
|
||||||
|
cfg := config.NewConfigFromString(logger, []byte(testConfig))
|
||||||
|
sgw := New(cfg)
|
||||||
|
configs := sgw.GetConfig()
|
||||||
|
assert.Equal(t, []config.Gateway{expectedConfig}, configs)
|
||||||
|
}
|
74
go.mod
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
module github.com/42wim/matterbridge
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/42wim/go-gitter v0.0.0-20170828205020-017310c2d557
|
||||||
|
github.com/Baozisoftware/qrcode-terminal-go v0.0.0-20170407111555-c0650d8dff0f
|
||||||
|
github.com/BurntSushi/toml v0.0.0-20170318202913-d94612f9fc14 // indirect
|
||||||
|
github.com/Jeffail/gabs v1.1.1 // indirect
|
||||||
|
github.com/Philipp15b/go-steam v1.0.1-0.20180818081528-681bd9573329
|
||||||
|
github.com/bwmarrin/discordgo v0.19.0
|
||||||
|
github.com/d5/tengo v1.12.1
|
||||||
|
github.com/dfordsoft/golib v0.0.0-20180902042739-76ee6ab99bec
|
||||||
|
github.com/fsnotify/fsnotify v1.4.7
|
||||||
|
github.com/go-telegram-bot-api/telegram-bot-api v4.6.5-0.20181225215658-ec221ba9ea45+incompatible
|
||||||
|
github.com/google/gops v0.3.5
|
||||||
|
github.com/gopackage/ddp v0.0.0-20170117053602-652027933df4 // indirect
|
||||||
|
github.com/gopherjs/gopherjs v0.0.0-20180628210949-0892b62f0d9f // indirect
|
||||||
|
github.com/gorilla/schema v1.0.2
|
||||||
|
github.com/gorilla/websocket v1.4.0
|
||||||
|
github.com/hashicorp/golang-lru v0.5.0
|
||||||
|
github.com/hpcloud/tail v1.0.0 // indirect
|
||||||
|
github.com/jpillora/backoff v0.0.0-20180909062703-3050d21c67d7
|
||||||
|
github.com/jtolds/gls v4.2.1+incompatible // indirect
|
||||||
|
github.com/kardianos/osext v0.0.0-20170510131534-ae77be60afb1 // indirect
|
||||||
|
github.com/kr/pretty v0.1.0 // indirect
|
||||||
|
github.com/labstack/echo/v4 v4.0.0
|
||||||
|
github.com/lrstanley/girc v0.0.0-20190210212025-51b8e096d398
|
||||||
|
github.com/lusis/go-slackbot v0.0.0-20180109053408-401027ccfef5 // indirect
|
||||||
|
github.com/lusis/slack-test v0.0.0-20180109053238-3c758769bfa6 // indirect
|
||||||
|
github.com/matterbridge/Rocket.Chat.Go.SDK v0.0.0-20190210153444-cc9d05784d5d
|
||||||
|
github.com/matterbridge/go-whatsapp v0.0.1-0.20190301204034-f2f1b29d441b
|
||||||
|
github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91
|
||||||
|
github.com/matterbridge/gomatrix v0.0.0-20190102230110-6f9631ca6dea
|
||||||
|
github.com/matterbridge/gozulipbot v0.0.0-20190212232658-7aa251978a18
|
||||||
|
github.com/matterbridge/logrus-prefixed-formatter v0.0.0-20180806162718-01618749af61
|
||||||
|
github.com/matterbridge/mautrix-whatsapp v0.0.0-20190301210046-3539cf52ed6e
|
||||||
|
github.com/mattermost/mattermost-server v5.5.0+incompatible
|
||||||
|
github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b // indirect
|
||||||
|
github.com/mreiferson/go-httpclient v0.0.0-20160630210159-31f0106b4474 // indirect
|
||||||
|
github.com/mrexodia/wray v0.0.0-20160318003008-78a2c1f284ff // indirect
|
||||||
|
github.com/nelsonken/gomf v0.0.0-20180504123937-a9dd2f9deae9
|
||||||
|
github.com/nicksnyder/go-i18n v1.4.0 // indirect
|
||||||
|
github.com/nlopes/slack v0.5.0
|
||||||
|
github.com/onsi/ginkgo v1.6.0 // indirect
|
||||||
|
github.com/onsi/gomega v1.4.1 // indirect
|
||||||
|
github.com/paulrosania/go-charset v0.0.0-20151028000031-621bb39fcc83
|
||||||
|
github.com/pborman/uuid v0.0.0-20160216163710-c55201b03606 // indirect
|
||||||
|
github.com/peterhellberg/emojilib v0.0.0-20190124112554-c18758d55320
|
||||||
|
github.com/pkg/errors v0.8.0 // indirect
|
||||||
|
github.com/rs/xid v1.2.1
|
||||||
|
github.com/russross/blackfriday v1.5.2
|
||||||
|
github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca
|
||||||
|
github.com/shazow/ssh-chat v0.0.0-20190125184227-81d7e1686296
|
||||||
|
github.com/sirupsen/logrus v1.3.0
|
||||||
|
github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9 // indirect
|
||||||
|
github.com/smartystreets/goconvey v0.0.0-20180222194500-ef6db91d284a // indirect
|
||||||
|
github.com/spf13/viper v1.3.1
|
||||||
|
github.com/stretchr/testify v1.3.0
|
||||||
|
github.com/technoweenie/multipartstreamer v1.0.1 // indirect
|
||||||
|
github.com/x-cray/logrus-prefixed-formatter v0.5.2 // indirect
|
||||||
|
github.com/zfjagann/golang-ring v0.0.0-20190106091943-a88bb6aef447
|
||||||
|
gitlab.com/golang-commonmark/html v0.0.0-20180917080848-cfaf75183c4a // indirect
|
||||||
|
gitlab.com/golang-commonmark/linkify v0.0.0-20180917065525-c22b7bdb1179 // indirect
|
||||||
|
gitlab.com/golang-commonmark/markdown v0.0.0-20181102083822-772775880e1f
|
||||||
|
gitlab.com/golang-commonmark/mdurl v0.0.0-20180912090424-e5bce34c34f2 // indirect
|
||||||
|
gitlab.com/golang-commonmark/puny v0.0.0-20180912090636-2cd490539afe // indirect
|
||||||
|
gitlab.com/opennota/wd v0.0.0-20180912061657-c5d65f63c638 // indirect
|
||||||
|
go.uber.org/atomic v1.3.2 // indirect
|
||||||
|
go.uber.org/multierr v1.1.0 // indirect
|
||||||
|
go.uber.org/zap v1.9.1 // indirect
|
||||||
|
golang.org/x/image v0.0.0-20190220214146-31aff87c08e9
|
||||||
|
gopkg.in/fsnotify.v1 v1.4.7 // indirect
|
||||||
|
gopkg.in/natefinch/lumberjack.v2 v2.0.0 // indirect
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 // indirect
|
||||||
|
)
|
232
go.sum
Normal file
@ -0,0 +1,232 @@
|
|||||||
|
github.com/42wim/go-gitter v0.0.0-20170828205020-017310c2d557 h1:IZtuWGfzQnKnCSu+vl8WGLhpVQ5Uvy3rlSwqXSg+sQg=
|
||||||
|
github.com/42wim/go-gitter v0.0.0-20170828205020-017310c2d557/go.mod h1:jL0YSXMs/txjtGJ4PWrmETOk6KUHMDPMshgQZlTeB3Y=
|
||||||
|
github.com/Baozisoftware/qrcode-terminal-go v0.0.0-20170407111555-c0650d8dff0f h1:2dk3eOnYllh+wUOuDhOoC2vUVoJF/5z478ryJ+wzEII=
|
||||||
|
github.com/Baozisoftware/qrcode-terminal-go v0.0.0-20170407111555-c0650d8dff0f/go.mod h1:4a58ifQTEe2uwwsaqbh3i2un5/CBPg+At/qHpt18Tmk=
|
||||||
|
github.com/BurntSushi/toml v0.0.0-20170318202913-d94612f9fc14 h1:v/zr4ns/4sSahF9KBm4Uc933bLsEEv7LuT63CJ019yo=
|
||||||
|
github.com/BurntSushi/toml v0.0.0-20170318202913-d94612f9fc14/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
||||||
|
github.com/Jeffail/gabs v1.1.1 h1:V0uzR08Hj22EX8+8QMhyI9sX2hwRu+/RJhJUmnwda/E=
|
||||||
|
github.com/Jeffail/gabs v1.1.1/go.mod h1:6xMvQMK4k33lb7GUUpaAPh6nKMmemQeg5d4gn7/bOXc=
|
||||||
|
github.com/Philipp15b/go-steam v1.0.1-0.20180818081528-681bd9573329 h1:xZBoq249G9MSt+XuY7sVQzcfONJ6IQuwpCK+KAaOpnY=
|
||||||
|
github.com/Philipp15b/go-steam v1.0.1-0.20180818081528-681bd9573329/go.mod h1:HuVM+sZFzumUdKPWiz+IlCMb4RdsKdT3T+nQBKL+sYg=
|
||||||
|
github.com/alexcesaro/log v0.0.0-20150915221235-61e686294e58/go.mod h1:YNfsMyWSs+h+PaYkxGeMVmVCX75Zj/pqdjbu12ciCYE=
|
||||||
|
github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8=
|
||||||
|
github.com/bwmarrin/discordgo v0.19.0 h1:kMED/DB0NR1QhRcalb85w0Cu3Ep2OrGAqZH1R5awQiY=
|
||||||
|
github.com/bwmarrin/discordgo v0.19.0/go.mod h1:O9S4p+ofTFwB02em7jkpkV8M3R0/PUVOwN61zSZ0r4Q=
|
||||||
|
github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE=
|
||||||
|
github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk=
|
||||||
|
github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
|
||||||
|
github.com/d5/tengo v1.12.1 h1:libKkDM95CsZgYs6E5eiEaM9sbcw2EzJRSkr9o5NO4s=
|
||||||
|
github.com/d5/tengo v1.12.1/go.mod h1:gsbjo7lBXzBIWBd6NQp1lRKqqiDDANqBOyhW8rTlFsY=
|
||||||
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/dfordsoft/golib v0.0.0-20180902042739-76ee6ab99bec h1:JEUiu7P9smN7zgX87a2zVnnbPPickIM9Gf9OIhsIgWQ=
|
||||||
|
github.com/dfordsoft/golib v0.0.0-20180902042739-76ee6ab99bec/go.mod h1:UGa5M2Sz/Uh13AMse4+RELKCDw7kqgqlTjeGae+7vUY=
|
||||||
|
github.com/dgrijalva/jwt-go v3.2.0+incompatible h1:7qlOGliEKZXTDg6OTjfoBKDXWrumCAMpl/TFQ4/5kLM=
|
||||||
|
github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
|
||||||
|
github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4=
|
||||||
|
github.com/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV9I=
|
||||||
|
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
|
||||||
|
github.com/go-telegram-bot-api/telegram-bot-api v4.6.5-0.20181225215658-ec221ba9ea45+incompatible h1:i64CCJcSqkRIkm5OSdZQjZq84/gJsk2zNwHWIRYWlKE=
|
||||||
|
github.com/go-telegram-bot-api/telegram-bot-api v4.6.5-0.20181225215658-ec221ba9ea45+incompatible/go.mod h1:qf9acutJ8cwBUhm1bqgz6Bei9/C/c93FPDljKWwsOgM=
|
||||||
|
github.com/golang/protobuf v1.2.0 h1:P3YflyNX/ehuJFLhxviNdFxQPkGK5cDcApsge1SqnvM=
|
||||||
|
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||||
|
github.com/google/gops v0.3.5 h1:SIWvPLiYvy5vMwjxB3rVFTE4QBhUFj2KKWr3Xm7CKhw=
|
||||||
|
github.com/google/gops v0.3.5/go.mod h1:pMQgrscwEK/aUSW1IFSaBPbJX82FPHWaSoJw1axQfD0=
|
||||||
|
github.com/gopackage/ddp v0.0.0-20170117053602-652027933df4 h1:4EZlYQIiyecYJlUbVkFXCXHz1QPhVXcHnQKAzBTPfQo=
|
||||||
|
github.com/gopackage/ddp v0.0.0-20170117053602-652027933df4/go.mod h1:lEO7XoHJ/xNRBCxrn4h/CEB67h0kW1B0t4ooP2yrjUA=
|
||||||
|
github.com/gopherjs/gopherjs v0.0.0-20180628210949-0892b62f0d9f h1:FDM3EtwZLyhW48YRiyqjivNlNZjAObv4xt4NnJaU+NQ=
|
||||||
|
github.com/gopherjs/gopherjs v0.0.0-20180628210949-0892b62f0d9f/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY=
|
||||||
|
github.com/gorilla/mux v1.6.2/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs=
|
||||||
|
github.com/gorilla/mux v1.7.0/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs=
|
||||||
|
github.com/gorilla/schema v1.0.2 h1:sAgNfOcNYvdDSrzGHVy9nzCQahG+qmsg+nE8dK85QRA=
|
||||||
|
github.com/gorilla/schema v1.0.2/go.mod h1:kgLaKoK1FELgZqMAVxx/5cbj0kT+57qxUrAlIO2eleU=
|
||||||
|
github.com/gorilla/websocket v1.4.0 h1:WDFjx/TMzVgy9VdMMQi2K2Emtwi2QcUQsztZ/zLaH/Q=
|
||||||
|
github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
|
||||||
|
github.com/hashicorp/golang-lru v0.5.0 h1:CL2msUPvZTLb5O648aiLNJw3hnBxN2+1Jq8rCOH9wdo=
|
||||||
|
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||||
|
github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
||||||
|
github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
|
||||||
|
github.com/howeyc/gopass v0.0.0-20170109162249-bf9dde6d0d2c/go.mod h1:lADxMC39cJJqL93Duh1xhAs4I2Zs8mKS89XWXFGp9cs=
|
||||||
|
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
||||||
|
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
||||||
|
github.com/jessevdk/go-flags v1.3.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI=
|
||||||
|
github.com/jpillora/backoff v0.0.0-20180909062703-3050d21c67d7 h1:K//n/AqR5HjG3qxbrBCL4vJPW0MVFSs9CPK1OOJdRME=
|
||||||
|
github.com/jpillora/backoff v0.0.0-20180909062703-3050d21c67d7/go.mod h1:2iMrUgbbvHEiQClaW2NsSzMyGHqN+rDFqY705q49KG0=
|
||||||
|
github.com/jtolds/gls v4.2.1+incompatible h1:fSuqC+Gmlu6l/ZYAoZzx2pyucC8Xza35fpRVWLVmUEE=
|
||||||
|
github.com/jtolds/gls v4.2.1+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU=
|
||||||
|
github.com/kardianos/osext v0.0.0-20170510131534-ae77be60afb1 h1:PJPDf8OUfOK1bb/NeTKd4f1QXZItOX389VN3B6qC8ro=
|
||||||
|
github.com/kardianos/osext v0.0.0-20170510131534-ae77be60afb1/go.mod h1:1NbS8ALrpOvjt0rHPNLyCIeMtbizbir8U//inJ+zuB8=
|
||||||
|
github.com/konsorten/go-windows-terminal-sequences v1.0.1 h1:mweAR1A6xJ3oS2pRaGiHgQ4OO8tzTaLawm8vnODuwDk=
|
||||||
|
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
||||||
|
github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI=
|
||||||
|
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
|
||||||
|
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
|
||||||
|
github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
|
||||||
|
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
|
||||||
|
github.com/labstack/echo/v4 v4.0.0 h1:q1GH+caIXPP7H2StPIdzy/ez9CO0EepqYeUg6vi9SWM=
|
||||||
|
github.com/labstack/echo/v4 v4.0.0/go.mod h1:tZv7nai5buKSg5h/8E6zz4LsD/Dqh9/91Mvs7Z5Zyno=
|
||||||
|
github.com/labstack/gommon v0.2.8 h1:JvRqmeZcfrHC5u6uVleB4NxxNbzx6gpbJiQknDbKQu0=
|
||||||
|
github.com/labstack/gommon v0.2.8/go.mod h1:/tj9csK2iPSBvn+3NLM9e52usepMtrd5ilFYA+wQNJ4=
|
||||||
|
github.com/lrstanley/girc v0.0.0-20190210212025-51b8e096d398 h1:a40kRmhA1p2XFJ6gqXfCExSyuDDCp/U9LA8ZY27u2Lk=
|
||||||
|
github.com/lrstanley/girc v0.0.0-20190210212025-51b8e096d398/go.mod h1:7cRs1SIBfKQ7e3Tam6GKTILSNHzR862JD0JpINaZoJk=
|
||||||
|
github.com/lusis/go-slackbot v0.0.0-20180109053408-401027ccfef5 h1:AsEBgzv3DhuYHI/GiQh2HxvTP71HCCE9E/tzGUzGdtU=
|
||||||
|
github.com/lusis/go-slackbot v0.0.0-20180109053408-401027ccfef5/go.mod h1:c2mYKRyMb1BPkO5St0c/ps62L4S0W2NAkaTXj9qEI+0=
|
||||||
|
github.com/lusis/slack-test v0.0.0-20180109053238-3c758769bfa6 h1:iOAVXzZyXtW408TMYejlUPo6BIn92HmOacWtIfNyYns=
|
||||||
|
github.com/lusis/slack-test v0.0.0-20180109053238-3c758769bfa6/go.mod h1:sFlOUpQL1YcjhFVXhg1CG8ZASEs/Mf1oVb6H75JL/zg=
|
||||||
|
github.com/magiconair/properties v1.8.0 h1:LLgXmsheXeRoUOBOjtwPQCWIYqM/LU1ayDtDePerRcY=
|
||||||
|
github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ=
|
||||||
|
github.com/matterbridge/Rocket.Chat.Go.SDK v0.0.0-20190210153444-cc9d05784d5d h1:F+Sr+C0ojSlYQ37BLylQtSFmyQULe3jbAygcyXQ9mVs=
|
||||||
|
github.com/matterbridge/Rocket.Chat.Go.SDK v0.0.0-20190210153444-cc9d05784d5d/go.mod h1:c6MxwqHD+0HvtAJjsHMIdPCiAwGiQwPRPTp69ACMg8A=
|
||||||
|
github.com/matterbridge/go-whatsapp v0.0.1-0.20190301204034-f2f1b29d441b h1:cO6Z+yj4Ivq/ay/IxSrV90oSIW/SSXWLa+XHsiLKMrw=
|
||||||
|
github.com/matterbridge/go-whatsapp v0.0.1-0.20190301204034-f2f1b29d441b/go.mod h1:dW19fYkkdUZsBAx7zv9fDh0n6NRqYIaKwB2JEBw8d0U=
|
||||||
|
github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91 h1:KzDEcy8eDbTx881giW8a6llsAck3e2bJvMyKvh1IK+k=
|
||||||
|
github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91/go.mod h1:ECDRehsR9TYTKCAsRS8/wLeOk6UUqDydw47ln7wG41Q=
|
||||||
|
github.com/matterbridge/gomatrix v0.0.0-20190102230110-6f9631ca6dea h1:kaADGqpK4gGO2BpzEyJrBxq2Jc57Rsar4i2EUxcACUc=
|
||||||
|
github.com/matterbridge/gomatrix v0.0.0-20190102230110-6f9631ca6dea/go.mod h1:+jWeaaUtXQbBRdKYWfjW6JDDYiI2XXE+3NnTjW5kg8g=
|
||||||
|
github.com/matterbridge/gozulipbot v0.0.0-20190212232658-7aa251978a18 h1:fLhwXtWGtfTgZVxHG1lcKjv+re7dRwyyuYFNu69xdho=
|
||||||
|
github.com/matterbridge/gozulipbot v0.0.0-20190212232658-7aa251978a18/go.mod h1:yAjnZ34DuDyPHMPHHjOsTk/FefW4JJjoMMCGt/8uuQA=
|
||||||
|
github.com/matterbridge/logrus-prefixed-formatter v0.0.0-20180806162718-01618749af61 h1:R/MgM/eUyRBQx2FiH6JVmXck8PaAuKfe2M1tWIzW7nE=
|
||||||
|
github.com/matterbridge/logrus-prefixed-formatter v0.0.0-20180806162718-01618749af61/go.mod h1:iXGEotOvwI1R1SjLxRc+BF5rUORTMtE0iMZBT2lxqAU=
|
||||||
|
github.com/matterbridge/mautrix-whatsapp v0.0.0-20190301210046-3539cf52ed6e h1:1NqciL8sz+0UYeFrd/UQlL8tJPhFxOBmg+a94DN2sJU=
|
||||||
|
github.com/matterbridge/mautrix-whatsapp v0.0.0-20190301210046-3539cf52ed6e/go.mod h1:DrIFGcFumRlEW5k3PJjWGKPd4+w37d3SwOxlh1ZAL+4=
|
||||||
|
github.com/mattermost/mattermost-server v5.5.0+incompatible h1:0wcLGgYtd+YImtLDPf2AOfpBHxbU4suATx+6XKw1XbU=
|
||||||
|
github.com/mattermost/mattermost-server v5.5.0+incompatible/go.mod h1:5L6MjAec+XXQwMIt791Ganu45GKsSiM+I0tLR9wUj8Y=
|
||||||
|
github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU=
|
||||||
|
github.com/mattn/go-colorable v0.1.1 h1:G1f5SKeVxmagw/IyvzvtZE4Gybcc4Tr1tf7I8z0XgOg=
|
||||||
|
github.com/mattn/go-colorable v0.1.1/go.mod h1:FuOcm+DKB9mbwrcAfNl7/TZVBZ6rcnceauSikq3lYCQ=
|
||||||
|
github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
|
||||||
|
github.com/mattn/go-isatty v0.0.5 h1:tHXDdz1cpzGaovsTB+TVB8q90WEokoVmfMqoVcrLUgw=
|
||||||
|
github.com/mattn/go-isatty v0.0.5/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
|
||||||
|
github.com/mattn/go-sqlite3 v1.10.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
|
||||||
|
github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b h1:j7+1HpAFS1zy5+Q4qx1fWh90gTKwiN4QCGoY9TWyyO4=
|
||||||
|
github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b/go.mod h1:01TrycV0kFyexm33Z7vhZRXopbI8J3TDReVlkTgMUxE=
|
||||||
|
github.com/mitchellh/mapstructure v1.1.2 h1:fmNYVwqnSfB9mZU6OS2O6GsXM+wcskZDuKQzvN1EDeE=
|
||||||
|
github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
|
||||||
|
github.com/mreiferson/go-httpclient v0.0.0-20160630210159-31f0106b4474 h1:oKIteTqeSpenyTrOVj5zkiyCaflLa8B+CD0324otT+o=
|
||||||
|
github.com/mreiferson/go-httpclient v0.0.0-20160630210159-31f0106b4474/go.mod h1:OQA4XLvDbMgS8P0CevmM4m9Q3Jq4phKUzcocxuGJ5m8=
|
||||||
|
github.com/mrexodia/wray v0.0.0-20160318003008-78a2c1f284ff h1:HLGD5/9UxxfEuO9DtP8gnTmNtMxbPyhYltfxsITel8g=
|
||||||
|
github.com/mrexodia/wray v0.0.0-20160318003008-78a2c1f284ff/go.mod h1:B8jLfIIPn2sKyWr0D7cL2v7tnrDD5z291s2Zypdu89E=
|
||||||
|
github.com/nelsonken/gomf v0.0.0-20180504123937-a9dd2f9deae9 h1:mp6tU1r0xLostUGLkTspf/9/AiHuVD7ptyXhySkDEsE=
|
||||||
|
github.com/nelsonken/gomf v0.0.0-20180504123937-a9dd2f9deae9/go.mod h1:A5SRAcpTemjGgIuBq6Kic2yHcoeUFWUinOAlMP/i9xo=
|
||||||
|
github.com/nicksnyder/go-i18n v1.4.0 h1:AgLl+Yq7kg5OYlzCgu9cKTZOyI4tD/NgukKqLqC8E+I=
|
||||||
|
github.com/nicksnyder/go-i18n v1.4.0/go.mod h1:HrK7VCrbOvQoUAQ7Vpy7i87N7JZZZ7R2xBGjv0j365Q=
|
||||||
|
github.com/nlopes/slack v0.5.0 h1:NbIae8Kd0NpqaEI3iUrsuS0KbcEDhzhc939jLW5fNm0=
|
||||||
|
github.com/nlopes/slack v0.5.0/go.mod h1:jVI4BBK3lSktibKahxBF74txcK2vyvkza1z/+rRnVAM=
|
||||||
|
github.com/onsi/ginkgo v1.6.0 h1:Ix8l273rp3QzYgXSR+c8d1fTG7UPgYkOSELPhiY/YGw=
|
||||||
|
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
||||||
|
github.com/onsi/gomega v1.4.1 h1:PZSj/UFNaVp3KxrzHOcS7oyuWA7LoOY/77yCTEFu21U=
|
||||||
|
github.com/onsi/gomega v1.4.1/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
|
||||||
|
github.com/paulrosania/go-charset v0.0.0-20151028000031-621bb39fcc83 h1:XQonH5Iv5rbyIkMJOQ4xKmKHQTh8viXtRSmep5Ca5I4=
|
||||||
|
github.com/paulrosania/go-charset v0.0.0-20151028000031-621bb39fcc83/go.mod h1:YnNlZP7l4MhyGQ4CBRwv6ohZTPrUJJZtEv4ZgADkbs4=
|
||||||
|
github.com/pborman/uuid v0.0.0-20160216163710-c55201b03606 h1:/CPgDYrfeK2LMK6xcUhvI17yO9SlpAdDIJGkhDEgO8A=
|
||||||
|
github.com/pborman/uuid v0.0.0-20160216163710-c55201b03606/go.mod h1:VyrYX9gd7irzKovcSS6BIIEwPRkP2Wm2m9ufcdFSJ34=
|
||||||
|
github.com/pelletier/go-toml v1.2.0 h1:T5zMGML61Wp+FlcbWjRDT7yAxhJNAiPPLOFECq181zc=
|
||||||
|
github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic=
|
||||||
|
github.com/peterhellberg/emojilib v0.0.0-20190124112554-c18758d55320 h1:YxcQy/DV+48NGv1lxx1vsWBzs6W1f1ogubkuCozxpX0=
|
||||||
|
github.com/peterhellberg/emojilib v0.0.0-20190124112554-c18758d55320/go.mod h1:G7LufuPajuIvdt9OitkNt2qh0mmvD4bfRgRM7bhDIOA=
|
||||||
|
github.com/pkg/errors v0.8.0 h1:WdK/asTD0HN+q6hsWO3/vpuAkAr+tw6aNJNDFFf0+qw=
|
||||||
|
github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
|
github.com/rs/xid v1.2.1 h1:mhH9Nq+C1fY2l1XIpgxIiUOfNpRBYH1kKcr+qfKgjRc=
|
||||||
|
github.com/rs/xid v1.2.1/go.mod h1:+uKXf+4Djp6Md1KODXJxgGQPKngRmWyn10oCKFzNHOQ=
|
||||||
|
github.com/russross/blackfriday v1.5.2 h1:HyvC0ARfnZBqnXwABFeSZHpKvJHJJfPz81GNueLj0oo=
|
||||||
|
github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g=
|
||||||
|
github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
||||||
|
github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca h1:NugYot0LIVPxTvN8n+Kvkn6TrbMyxQiuvKdEwFdR9vI=
|
||||||
|
github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca/go.mod h1:uugorj2VCxiV1x+LzaIdVa9b4S4qGAcH6cbhh4qVxOU=
|
||||||
|
github.com/shazow/rateio v0.0.0-20150116013248-e8e00881e5c1 h1:Lx3BlDGFElJt4u/zKc9A3BuGYbQAGlEFyPuUA3jeMD0=
|
||||||
|
github.com/shazow/rateio v0.0.0-20150116013248-e8e00881e5c1/go.mod h1:vt2jWY/3Qw1bIzle5thrJWucsLuuX9iUNnp20CqCciI=
|
||||||
|
github.com/shazow/ssh-chat v0.0.0-20190125184227-81d7e1686296 h1:8RLq547MSVc6vhOuCl4Ca0TsAQknj6NX6ZLSZ3+xmio=
|
||||||
|
github.com/shazow/ssh-chat v0.0.0-20190125184227-81d7e1686296/go.mod h1:1GLXsL4esywkpNId3v4QWuMf3THtWGitWvtQ/L3aSA4=
|
||||||
|
github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
|
||||||
|
github.com/sirupsen/logrus v1.3.0 h1:hI/7Q+DtNZ2kINb6qt/lS+IyXnHQe9e90POfeewL/ME=
|
||||||
|
github.com/sirupsen/logrus v1.3.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
|
||||||
|
github.com/skip2/go-qrcode v0.0.0-20190110000554-dc11ecdae0a9 h1:lpEzuenPuO1XNTeikEmvqYFcU37GVLl8SRNblzyvGBE=
|
||||||
|
github.com/skip2/go-qrcode v0.0.0-20190110000554-dc11ecdae0a9/go.mod h1:PLPIyL7ikehBD1OAjmKKiOEhbvWyHGaNDjquXMcYABo=
|
||||||
|
github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9 h1:lXQ+j+KwZcbwrbgU0Rp4Eglg3EJLHbuZU3BbOqAGBmg=
|
||||||
|
github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc=
|
||||||
|
github.com/smartystreets/goconvey v0.0.0-20180222194500-ef6db91d284a h1:JSvGDIbmil4Ui/dDdFBExb7/cmkNjyX5F97oglmvCDo=
|
||||||
|
github.com/smartystreets/goconvey v0.0.0-20180222194500-ef6db91d284a/go.mod h1:XDJAKZRPZ1CvBcN2aX5YOUTYGHki24fSF0Iv48Ibg0s=
|
||||||
|
github.com/spf13/afero v1.1.2 h1:m8/z1t7/fwjysjQRYbP0RD+bUIF/8tJwPdEZsI83ACI=
|
||||||
|
github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ=
|
||||||
|
github.com/spf13/cast v1.3.0 h1:oget//CVOEoFewqQxwr0Ej5yjygnqGkvggSE/gB35Q8=
|
||||||
|
github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE=
|
||||||
|
github.com/spf13/jwalterweatherman v1.0.0 h1:XHEdyB+EcvlqZamSM4ZOMGlc93t6AcsBEu9Gc1vn7yk=
|
||||||
|
github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo=
|
||||||
|
github.com/spf13/pflag v1.0.3 h1:zPAT6CGy6wXeQ7NtTnaTerfKOsV6V6F8agHXFiazDkg=
|
||||||
|
github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
|
||||||
|
github.com/spf13/viper v1.3.1 h1:5+8j8FTpnFV4nEImW/ofkzEt8VoOiLXxdYIDsB73T38=
|
||||||
|
github.com/spf13/viper v1.3.1/go.mod h1:ZiWeW+zYFKm7srdB9IoDzzZXaJaI5eL9QjNiN/DMA2s=
|
||||||
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
|
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
|
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
||||||
|
github.com/stretchr/testify v1.3.0 h1:TivCn/peBQ7UY8ooIcPgZFpTNSz0Q2U6UrFlUfqbe0Q=
|
||||||
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
|
github.com/technoweenie/multipartstreamer v1.0.1 h1:XRztA5MXiR1TIRHxH2uNxXxaIkKQDeX7m2XsSOlQEnM=
|
||||||
|
github.com/technoweenie/multipartstreamer v1.0.1/go.mod h1:jNVxdtShOxzAsukZwTSw6MDx5eUJoiEBsSvzDU9uzog=
|
||||||
|
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
|
||||||
|
github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw=
|
||||||
|
github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=
|
||||||
|
github.com/valyala/fasttemplate v0.0.0-20170224212429-dcecefd839c4 h1:gKMu1Bf6QINDnvyZuTaACm9ofY+PRh+5vFz4oxBZeF8=
|
||||||
|
github.com/valyala/fasttemplate v0.0.0-20170224212429-dcecefd839c4/go.mod h1:50wTf68f99/Zt14pr046Tgt3Lp2vLyFZKzbFXTOabXw=
|
||||||
|
github.com/x-cray/logrus-prefixed-formatter v0.5.2 h1:00txxvfBM9muc0jiLIEAkAcIMJzfthRT6usrui8uGmg=
|
||||||
|
github.com/x-cray/logrus-prefixed-formatter v0.5.2/go.mod h1:2duySbKsL6M18s5GU7VPsoEPHyzalCE06qoARUCeBBE=
|
||||||
|
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
|
||||||
|
github.com/zfjagann/golang-ring v0.0.0-20190106091943-a88bb6aef447 h1:CHgPZh8bFkZmislPrr/0gd7MciDAX+JJB70A2/5Lvmo=
|
||||||
|
github.com/zfjagann/golang-ring v0.0.0-20190106091943-a88bb6aef447/go.mod h1:0MsIttMJIF/8Y7x0XjonJP7K99t3sR6bjj4m5S4JmqU=
|
||||||
|
gitlab.com/golang-commonmark/html v0.0.0-20180917080848-cfaf75183c4a h1:Ax7kdHNICZiIeFpmevmaEWb0Ae3BUj3zCTKhZHZ+zd0=
|
||||||
|
gitlab.com/golang-commonmark/html v0.0.0-20180917080848-cfaf75183c4a/go.mod h1:JT4uoTz0tfPoyVH88GZoWDNm5NHJI2VbUW+eyPClueI=
|
||||||
|
gitlab.com/golang-commonmark/linkify v0.0.0-20180917065525-c22b7bdb1179 h1:rbON2KwBnWuFMlSHM8LELLlwroDRZw6xv0e6il6e5dk=
|
||||||
|
gitlab.com/golang-commonmark/linkify v0.0.0-20180917065525-c22b7bdb1179/go.mod h1:Gn+LZmCrhPECMD3SOKlE+BOHwhOYD9j7WT9NUtkCrC8=
|
||||||
|
gitlab.com/golang-commonmark/markdown v0.0.0-20181102083822-772775880e1f h1:jwXy/CsM4xS2aoiF2fHAlukmInWhd2TlWB+HDCyvzKc=
|
||||||
|
gitlab.com/golang-commonmark/markdown v0.0.0-20181102083822-772775880e1f/go.mod h1:SIHlEr9462fpIfTrVWf3GqQDxnA65Vm3BMMsUtuA6W0=
|
||||||
|
gitlab.com/golang-commonmark/mdurl v0.0.0-20180912090424-e5bce34c34f2 h1:wD/sPUgx2QJFPTyXZpJnLaROolfeKuruh06U4pRV0WY=
|
||||||
|
gitlab.com/golang-commonmark/mdurl v0.0.0-20180912090424-e5bce34c34f2/go.mod h1:wQk4rLkWrdOPjUAtqJRJ10hIlseLSVYWP95PLrjDF9s=
|
||||||
|
gitlab.com/golang-commonmark/puny v0.0.0-20180912090636-2cd490539afe h1:5kUPFAF52umOUPH12MuNUmyVTseJRNBftDl/KfsvX3I=
|
||||||
|
gitlab.com/golang-commonmark/puny v0.0.0-20180912090636-2cd490539afe/go.mod h1:P9LSM1KVzrIstFgUaveuwiAm8PK5VTB3yJEU8kqlbrU=
|
||||||
|
gitlab.com/opennota/wd v0.0.0-20180912061657-c5d65f63c638 h1:uPZaMiz6Sz0PZs3IZJWpU5qHKGNy///1pacZC9txiUI=
|
||||||
|
gitlab.com/opennota/wd v0.0.0-20180912061657-c5d65f63c638/go.mod h1:EGRJaqe2eO9XGmFtQCvV3Lm9NLico3UhFwUpCG/+mVU=
|
||||||
|
go.uber.org/atomic v1.3.2 h1:2Oa65PReHzfn29GpvgsYwloV9AVFHPDk8tYxt2c2tr4=
|
||||||
|
go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
|
||||||
|
go.uber.org/multierr v1.1.0 h1:HoEmRHQPVSqub6w2z2d2EOVs2fjyFRGyofhKuyDq0QI=
|
||||||
|
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
|
||||||
|
go.uber.org/zap v1.9.1 h1:XCJQEf3W6eZaVwhRBof6ImoYGJSITeKWsyeh3HFu/5o=
|
||||||
|
go.uber.org/zap v1.9.1/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
|
||||||
|
golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
|
golang.org/x/crypto v0.0.0-20181030102418-4d3f4d9ffa16/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
|
golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
|
golang.org/x/crypto v0.0.0-20190130090550-b01c7a725664/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
|
golang.org/x/crypto v0.0.0-20190131182504-b8fe1690c613/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
|
golang.org/x/crypto v0.0.0-20190222235706-ffb98f73852f h1:qWFY9ZxP3tfI37wYIs/MnIAqK0vlXp1xnYEa5HxFSSY=
|
||||||
|
golang.org/x/crypto v0.0.0-20190222235706-ffb98f73852f/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
|
golang.org/x/image v0.0.0-20190220214146-31aff87c08e9 h1:+vH8qNweCrORN49012OX3h0oWEXO3p+rRnpAGQinddk=
|
||||||
|
golang.org/x/image v0.0.0-20190220214146-31aff87c08e9/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js=
|
||||||
|
golang.org/x/net v0.0.0-20190110200230-915654e7eabc h1:Yx9JGxI1SBhVLFjpAkWMaO1TF+xyqtHLjZpvQboJGiM=
|
||||||
|
golang.org/x/net v0.0.0-20190110200230-915654e7eabc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
|
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4 h1:YUO/7uOKsKeq9UokNS62b8FYywz3ker1l1vDZRCRefw=
|
||||||
|
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/sys v0.0.0-20190129075346-302c3dd5f1cc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/sys v0.0.0-20190222171317-cd391775e71e h1:oF7qaQxUH6KzFdKN4ww7NpPdo53SZi4UlcksLrb2y/o=
|
||||||
|
golang.org/x/sys v0.0.0-20190222171317-cd391775e71e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg=
|
||||||
|
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY=
|
||||||
|
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4=
|
||||||
|
gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys=
|
||||||
|
gopkg.in/natefinch/lumberjack.v2 v2.0.0 h1:1Lc07Kr7qY4U2YPouBjpCLxpiyxIVoxqXgkXLknAOE8=
|
||||||
|
gopkg.in/natefinch/lumberjack.v2 v2.0.0/go.mod h1:l0ndWWf7gzL7RNwBG7wST/UCcT4T24xpD6X8LsfU/+k=
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ=
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
|
||||||
|
gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw=
|
||||||
|
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
|
maunium.net/go/mauflag v1.0.0/go.mod h1:nLivPOpTpHnpzEh8jEdSL9UqO9+/KBJFmNRlwKfkPeA=
|
||||||
|
maunium.net/go/maulogger/v2 v2.0.0/go.mod h1:Hbbkq3NV6jvJodByZu1mgEF3fpT7Kz9z0MjEZ3/BusI=
|
||||||
|
maunium.net/go/mautrix v0.1.0-alpha.3/go.mod h1:GTVu6WDHR+98DKOrYetWsXorvUeKQV3jsSWO6ScbuFI=
|
||||||
|
maunium.net/go/mautrix-appservice v0.1.0-alpha.3/go.mod h1:wOnWOIuprYad7ly12rHIo3JLCPh4jwvx1prVrAB9RhM=
|
@ -38,7 +38,7 @@ type Config struct {
|
|||||||
func New(url string, config Config) *Client {
|
func New(url string, config Config) *Client {
|
||||||
c := &Client{In: make(chan Message), Config: config}
|
c := &Client{In: make(chan Message), Config: config}
|
||||||
tr := &http.Transport{
|
tr := &http.Transport{
|
||||||
TLSClientConfig: &tls.Config{InsecureSkipVerify: config.InsecureSkipVerify},
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: config.InsecureSkipVerify}, //nolint:gosec
|
||||||
}
|
}
|
||||||
c.httpclient = &http.Client{Transport: tr}
|
c.httpclient = &http.Client{Transport: tr}
|
||||||
_, _, err := net.SplitHostPort(c.BindAddress)
|
_, _, err := net.SplitHostPort(c.BindAddress)
|
||||||
@ -99,10 +99,9 @@ func (c *Client) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
// Receive returns an incoming message from mattermost outgoing webhooks URL.
|
// Receive returns an incoming message from mattermost outgoing webhooks URL.
|
||||||
func (c *Client) Receive() Message {
|
func (c *Client) Receive() Message {
|
||||||
for {
|
var msg Message
|
||||||
select {
|
for msg = range c.In {
|
||||||
case msg := <-c.In:
|
return msg
|
||||||
return msg
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
return msg
|
||||||
}
|
}
|
||||||
|
BIN
img/matterbridge-notext.gif
Normal file
After Width: | Height: | Size: 70 KiB |
BIN
img/matterbridge.gif
Normal file
After Width: | Height: | Size: 77 KiB |
BIN
img/slack-setup-add-scopes.png
Normal file
After Width: | Height: | Size: 270 KiB |
BIN
img/slack-setup-app-page.png
Normal file
After Width: | Height: | Size: 170 KiB |
BIN
img/slack-setup-create-app.png
Normal file
After Width: | Height: | Size: 282 KiB |
BIN
img/slack-setup-create-bot.png
Normal file
After Width: | Height: | Size: 204 KiB |
BIN
img/slack-setup-finished.png
Normal file
After Width: | Height: | Size: 48 KiB |
BIN
img/slack-setup-install-app.png
Normal file
After Width: | Height: | Size: 73 KiB |
BIN
img/slack-setup-invite-bot.png
Normal file
After Width: | Height: | Size: 62 KiB |
100
matterbridge.go
@ -3,57 +3,83 @@ package main
|
|||||||
import (
|
import (
|
||||||
"flag"
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/42wim/matterbridge/bridge/config"
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
"github.com/42wim/matterbridge/gateway"
|
"github.com/42wim/matterbridge/gateway"
|
||||||
"github.com/42wim/matterbridge/gateway/samechannel"
|
"github.com/42wim/matterbridge/gateway/bridgemap"
|
||||||
log "github.com/Sirupsen/logrus"
|
"github.com/google/gops/agent"
|
||||||
|
prefixed "github.com/matterbridge/logrus-prefixed-formatter"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
var version = "0.9.2"
|
var (
|
||||||
|
version = "1.14.1"
|
||||||
|
githash string
|
||||||
|
|
||||||
func init() {
|
flagConfig = flag.String("conf", "matterbridge.toml", "config file")
|
||||||
log.SetFormatter(&log.TextFormatter{FullTimestamp: true})
|
flagDebug = flag.Bool("debug", false, "enable debug")
|
||||||
}
|
flagVersion = flag.Bool("version", false, "show version")
|
||||||
|
flagGops = flag.Bool("gops", false, "enable gops agent")
|
||||||
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
flagConfig := flag.String("conf", "matterbridge.toml", "config file")
|
|
||||||
flagDebug := flag.Bool("debug", false, "enable debug")
|
|
||||||
flagVersion := flag.Bool("version", false, "show version")
|
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
if *flagVersion {
|
if *flagVersion {
|
||||||
fmt.Println("version:", version)
|
fmt.Printf("version: %s %s\n", version, githash)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
flag.Parse()
|
|
||||||
if *flagDebug {
|
rootLogger := setupLogger()
|
||||||
log.Info("enabling debug")
|
logger := rootLogger.WithFields(logrus.Fields{"prefix": "main"})
|
||||||
log.SetLevel(log.DebugLevel)
|
|
||||||
}
|
if *flagGops {
|
||||||
fmt.Println("running version", version)
|
if err := agent.Listen(agent.Options{}); err != nil {
|
||||||
cfg := config.NewConfig(*flagConfig)
|
logger.Errorf("Failed to start gops agent: %#v", err)
|
||||||
for _, gw := range cfg.SameChannelGateway {
|
} else {
|
||||||
if !gw.Enable {
|
defer agent.Close()
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
fmt.Printf("starting samechannel gateway %#v\n", gw.Name)
|
|
||||||
go func(gw config.SameChannelGateway) {
|
|
||||||
err := samechannelgateway.New(cfg, &gw)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("starting gateway failed %#v", err)
|
|
||||||
}
|
|
||||||
}(gw)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, gw := range cfg.Gateway {
|
logger.Printf("Running version %s %s", version, githash)
|
||||||
if !gw.Enable {
|
if strings.Contains(version, "-dev") {
|
||||||
continue
|
logger.Println("WARNING: THIS IS A DEVELOPMENT VERSION. Things may break.")
|
||||||
}
|
|
||||||
fmt.Printf("starting gateway %#v\n", gw.Name)
|
|
||||||
g := gateway.New(cfg, &gw)
|
|
||||||
err := g.Start()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("starting gateway failed %#v", err)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cfg := config.NewConfig(rootLogger, *flagConfig)
|
||||||
|
cfg.BridgeValues().General.Debug = *flagDebug
|
||||||
|
|
||||||
|
r, err := gateway.NewRouter(rootLogger, cfg, bridgemap.FullMap)
|
||||||
|
if err != nil {
|
||||||
|
logger.Fatalf("Starting gateway failed: %s", err)
|
||||||
|
}
|
||||||
|
if err = r.Start(); err != nil {
|
||||||
|
logger.Fatalf("Starting gateway failed: %s", err)
|
||||||
|
}
|
||||||
|
logger.Printf("Gateway(s) started succesfully. Now relaying messages")
|
||||||
select {}
|
select {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func setupLogger() *logrus.Logger {
|
||||||
|
logger := &logrus.Logger{
|
||||||
|
Out: os.Stdout,
|
||||||
|
Formatter: &prefixed.TextFormatter{
|
||||||
|
PrefixPadding: 13,
|
||||||
|
DisableColors: true,
|
||||||
|
FullTimestamp: true,
|
||||||
|
},
|
||||||
|
Level: logrus.InfoLevel,
|
||||||
|
}
|
||||||
|
if *flagDebug || os.Getenv("DEBUG") == "1" {
|
||||||
|
logger.Formatter = &prefixed.TextFormatter{
|
||||||
|
PrefixPadding: 13,
|
||||||
|
DisableColors: true,
|
||||||
|
FullTimestamp: false,
|
||||||
|
ForceFormatting: true,
|
||||||
|
}
|
||||||
|
logger.Level = logrus.DebugLevel
|
||||||
|
logger.WithFields(logrus.Fields{"prefix": "main"}).Info("Enabling debug logging.")
|
||||||
|
}
|
||||||
|
return logger
|
||||||
|
}
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
#WARNING: as this file contains credentials, be sure to set correct file permissions
|
||||||
[irc]
|
[irc]
|
||||||
[irc.freenode]
|
[irc.freenode]
|
||||||
Server="irc.freenode.net:6667"
|
Server="irc.freenode.net:6667"
|
||||||
@ -5,8 +6,8 @@
|
|||||||
|
|
||||||
[mattermost]
|
[mattermost]
|
||||||
[mattermost.work]
|
[mattermost.work]
|
||||||
useAPI=true
|
#do not prefix it wit http:// or https://
|
||||||
Server="yourmattermostserver.domain"
|
Server="yourmattermostserver.domain"
|
||||||
Team="yourteam"
|
Team="yourteam"
|
||||||
Login="yourlogin"
|
Login="yourlogin"
|
||||||
Password="yourpass"
|
Password="yourpass"
|
||||||
@ -15,18 +16,19 @@
|
|||||||
[[gateway]]
|
[[gateway]]
|
||||||
name="gateway1"
|
name="gateway1"
|
||||||
enable=true
|
enable=true
|
||||||
[[gateway.in]]
|
[[gateway.inout]]
|
||||||
account="irc.freenode"
|
account="irc.freenode"
|
||||||
channel="#testing"
|
channel="#testing"
|
||||||
|
|
||||||
[[gateway.out]]
|
[[gateway.inout]]
|
||||||
account="irc.freenode"
|
|
||||||
channel="#testing"
|
|
||||||
|
|
||||||
[[gateway.in]]
|
|
||||||
account="mattermost.work"
|
account="mattermost.work"
|
||||||
channel="off-topic"
|
channel="off-topic"
|
||||||
|
|
||||||
[[gateway.out]]
|
#simpler config possible since v0.10.2
|
||||||
account="mattermost.work"
|
#[[gateway]]
|
||||||
channel="off-topic"
|
#name="gateway2"
|
||||||
|
#enable=true
|
||||||
|
#inout = [
|
||||||
|
# { account="irc.freenode", channel="#testing", options={key="channelkey"}},
|
||||||
|
# { account="mattermost.work", channel="off-topic" },
|
||||||
|
#]
|
||||||
|
207
matterclient/channels.go
Normal file
@ -0,0 +1,207 @@
|
|||||||
|
package matterclient
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/mattermost/mattermost-server/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GetChannels returns all channels we're members off
|
||||||
|
func (m *MMClient) GetChannels() []*model.Channel {
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
var channels []*model.Channel
|
||||||
|
// our primary team channels first
|
||||||
|
channels = append(channels, m.Team.Channels...)
|
||||||
|
for _, t := range m.OtherTeams {
|
||||||
|
if t.Id != m.Team.Id {
|
||||||
|
channels = append(channels, t.Channels...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return channels
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetChannelHeader(channelId string) string { //nolint:golint
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
for _, t := range m.OtherTeams {
|
||||||
|
for _, channel := range append(t.Channels, t.MoreChannels...) {
|
||||||
|
if channel.Id == channelId {
|
||||||
|
return channel.Header
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetChannelId(name string, teamId string) string { //nolint:golint
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
if teamId != "" {
|
||||||
|
return m.getChannelIdTeam(name, teamId)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, t := range m.OtherTeams {
|
||||||
|
for _, channel := range append(t.Channels, t.MoreChannels...) {
|
||||||
|
if channel.Type == model.CHANNEL_GROUP {
|
||||||
|
res := strings.Replace(channel.DisplayName, ", ", "-", -1)
|
||||||
|
res = strings.Replace(res, " ", "_", -1)
|
||||||
|
if res == name {
|
||||||
|
return channel.Id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) getChannelIdTeam(name string, teamId string) string { //nolint:golint
|
||||||
|
for _, t := range m.OtherTeams {
|
||||||
|
if t.Id == teamId {
|
||||||
|
for _, channel := range append(t.Channels, t.MoreChannels...) {
|
||||||
|
if channel.Name == name {
|
||||||
|
return channel.Id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetChannelName(channelId string) string { //nolint:golint
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
for _, t := range m.OtherTeams {
|
||||||
|
if t == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
for _, channel := range append(t.Channels, t.MoreChannels...) {
|
||||||
|
if channel.Id == channelId {
|
||||||
|
if channel.Type == model.CHANNEL_GROUP {
|
||||||
|
res := strings.Replace(channel.DisplayName, ", ", "-", -1)
|
||||||
|
res = strings.Replace(res, " ", "_", -1)
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
return channel.Name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetChannelTeamId(id string) string { //nolint:golint
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
for _, t := range append(m.OtherTeams, m.Team) {
|
||||||
|
for _, channel := range append(t.Channels, t.MoreChannels...) {
|
||||||
|
if channel.Id == id {
|
||||||
|
return channel.TeamId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetLastViewedAt(channelId string) int64 { //nolint:golint
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
res, resp := m.Client.GetChannelMember(channelId, m.User.Id, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return model.GetMillis()
|
||||||
|
}
|
||||||
|
return res.LastViewedAt
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetMoreChannels returns existing channels where we're not a member off.
|
||||||
|
func (m *MMClient) GetMoreChannels() []*model.Channel {
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
var channels []*model.Channel
|
||||||
|
for _, t := range m.OtherTeams {
|
||||||
|
channels = append(channels, t.MoreChannels...)
|
||||||
|
}
|
||||||
|
return channels
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetTeamFromChannel returns teamId belonging to channel (DM channels have no teamId).
|
||||||
|
func (m *MMClient) GetTeamFromChannel(channelId string) string { //nolint:golint
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
var channels []*model.Channel
|
||||||
|
for _, t := range m.OtherTeams {
|
||||||
|
channels = append(channels, t.Channels...)
|
||||||
|
if t.MoreChannels != nil {
|
||||||
|
channels = append(channels, t.MoreChannels...)
|
||||||
|
}
|
||||||
|
for _, c := range channels {
|
||||||
|
if c.Id == channelId {
|
||||||
|
if c.Type == model.CHANNEL_GROUP {
|
||||||
|
return "G"
|
||||||
|
}
|
||||||
|
return t.Id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
channels = nil
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) JoinChannel(channelId string) error { //nolint:golint
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
for _, c := range m.Team.Channels {
|
||||||
|
if c.Id == channelId {
|
||||||
|
m.logger.Debug("Not joining ", channelId, " already joined.")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
m.logger.Debug("Joining ", channelId)
|
||||||
|
_, resp := m.Client.AddChannelMember(channelId, m.User.Id)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UpdateChannels() error {
|
||||||
|
mmchannels, resp := m.Client.GetChannelsForTeamForUser(m.Team.Id, m.User.Id, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return errors.New(resp.Error.DetailedError)
|
||||||
|
}
|
||||||
|
m.Lock()
|
||||||
|
m.Team.Channels = mmchannels
|
||||||
|
m.Unlock()
|
||||||
|
|
||||||
|
mmchannels, resp = m.Client.GetPublicChannelsForTeam(m.Team.Id, 0, 5000, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return errors.New(resp.Error.DetailedError)
|
||||||
|
}
|
||||||
|
|
||||||
|
m.Lock()
|
||||||
|
m.Team.MoreChannels = mmchannels
|
||||||
|
m.Unlock()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UpdateChannelHeader(channelId string, header string) { //nolint:golint
|
||||||
|
channel := &model.Channel{Id: channelId, Header: header}
|
||||||
|
m.logger.Debugf("updating channelheader %#v, %#v", channelId, header)
|
||||||
|
_, resp := m.Client.UpdateChannel(channel)
|
||||||
|
if resp.Error != nil {
|
||||||
|
m.logger.Error(resp.Error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UpdateLastViewed(channelId string) error { //nolint:golint
|
||||||
|
m.logger.Debugf("posting lastview %#v", channelId)
|
||||||
|
view := &model.ChannelView{ChannelId: channelId}
|
||||||
|
_, resp := m.Client.ViewChannel(m.User.Id, view)
|
||||||
|
if resp.Error != nil {
|
||||||
|
m.logger.Errorf("ChannelView update for %s failed: %s", channelId, resp.Error)
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
282
matterclient/helpers.go
Normal file
@ -0,0 +1,282 @@
|
|||||||
|
package matterclient
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/md5" //nolint:gosec
|
||||||
|
"crypto/tls"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/http/cookiejar"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gorilla/websocket"
|
||||||
|
"github.com/jpillora/backoff"
|
||||||
|
"github.com/mattermost/mattermost-server/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (m *MMClient) doLogin(firstConnection bool, b *backoff.Backoff) error {
|
||||||
|
var resp *model.Response
|
||||||
|
var appErr *model.AppError
|
||||||
|
var logmsg = "trying login"
|
||||||
|
var err error
|
||||||
|
for {
|
||||||
|
m.logger.Debugf("%s %s %s %s", logmsg, m.Credentials.Team, m.Credentials.Login, m.Credentials.Server)
|
||||||
|
if m.Credentials.Token != "" {
|
||||||
|
resp, err = m.doLoginToken()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
m.User, resp = m.Client.Login(m.Credentials.Login, m.Credentials.Pass)
|
||||||
|
}
|
||||||
|
appErr = resp.Error
|
||||||
|
if appErr != nil {
|
||||||
|
d := b.Duration()
|
||||||
|
m.logger.Debug(appErr.DetailedError)
|
||||||
|
if firstConnection {
|
||||||
|
if appErr.Message == "" {
|
||||||
|
return errors.New(appErr.DetailedError)
|
||||||
|
}
|
||||||
|
return errors.New(appErr.Message)
|
||||||
|
}
|
||||||
|
m.logger.Debugf("LOGIN: %s, reconnecting in %s", appErr, d)
|
||||||
|
time.Sleep(d)
|
||||||
|
logmsg = "retrying login"
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
// reset timer
|
||||||
|
b.Reset()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) doLoginToken() (*model.Response, error) {
|
||||||
|
var resp *model.Response
|
||||||
|
var logmsg = "trying login"
|
||||||
|
m.Client.AuthType = model.HEADER_BEARER
|
||||||
|
m.Client.AuthToken = m.Credentials.Token
|
||||||
|
if m.Credentials.CookieToken {
|
||||||
|
m.logger.Debugf(logmsg + " with cookie (MMAUTH) token")
|
||||||
|
m.Client.HttpClient.Jar = m.createCookieJar(m.Credentials.Token)
|
||||||
|
} else {
|
||||||
|
m.logger.Debugf(logmsg + " with personal token")
|
||||||
|
}
|
||||||
|
m.User, resp = m.Client.GetMe("")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp, resp.Error
|
||||||
|
}
|
||||||
|
if m.User == nil {
|
||||||
|
m.logger.Errorf("LOGIN TOKEN: %s is invalid", m.Credentials.Pass)
|
||||||
|
return resp, errors.New("invalid token")
|
||||||
|
}
|
||||||
|
return resp, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) handleLoginToken() error {
|
||||||
|
switch {
|
||||||
|
case strings.Contains(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN):
|
||||||
|
token := strings.Split(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN+"=")
|
||||||
|
if len(token) != 2 {
|
||||||
|
return errors.New("incorrect MMAUTHTOKEN. valid input is MMAUTHTOKEN=yourtoken")
|
||||||
|
}
|
||||||
|
m.Credentials.Token = token[1]
|
||||||
|
m.Credentials.CookieToken = true
|
||||||
|
case strings.Contains(m.Credentials.Pass, "token="):
|
||||||
|
token := strings.Split(m.Credentials.Pass, "token=")
|
||||||
|
if len(token) != 2 {
|
||||||
|
return errors.New("incorrect personal token. valid input is token=yourtoken")
|
||||||
|
}
|
||||||
|
m.Credentials.Token = token[1]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) initClient(firstConnection bool, b *backoff.Backoff) error {
|
||||||
|
uriScheme := "https://"
|
||||||
|
if m.NoTLS {
|
||||||
|
uriScheme = "http://"
|
||||||
|
}
|
||||||
|
// login to mattermost
|
||||||
|
m.Client = model.NewAPIv4Client(uriScheme + m.Credentials.Server)
|
||||||
|
m.Client.HttpClient.Transport = &http.Transport{
|
||||||
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}, //nolint:gosec
|
||||||
|
Proxy: http.ProxyFromEnvironment,
|
||||||
|
}
|
||||||
|
m.Client.HttpClient.Timeout = time.Second * 10
|
||||||
|
|
||||||
|
// handle MMAUTHTOKEN and personal token
|
||||||
|
if err := m.handleLoginToken(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if server alive, retry until
|
||||||
|
if err := m.serverAlive(firstConnection, b); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// initialize user and teams
|
||||||
|
func (m *MMClient) initUser() error {
|
||||||
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
|
// we only load all team data on initial login.
|
||||||
|
// all other updates are for channels from our (primary) team only.
|
||||||
|
//m.logger.Debug("initUser(): loading all team data")
|
||||||
|
teams, resp := m.Client.GetTeamsForUser(m.User.Id, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
for _, team := range teams {
|
||||||
|
mmusers, resp := m.Client.GetUsersInTeam(team.Id, 0, 50000, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return errors.New(resp.Error.DetailedError)
|
||||||
|
}
|
||||||
|
usermap := make(map[string]*model.User)
|
||||||
|
for _, user := range mmusers {
|
||||||
|
usermap[user.Id] = user
|
||||||
|
}
|
||||||
|
|
||||||
|
t := &Team{Team: team, Users: usermap, Id: team.Id}
|
||||||
|
|
||||||
|
mmchannels, resp := m.Client.GetChannelsForTeamForUser(team.Id, m.User.Id, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
t.Channels = mmchannels
|
||||||
|
mmchannels, resp = m.Client.GetPublicChannelsForTeam(team.Id, 0, 5000, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
t.MoreChannels = mmchannels
|
||||||
|
m.OtherTeams = append(m.OtherTeams, t)
|
||||||
|
if team.Name == m.Credentials.Team {
|
||||||
|
m.Team = t
|
||||||
|
m.logger.Debugf("initUser(): found our team %s (id: %s)", team.Name, team.Id)
|
||||||
|
}
|
||||||
|
// add all users
|
||||||
|
for k, v := range t.Users {
|
||||||
|
m.Users[k] = v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) serverAlive(firstConnection bool, b *backoff.Backoff) error {
|
||||||
|
defer b.Reset()
|
||||||
|
for {
|
||||||
|
d := b.Duration()
|
||||||
|
// bogus call to get the serverversion
|
||||||
|
_, resp := m.Client.Logout()
|
||||||
|
if resp.Error != nil {
|
||||||
|
return fmt.Errorf("%#v", resp.Error.Error())
|
||||||
|
}
|
||||||
|
if firstConnection && !supportedVersion(resp.ServerVersion) {
|
||||||
|
return fmt.Errorf("unsupported mattermost version: %s", resp.ServerVersion)
|
||||||
|
}
|
||||||
|
m.ServerVersion = resp.ServerVersion
|
||||||
|
if m.ServerVersion == "" {
|
||||||
|
m.logger.Debugf("Server not up yet, reconnecting in %s", d)
|
||||||
|
time.Sleep(d)
|
||||||
|
} else {
|
||||||
|
m.logger.Infof("Found version %s", m.ServerVersion)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) wsConnect() {
|
||||||
|
b := &backoff.Backoff{
|
||||||
|
Min: time.Second,
|
||||||
|
Max: 5 * time.Minute,
|
||||||
|
Jitter: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
m.WsConnected = false
|
||||||
|
wsScheme := "wss://"
|
||||||
|
if m.NoTLS {
|
||||||
|
wsScheme = "ws://"
|
||||||
|
}
|
||||||
|
|
||||||
|
// setup websocket connection
|
||||||
|
wsurl := wsScheme + m.Credentials.Server + model.API_URL_SUFFIX_V4 + "/websocket"
|
||||||
|
header := http.Header{}
|
||||||
|
header.Set(model.HEADER_AUTH, "BEARER "+m.Client.AuthToken)
|
||||||
|
|
||||||
|
m.logger.Debugf("WsClient: making connection: %s", wsurl)
|
||||||
|
for {
|
||||||
|
wsDialer := &websocket.Dialer{
|
||||||
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}, //nolint:gosec
|
||||||
|
Proxy: http.ProxyFromEnvironment,
|
||||||
|
}
|
||||||
|
var err error
|
||||||
|
m.WsClient, _, err = wsDialer.Dial(wsurl, header)
|
||||||
|
if err != nil {
|
||||||
|
d := b.Duration()
|
||||||
|
m.logger.Debugf("WSS: %s, reconnecting in %s", err, d)
|
||||||
|
time.Sleep(d)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
m.logger.Debug("WsClient: connected")
|
||||||
|
m.WsSequence = 1
|
||||||
|
m.WsPingChan = make(chan *model.WebSocketResponse)
|
||||||
|
// only start to parse WS messages when login is completely done
|
||||||
|
m.WsConnected = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) createCookieJar(token string) *cookiejar.Jar {
|
||||||
|
var cookies []*http.Cookie
|
||||||
|
jar, _ := cookiejar.New(nil)
|
||||||
|
firstCookie := &http.Cookie{
|
||||||
|
Name: "MMAUTHTOKEN",
|
||||||
|
Value: token,
|
||||||
|
Path: "/",
|
||||||
|
Domain: m.Credentials.Server,
|
||||||
|
}
|
||||||
|
cookies = append(cookies, firstCookie)
|
||||||
|
cookieURL, _ := url.Parse("https://" + m.Credentials.Server)
|
||||||
|
jar.SetCookies(cookieURL, cookies)
|
||||||
|
return jar
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) checkAlive() error {
|
||||||
|
// check if session still is valid
|
||||||
|
_, resp := m.Client.GetMe("")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
m.logger.Debug("WS PING")
|
||||||
|
return m.sendWSRequest("ping", nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) sendWSRequest(action string, data map[string]interface{}) error {
|
||||||
|
req := &model.WebSocketRequest{}
|
||||||
|
req.Seq = m.WsSequence
|
||||||
|
req.Action = action
|
||||||
|
req.Data = data
|
||||||
|
m.WsSequence++
|
||||||
|
m.logger.Debugf("sendWsRequest %#v", req)
|
||||||
|
return m.WsClient.WriteJSON(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
func supportedVersion(version string) bool {
|
||||||
|
if strings.HasPrefix(version, "3.8.0") ||
|
||||||
|
strings.HasPrefix(version, "3.9.0") ||
|
||||||
|
strings.HasPrefix(version, "3.10.0") ||
|
||||||
|
strings.HasPrefix(version, "4.") ||
|
||||||
|
strings.HasPrefix(version, "5.") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func digestString(s string) string {
|
||||||
|
return fmt.Sprintf("%x", md5.Sum([]byte(s))) //nolint:gosec
|
||||||
|
}
|
@ -1,27 +1,26 @@
|
|||||||
package matterclient
|
package matterclient
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/tls"
|
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"fmt"
|
||||||
"net/http"
|
|
||||||
"net/http/cookiejar"
|
|
||||||
"net/url"
|
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
log "github.com/Sirupsen/logrus"
|
|
||||||
|
|
||||||
"github.com/gorilla/websocket"
|
"github.com/gorilla/websocket"
|
||||||
|
lru "github.com/hashicorp/golang-lru"
|
||||||
"github.com/jpillora/backoff"
|
"github.com/jpillora/backoff"
|
||||||
"github.com/mattermost/platform/model"
|
prefixed "github.com/matterbridge/logrus-prefixed-formatter"
|
||||||
|
"github.com/mattermost/mattermost-server/model"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Credentials struct {
|
type Credentials struct {
|
||||||
Login string
|
Login string
|
||||||
Team string
|
Team string
|
||||||
Pass string
|
Pass string
|
||||||
|
Token string
|
||||||
|
CookieToken bool
|
||||||
Server string
|
Server string
|
||||||
NoTLS bool
|
NoTLS bool
|
||||||
SkipTLSVerify bool
|
SkipTLSVerify bool
|
||||||
@ -34,55 +33,96 @@ type Message struct {
|
|||||||
Channel string
|
Channel string
|
||||||
Username string
|
Username string
|
||||||
Text string
|
Text string
|
||||||
|
Type string
|
||||||
|
UserID string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//nolint:golint
|
||||||
type Team struct {
|
type Team struct {
|
||||||
Team *model.Team
|
Team *model.Team
|
||||||
Id string
|
Id string
|
||||||
Channels *model.ChannelList
|
Channels []*model.Channel
|
||||||
MoreChannels *model.ChannelList
|
MoreChannels []*model.Channel
|
||||||
Users map[string]*model.User
|
Users map[string]*model.User
|
||||||
}
|
}
|
||||||
|
|
||||||
type MMClient struct {
|
type MMClient struct {
|
||||||
sync.RWMutex
|
sync.RWMutex
|
||||||
*Credentials
|
*Credentials
|
||||||
Team *Team
|
|
||||||
OtherTeams []*Team
|
Team *Team
|
||||||
Client *model.Client
|
OtherTeams []*Team
|
||||||
User *model.User
|
Client *model.Client4
|
||||||
Users map[string]*model.User
|
User *model.User
|
||||||
MessageChan chan *Message
|
Users map[string]*model.User
|
||||||
log *log.Entry
|
MessageChan chan *Message
|
||||||
WsClient *websocket.Conn
|
WsClient *websocket.Conn
|
||||||
WsQuit bool
|
WsQuit bool
|
||||||
WsAway bool
|
WsAway bool
|
||||||
WsConnected bool
|
WsConnected bool
|
||||||
WsSequence int64
|
WsSequence int64
|
||||||
WsPingChan chan *model.WebSocketResponse
|
WsPingChan chan *model.WebSocketResponse
|
||||||
|
ServerVersion string
|
||||||
|
OnWsConnect func()
|
||||||
|
|
||||||
|
logger *logrus.Entry
|
||||||
|
rootLogger *logrus.Logger
|
||||||
|
lruCache *lru.Cache
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(login, pass, team, server string) *MMClient {
|
// New will instantiate a new Matterclient with the specified login details without connecting.
|
||||||
cred := &Credentials{Login: login, Pass: pass, Team: team, Server: server}
|
func New(login string, pass string, team string, server string) *MMClient {
|
||||||
mmclient := &MMClient{Credentials: cred, MessageChan: make(chan *Message, 100), Users: make(map[string]*model.User)}
|
rootLogger := logrus.New()
|
||||||
mmclient.log = log.WithFields(log.Fields{"module": "matterclient"})
|
rootLogger.SetFormatter(&prefixed.TextFormatter{
|
||||||
log.SetFormatter(&log.TextFormatter{FullTimestamp: true})
|
PrefixPadding: 13,
|
||||||
return mmclient
|
DisableColors: true,
|
||||||
}
|
})
|
||||||
|
|
||||||
func (m *MMClient) SetLogLevel(level string) {
|
cred := &Credentials{
|
||||||
l, err := log.ParseLevel(level)
|
Login: login,
|
||||||
if err != nil {
|
Pass: pass,
|
||||||
log.SetLevel(log.InfoLevel)
|
Team: team,
|
||||||
return
|
Server: server,
|
||||||
|
}
|
||||||
|
|
||||||
|
cache, _ := lru.New(500)
|
||||||
|
return &MMClient{
|
||||||
|
Credentials: cred,
|
||||||
|
MessageChan: make(chan *Message, 100),
|
||||||
|
Users: make(map[string]*model.User),
|
||||||
|
rootLogger: rootLogger,
|
||||||
|
lruCache: cache,
|
||||||
|
logger: rootLogger.WithFields(logrus.Fields{"prefix": "matterclient"}),
|
||||||
}
|
}
|
||||||
log.SetLevel(l)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SetDebugLog activates debugging logging on all Matterclient log output.
|
||||||
|
func (m *MMClient) SetDebugLog() {
|
||||||
|
m.rootLogger.SetFormatter(&prefixed.TextFormatter{
|
||||||
|
PrefixPadding: 13,
|
||||||
|
DisableColors: true,
|
||||||
|
FullTimestamp: false,
|
||||||
|
ForceFormatting: true,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetLogLevel tries to parse the specified level and if successful sets
|
||||||
|
// the log level accordingly. Accepted levels are: 'debug', 'info', 'warn',
|
||||||
|
// 'error', 'fatal' and 'panic'.
|
||||||
|
func (m *MMClient) SetLogLevel(level string) {
|
||||||
|
l, err := logrus.ParseLevel(level)
|
||||||
|
if err != nil {
|
||||||
|
m.logger.Warnf("Failed to parse specified log-level '%s': %#v", level, err)
|
||||||
|
} else {
|
||||||
|
m.rootLogger.SetLevel(l)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Login tries to connect the client with the loging details with which it was initialized.
|
||||||
func (m *MMClient) Login() error {
|
func (m *MMClient) Login() error {
|
||||||
// check if this is a first connect or a reconnection
|
// check if this is a first connect or a reconnection
|
||||||
firstConnection := true
|
firstConnection := true
|
||||||
if m.WsConnected == true {
|
if m.WsConnected {
|
||||||
firstConnection = false
|
firstConnection = false
|
||||||
}
|
}
|
||||||
m.WsConnected = false
|
m.WsConnected = false
|
||||||
@ -94,115 +134,60 @@ func (m *MMClient) Login() error {
|
|||||||
Max: 5 * time.Minute,
|
Max: 5 * time.Minute,
|
||||||
Jitter: true,
|
Jitter: true,
|
||||||
}
|
}
|
||||||
uriScheme := "https://"
|
|
||||||
wsScheme := "wss://"
|
|
||||||
if m.NoTLS {
|
|
||||||
uriScheme = "http://"
|
|
||||||
wsScheme = "ws://"
|
|
||||||
}
|
|
||||||
// login to mattermost
|
|
||||||
m.Client = model.NewClient(uriScheme + m.Credentials.Server)
|
|
||||||
m.Client.HttpClient.Transport = &http.Transport{TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}}
|
|
||||||
var myinfo *model.Result
|
|
||||||
var appErr *model.AppError
|
|
||||||
var logmsg = "trying login"
|
|
||||||
for {
|
|
||||||
m.log.Debugf("%s %s %s %s", logmsg, m.Credentials.Team, m.Credentials.Login, m.Credentials.Server)
|
|
||||||
if strings.Contains(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN) {
|
|
||||||
m.log.Debugf(logmsg+" with %s", model.SESSION_COOKIE_TOKEN)
|
|
||||||
token := strings.Split(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN+"=")
|
|
||||||
if len(token) != 2 {
|
|
||||||
return errors.New("incorrect MMAUTHTOKEN. valid input is MMAUTHTOKEN=yourtoken")
|
|
||||||
}
|
|
||||||
m.Client.HttpClient.Jar = m.createCookieJar(token[1])
|
|
||||||
m.Client.MockSession(token[1])
|
|
||||||
myinfo, appErr = m.Client.GetMe("")
|
|
||||||
if appErr != nil {
|
|
||||||
return errors.New(appErr.DetailedError)
|
|
||||||
}
|
|
||||||
if myinfo.Data.(*model.User) == nil {
|
|
||||||
m.log.Errorf("LOGIN TOKEN: %s is invalid", m.Credentials.Pass)
|
|
||||||
return errors.New("invalid " + model.SESSION_COOKIE_TOKEN)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
myinfo, appErr = m.Client.Login(m.Credentials.Login, m.Credentials.Pass)
|
|
||||||
}
|
|
||||||
if appErr != nil {
|
|
||||||
d := b.Duration()
|
|
||||||
m.log.Debug(appErr.DetailedError)
|
|
||||||
if firstConnection {
|
|
||||||
if appErr.Message == "" {
|
|
||||||
return errors.New(appErr.DetailedError)
|
|
||||||
}
|
|
||||||
return errors.New(appErr.Message)
|
|
||||||
}
|
|
||||||
m.log.Debugf("LOGIN: %s, reconnecting in %s", appErr, d)
|
|
||||||
time.Sleep(d)
|
|
||||||
logmsg = "retrying login"
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
// reset timer
|
|
||||||
b.Reset()
|
|
||||||
|
|
||||||
err := m.initUser()
|
// do initialization setup
|
||||||
if err != nil {
|
if err := m.initClient(firstConnection, b); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := m.doLogin(firstConnection, b); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := m.initUser(); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if m.Team == nil {
|
if m.Team == nil {
|
||||||
return errors.New("team not found")
|
validTeamNames := make([]string, len(m.OtherTeams))
|
||||||
}
|
for i, t := range m.OtherTeams {
|
||||||
// set our team id as default route
|
validTeamNames[i] = t.Team.Name
|
||||||
m.Client.SetTeamId(m.Team.Id)
|
|
||||||
|
|
||||||
// setup websocket connection
|
|
||||||
wsurl := wsScheme + m.Credentials.Server + model.API_URL_SUFFIX + "/users/websocket"
|
|
||||||
header := http.Header{}
|
|
||||||
header.Set(model.HEADER_AUTH, "BEARER "+m.Client.AuthToken)
|
|
||||||
|
|
||||||
m.log.Debug("WsClient: making connection")
|
|
||||||
for {
|
|
||||||
wsDialer := &websocket.Dialer{Proxy: http.ProxyFromEnvironment, TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}}
|
|
||||||
m.WsClient, _, err = wsDialer.Dial(wsurl, header)
|
|
||||||
if err != nil {
|
|
||||||
d := b.Duration()
|
|
||||||
m.log.Debugf("WSS: %s, reconnecting in %s", err, d)
|
|
||||||
time.Sleep(d)
|
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
break
|
return fmt.Errorf("Team '%s' not found in %v", m.Credentials.Team, validTeamNames)
|
||||||
}
|
}
|
||||||
b.Reset()
|
|
||||||
|
|
||||||
m.WsSequence = 1
|
m.wsConnect()
|
||||||
m.WsPingChan = make(chan *model.WebSocketResponse)
|
|
||||||
// only start to parse WS messages when login is completely done
|
|
||||||
m.WsConnected = true
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Logout disconnects the client from the chat server.
|
||||||
func (m *MMClient) Logout() error {
|
func (m *MMClient) Logout() error {
|
||||||
m.log.Debugf("logout as %s (team: %s) on %s", m.Credentials.Login, m.Credentials.Team, m.Credentials.Server)
|
m.logger.Debugf("logout as %s (team: %s) on %s", m.Credentials.Login, m.Credentials.Team, m.Credentials.Server)
|
||||||
m.WsQuit = true
|
m.WsQuit = true
|
||||||
m.WsClient.Close()
|
m.WsClient.Close()
|
||||||
m.WsClient.UnderlyingConn().Close()
|
m.WsClient.UnderlyingConn().Close()
|
||||||
_, err := m.Client.Logout()
|
if strings.Contains(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN) {
|
||||||
if err != nil {
|
m.logger.Debug("Not invalidating session in logout, credential is a token")
|
||||||
return err
|
return nil
|
||||||
|
}
|
||||||
|
_, resp := m.Client.Logout()
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// WsReceiver implements the core loop that manages the connection to the chat server. In
|
||||||
|
// case of a disconnect it will try to reconnect. A call to this method is blocking until
|
||||||
|
// the 'WsQuite' field of the MMClient object is set to 'true'.
|
||||||
func (m *MMClient) WsReceiver() {
|
func (m *MMClient) WsReceiver() {
|
||||||
for {
|
for {
|
||||||
var rawMsg json.RawMessage
|
var rawMsg json.RawMessage
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
if m.WsQuit {
|
if m.WsQuit {
|
||||||
m.log.Debug("exiting WsReceiver")
|
m.logger.Debug("exiting WsReceiver")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -212,472 +197,80 @@ func (m *MMClient) WsReceiver() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if _, rawMsg, err = m.WsClient.ReadMessage(); err != nil {
|
if _, rawMsg, err = m.WsClient.ReadMessage(); err != nil {
|
||||||
m.log.Error("error:", err)
|
m.logger.Error("error:", err)
|
||||||
// reconnect
|
// reconnect
|
||||||
m.Login()
|
m.wsConnect()
|
||||||
}
|
}
|
||||||
|
|
||||||
var event model.WebSocketEvent
|
var event model.WebSocketEvent
|
||||||
if err := json.Unmarshal(rawMsg, &event); err == nil && event.IsValid() {
|
if err := json.Unmarshal(rawMsg, &event); err == nil && event.IsValid() {
|
||||||
m.log.Debugf("WsReceiver: %#v", event)
|
m.logger.Debugf("WsReceiver event: %#v", event)
|
||||||
msg := &Message{Raw: &event, Team: m.Credentials.Team}
|
msg := &Message{Raw: &event, Team: m.Credentials.Team}
|
||||||
m.parseMessage(msg)
|
m.parseMessage(msg)
|
||||||
m.MessageChan <- msg
|
// check if we didn't empty the message
|
||||||
|
if msg.Text != "" {
|
||||||
|
m.MessageChan <- msg
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// if we have file attached but the message is empty, also send it
|
||||||
|
if msg.Post != nil {
|
||||||
|
if msg.Text != "" || len(msg.Post.FileIds) > 0 || msg.Post.Type == "slack_attachment" {
|
||||||
|
m.MessageChan <- msg
|
||||||
|
}
|
||||||
|
}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
var response model.WebSocketResponse
|
var response model.WebSocketResponse
|
||||||
if err := json.Unmarshal(rawMsg, &response); err == nil && response.IsValid() {
|
if err := json.Unmarshal(rawMsg, &response); err == nil && response.IsValid() {
|
||||||
m.log.Debugf("WsReceiver: %#v", response)
|
m.logger.Debugf("WsReceiver response: %#v", response)
|
||||||
m.parseResponse(response)
|
m.parseResponse(response)
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *MMClient) parseMessage(rmsg *Message) {
|
// StatusLoop implements a ping-cycle that ensures that the connection to the chat servers
|
||||||
switch rmsg.Raw.Event {
|
// remains alive. In case of a disconnect it will try to reconnect. A call to this method
|
||||||
case model.WEBSOCKET_EVENT_POSTED:
|
// is blocking until the 'WsQuite' field of the MMClient object is set to 'true'.
|
||||||
m.parseActionPost(rmsg)
|
|
||||||
/*
|
|
||||||
case model.ACTION_USER_REMOVED:
|
|
||||||
m.handleWsActionUserRemoved(&rmsg)
|
|
||||||
case model.ACTION_USER_ADDED:
|
|
||||||
m.handleWsActionUserAdded(&rmsg)
|
|
||||||
*/
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) parseResponse(rmsg model.WebSocketResponse) {
|
|
||||||
if rmsg.Data != nil {
|
|
||||||
// ping reply
|
|
||||||
if rmsg.Data["text"].(string) == "pong" {
|
|
||||||
m.WsPingChan <- &rmsg
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) parseActionPost(rmsg *Message) {
|
|
||||||
data := model.PostFromJson(strings.NewReader(rmsg.Raw.Data["post"].(string)))
|
|
||||||
// we don't have the user, refresh the userlist
|
|
||||||
if m.GetUser(data.UserId) == nil {
|
|
||||||
m.UpdateUsers()
|
|
||||||
}
|
|
||||||
rmsg.Username = m.GetUser(data.UserId).Username
|
|
||||||
rmsg.Channel = m.GetChannelName(data.ChannelId)
|
|
||||||
rmsg.Team = m.GetTeamName(rmsg.Raw.Data["team_id"].(string))
|
|
||||||
// direct message
|
|
||||||
if rmsg.Raw.Data["channel_type"] == "D" {
|
|
||||||
rmsg.Channel = m.GetUser(data.UserId).Username
|
|
||||||
}
|
|
||||||
rmsg.Text = data.Message
|
|
||||||
rmsg.Post = data
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UpdateUsers() error {
|
|
||||||
mmusers, err := m.Client.GetProfiles(0, 50000, "")
|
|
||||||
if err != nil {
|
|
||||||
return errors.New(err.DetailedError)
|
|
||||||
}
|
|
||||||
m.Lock()
|
|
||||||
m.Users = mmusers.Data.(map[string]*model.User)
|
|
||||||
m.Unlock()
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UpdateChannels() error {
|
|
||||||
mmchannels, err := m.Client.GetChannels("")
|
|
||||||
if err != nil {
|
|
||||||
return errors.New(err.DetailedError)
|
|
||||||
}
|
|
||||||
mmchannels2, err := m.Client.GetMoreChannels("")
|
|
||||||
if err != nil {
|
|
||||||
return errors.New(err.DetailedError)
|
|
||||||
}
|
|
||||||
m.Lock()
|
|
||||||
m.Team.Channels = mmchannels.Data.(*model.ChannelList)
|
|
||||||
m.Team.MoreChannels = mmchannels2.Data.(*model.ChannelList)
|
|
||||||
m.Unlock()
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetChannelName(channelId string) string {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
for _, t := range m.OtherTeams {
|
|
||||||
for _, channel := range append(*t.Channels, *t.MoreChannels...) {
|
|
||||||
if channel.Id == channelId {
|
|
||||||
return channel.Name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetChannelId(name string, teamId string) string {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
if teamId == "" {
|
|
||||||
teamId = m.Team.Id
|
|
||||||
}
|
|
||||||
for _, t := range m.OtherTeams {
|
|
||||||
if t.Id == teamId {
|
|
||||||
for _, channel := range append(*t.Channels, *t.MoreChannels...) {
|
|
||||||
if channel.Name == name {
|
|
||||||
return channel.Id
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetChannelHeader(channelId string) string {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
for _, t := range m.OtherTeams {
|
|
||||||
for _, channel := range append(*t.Channels, *t.MoreChannels...) {
|
|
||||||
if channel.Id == channelId {
|
|
||||||
return channel.Header
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) PostMessage(channelId string, text string) {
|
|
||||||
post := &model.Post{ChannelId: channelId, Message: text}
|
|
||||||
m.Client.CreatePost(post)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) JoinChannel(channelId string) error {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
for _, c := range *m.Team.Channels {
|
|
||||||
if c.Id == channelId {
|
|
||||||
m.log.Debug("Not joining ", channelId, " already joined.")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
m.log.Debug("Joining ", channelId)
|
|
||||||
_, err := m.Client.JoinChannel(channelId)
|
|
||||||
if err != nil {
|
|
||||||
return errors.New("failed to join")
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetPostsSince(channelId string, time int64) *model.PostList {
|
|
||||||
res, err := m.Client.GetPostsSince(channelId, time)
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return res.Data.(*model.PostList)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) SearchPosts(query string) *model.PostList {
|
|
||||||
res, err := m.Client.SearchPosts(query, false)
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return res.Data.(*model.PostList)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetPosts(channelId string, limit int) *model.PostList {
|
|
||||||
res, err := m.Client.GetPosts(channelId, 0, limit, "")
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return res.Data.(*model.PostList)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetPublicLink(filename string) string {
|
|
||||||
res, err := m.Client.GetPublicLink(filename)
|
|
||||||
if err != nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetPublicLinks(filenames []string) []string {
|
|
||||||
var output []string
|
|
||||||
for _, f := range filenames {
|
|
||||||
res, err := m.Client.GetPublicLink(f)
|
|
||||||
if err != nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
output = append(output, res)
|
|
||||||
}
|
|
||||||
return output
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UpdateChannelHeader(channelId string, header string) {
|
|
||||||
data := make(map[string]string)
|
|
||||||
data["channel_id"] = channelId
|
|
||||||
data["channel_header"] = header
|
|
||||||
m.log.Debugf("updating channelheader %#v, %#v", channelId, header)
|
|
||||||
_, err := m.Client.UpdateChannelHeader(data)
|
|
||||||
if err != nil {
|
|
||||||
log.Error(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UpdateLastViewed(channelId string) {
|
|
||||||
m.log.Debugf("posting lastview %#v", channelId)
|
|
||||||
_, err := m.Client.UpdateLastViewedAt(channelId, true)
|
|
||||||
if err != nil {
|
|
||||||
m.log.Error(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UsernamesInChannel(channelId string) []string {
|
|
||||||
res, err := m.Client.GetMyChannelMembers()
|
|
||||||
if err != nil {
|
|
||||||
m.log.Errorf("UsernamesInChannel(%s) failed: %s", channelId, err)
|
|
||||||
return []string{}
|
|
||||||
}
|
|
||||||
members := res.Data.(*model.ChannelMembers)
|
|
||||||
result := []string{}
|
|
||||||
for _, channel := range *members {
|
|
||||||
if channel.ChannelId == channelId {
|
|
||||||
result = append(result, m.GetUser(channel.UserId).Username)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) createCookieJar(token string) *cookiejar.Jar {
|
|
||||||
var cookies []*http.Cookie
|
|
||||||
jar, _ := cookiejar.New(nil)
|
|
||||||
firstCookie := &http.Cookie{
|
|
||||||
Name: "MMAUTHTOKEN",
|
|
||||||
Value: token,
|
|
||||||
Path: "/",
|
|
||||||
Domain: m.Credentials.Server,
|
|
||||||
}
|
|
||||||
cookies = append(cookies, firstCookie)
|
|
||||||
cookieURL, _ := url.Parse("https://" + m.Credentials.Server)
|
|
||||||
jar.SetCookies(cookieURL, cookies)
|
|
||||||
return jar
|
|
||||||
}
|
|
||||||
|
|
||||||
// SendDirectMessage sends a direct message to specified user
|
|
||||||
func (m *MMClient) SendDirectMessage(toUserId string, msg string) {
|
|
||||||
m.log.Debugf("SendDirectMessage to %s, msg %s", toUserId, msg)
|
|
||||||
// create DM channel (only happens on first message)
|
|
||||||
_, err := m.Client.CreateDirectChannel(toUserId)
|
|
||||||
if err != nil {
|
|
||||||
m.log.Debugf("SendDirectMessage to %#v failed: %s", toUserId, err)
|
|
||||||
}
|
|
||||||
channelName := model.GetDMNameFromIds(toUserId, m.User.Id)
|
|
||||||
|
|
||||||
// update our channels
|
|
||||||
mmchannels, _ := m.Client.GetChannels("")
|
|
||||||
m.Lock()
|
|
||||||
m.Team.Channels = mmchannels.Data.(*model.ChannelList)
|
|
||||||
m.Unlock()
|
|
||||||
|
|
||||||
// build & send the message
|
|
||||||
msg = strings.Replace(msg, "\r", "", -1)
|
|
||||||
post := &model.Post{ChannelId: m.GetChannelId(channelName, ""), Message: msg}
|
|
||||||
m.Client.CreatePost(post)
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetTeamName returns the name of the specified teamId
|
|
||||||
func (m *MMClient) GetTeamName(teamId string) string {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
for _, t := range m.OtherTeams {
|
|
||||||
if t.Id == teamId {
|
|
||||||
return t.Team.Name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetChannels returns all channels we're members off
|
|
||||||
func (m *MMClient) GetChannels() []*model.Channel {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
var channels []*model.Channel
|
|
||||||
// our primary team channels first
|
|
||||||
channels = append(channels, *m.Team.Channels...)
|
|
||||||
for _, t := range m.OtherTeams {
|
|
||||||
if t.Id != m.Team.Id {
|
|
||||||
channels = append(channels, *t.Channels...)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return channels
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetMoreChannels returns existing channels where we're not a member off.
|
|
||||||
func (m *MMClient) GetMoreChannels() []*model.Channel {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
var channels []*model.Channel
|
|
||||||
for _, t := range m.OtherTeams {
|
|
||||||
channels = append(channels, *t.MoreChannels...)
|
|
||||||
}
|
|
||||||
return channels
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetTeamFromChannel returns teamId belonging to channel (DM channels have no teamId).
|
|
||||||
func (m *MMClient) GetTeamFromChannel(channelId string) string {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
var channels []*model.Channel
|
|
||||||
for _, t := range m.OtherTeams {
|
|
||||||
channels = append(channels, *t.Channels...)
|
|
||||||
channels = append(channels, *t.MoreChannels...)
|
|
||||||
for _, c := range channels {
|
|
||||||
if c.Id == channelId {
|
|
||||||
return t.Id
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetLastViewedAt(channelId string) int64 {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
res, err := m.Client.GetChannel(channelId, "")
|
|
||||||
if err != nil {
|
|
||||||
return model.GetMillis()
|
|
||||||
}
|
|
||||||
data := res.Data.(*model.ChannelData)
|
|
||||||
return data.Member.LastViewedAt
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetUsers() map[string]*model.User {
|
|
||||||
users := make(map[string]*model.User)
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
for k, v := range m.Users {
|
|
||||||
users[k] = v
|
|
||||||
}
|
|
||||||
return users
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetUser(userId string) *model.User {
|
|
||||||
m.RLock()
|
|
||||||
defer m.RUnlock()
|
|
||||||
return m.Users[userId]
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetStatus(userId string) string {
|
|
||||||
res, err := m.Client.GetStatuses()
|
|
||||||
if err != nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
status := res.Data.(map[string]string)
|
|
||||||
if status[userId] == model.STATUS_AWAY {
|
|
||||||
return "away"
|
|
||||||
}
|
|
||||||
if status[userId] == model.STATUS_ONLINE {
|
|
||||||
return "online"
|
|
||||||
}
|
|
||||||
return "offline"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetStatuses() map[string]string {
|
|
||||||
var ok bool
|
|
||||||
statuses := make(map[string]string)
|
|
||||||
res, err := m.Client.GetStatuses()
|
|
||||||
if err != nil {
|
|
||||||
return statuses
|
|
||||||
}
|
|
||||||
if statuses, ok = res.Data.(map[string]string); ok {
|
|
||||||
for userId, status := range statuses {
|
|
||||||
statuses[userId] = "offline"
|
|
||||||
if status == model.STATUS_AWAY {
|
|
||||||
statuses[userId] = "away"
|
|
||||||
}
|
|
||||||
if status == model.STATUS_ONLINE {
|
|
||||||
statuses[userId] = "online"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return statuses
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetTeamId() string {
|
|
||||||
return m.Team.Id
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) StatusLoop() {
|
func (m *MMClient) StatusLoop() {
|
||||||
|
retries := 0
|
||||||
|
backoff := time.Second * 60
|
||||||
|
if m.OnWsConnect != nil {
|
||||||
|
m.OnWsConnect()
|
||||||
|
}
|
||||||
|
m.logger.Debug("StatusLoop:", m.OnWsConnect != nil)
|
||||||
for {
|
for {
|
||||||
if m.WsQuit {
|
if m.WsQuit {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if m.WsConnected {
|
if m.WsConnected {
|
||||||
m.log.Debug("WS PING")
|
if err := m.checkAlive(); err != nil {
|
||||||
m.sendWSRequest("ping", nil)
|
m.logger.Errorf("Connection is not alive: %#v", err)
|
||||||
|
}
|
||||||
select {
|
select {
|
||||||
case <-m.WsPingChan:
|
case <-m.WsPingChan:
|
||||||
m.log.Debug("WS PONG received")
|
m.logger.Debug("WS PONG received")
|
||||||
|
backoff = time.Second * 60
|
||||||
case <-time.After(time.Second * 5):
|
case <-time.After(time.Second * 5):
|
||||||
m.Logout()
|
if retries > 3 {
|
||||||
m.WsQuit = false
|
m.logger.Debug("StatusLoop() timeout")
|
||||||
m.Login()
|
m.Logout()
|
||||||
|
m.WsQuit = false
|
||||||
|
err := m.Login()
|
||||||
|
if err != nil {
|
||||||
|
m.logger.Errorf("Login failed: %#v", err)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if m.OnWsConnect != nil {
|
||||||
|
m.OnWsConnect()
|
||||||
|
}
|
||||||
|
go m.WsReceiver()
|
||||||
|
} else {
|
||||||
|
retries++
|
||||||
|
backoff = time.Second * 5
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
time.Sleep(time.Second * 60)
|
time.Sleep(backoff)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// initialize user and teams
|
|
||||||
func (m *MMClient) initUser() error {
|
|
||||||
m.Lock()
|
|
||||||
defer m.Unlock()
|
|
||||||
initLoad, err := m.Client.GetInitialLoad()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
initData := initLoad.Data.(*model.InitialLoad)
|
|
||||||
m.User = initData.User
|
|
||||||
// we only load all team data on initial login.
|
|
||||||
// all other updates are for channels from our (primary) team only.
|
|
||||||
//m.log.Debug("initUser(): loading all team data")
|
|
||||||
for _, v := range initData.Teams {
|
|
||||||
m.Client.SetTeamId(v.Id)
|
|
||||||
mmusers, err := m.Client.GetProfiles(0, 50000, "")
|
|
||||||
if err != nil {
|
|
||||||
return errors.New(err.DetailedError)
|
|
||||||
}
|
|
||||||
t := &Team{Team: v, Users: mmusers.Data.(map[string]*model.User), Id: v.Id}
|
|
||||||
mmchannels, err := m.Client.GetChannels("")
|
|
||||||
if err != nil {
|
|
||||||
return errors.New(err.DetailedError)
|
|
||||||
}
|
|
||||||
t.Channels = mmchannels.Data.(*model.ChannelList)
|
|
||||||
mmchannels, err = m.Client.GetMoreChannels("")
|
|
||||||
if err != nil {
|
|
||||||
return errors.New(err.DetailedError)
|
|
||||||
}
|
|
||||||
t.MoreChannels = mmchannels.Data.(*model.ChannelList)
|
|
||||||
m.OtherTeams = append(m.OtherTeams, t)
|
|
||||||
if v.Name == m.Credentials.Team {
|
|
||||||
m.Team = t
|
|
||||||
m.log.Debugf("initUser(): found our team %s (id: %s)", v.Name, v.Id)
|
|
||||||
}
|
|
||||||
// add all users
|
|
||||||
for k, v := range t.Users {
|
|
||||||
m.Users[k] = v
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) sendWSRequest(action string, data map[string]interface{}) error {
|
|
||||||
req := &model.WebSocketRequest{}
|
|
||||||
req.Seq = m.WsSequence
|
|
||||||
req.Action = action
|
|
||||||
req.Data = data
|
|
||||||
m.WsSequence++
|
|
||||||
m.log.Debugf("sendWsRequest %#v", req)
|
|
||||||
m.WsClient.WriteJSON(req)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
207
matterclient/messages.go
Normal file
@ -0,0 +1,207 @@
|
|||||||
|
package matterclient
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/mattermost/mattermost-server/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (m *MMClient) parseActionPost(rmsg *Message) {
|
||||||
|
// add post to cache, if it already exists don't relay this again.
|
||||||
|
// this should fix reposts
|
||||||
|
if ok, _ := m.lruCache.ContainsOrAdd(digestString(rmsg.Raw.Data["post"].(string)), true); ok {
|
||||||
|
m.logger.Debugf("message %#v in cache, not processing again", rmsg.Raw.Data["post"].(string))
|
||||||
|
rmsg.Text = ""
|
||||||
|
return
|
||||||
|
}
|
||||||
|
data := model.PostFromJson(strings.NewReader(rmsg.Raw.Data["post"].(string)))
|
||||||
|
// we don't have the user, refresh the userlist
|
||||||
|
if m.GetUser(data.UserId) == nil {
|
||||||
|
m.logger.Infof("User '%v' is not known, ignoring message '%#v'",
|
||||||
|
data.UserId, data)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
rmsg.Username = m.GetUserName(data.UserId)
|
||||||
|
rmsg.Channel = m.GetChannelName(data.ChannelId)
|
||||||
|
rmsg.UserID = data.UserId
|
||||||
|
rmsg.Type = data.Type
|
||||||
|
teamid, _ := rmsg.Raw.Data["team_id"].(string)
|
||||||
|
// edit messsages have no team_id for some reason
|
||||||
|
if teamid == "" {
|
||||||
|
// we can find the team_id from the channelid
|
||||||
|
teamid = m.GetChannelTeamId(data.ChannelId)
|
||||||
|
rmsg.Raw.Data["team_id"] = teamid
|
||||||
|
}
|
||||||
|
if teamid != "" {
|
||||||
|
rmsg.Team = m.GetTeamName(teamid)
|
||||||
|
}
|
||||||
|
// direct message
|
||||||
|
if rmsg.Raw.Data["channel_type"] == "D" {
|
||||||
|
rmsg.Channel = m.GetUser(data.UserId).Username
|
||||||
|
}
|
||||||
|
rmsg.Text = data.Message
|
||||||
|
rmsg.Post = data
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) parseMessage(rmsg *Message) {
|
||||||
|
switch rmsg.Raw.Event {
|
||||||
|
case model.WEBSOCKET_EVENT_POSTED, model.WEBSOCKET_EVENT_POST_EDITED, model.WEBSOCKET_EVENT_POST_DELETED:
|
||||||
|
m.parseActionPost(rmsg)
|
||||||
|
case "user_updated":
|
||||||
|
user := rmsg.Raw.Data["user"].(map[string]interface{})
|
||||||
|
if _, ok := user["id"].(string); ok {
|
||||||
|
m.UpdateUser(user["id"].(string))
|
||||||
|
}
|
||||||
|
case "group_added":
|
||||||
|
if err := m.UpdateChannels(); err != nil {
|
||||||
|
m.logger.Errorf("failed to update channels: %#v", err)
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
case model.ACTION_USER_REMOVED:
|
||||||
|
m.handleWsActionUserRemoved(&rmsg)
|
||||||
|
case model.ACTION_USER_ADDED:
|
||||||
|
m.handleWsActionUserAdded(&rmsg)
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) parseResponse(rmsg model.WebSocketResponse) {
|
||||||
|
if rmsg.Data != nil {
|
||||||
|
// ping reply
|
||||||
|
if rmsg.Data["text"].(string) == "pong" {
|
||||||
|
m.WsPingChan <- &rmsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) DeleteMessage(postId string) error { //nolint:golint
|
||||||
|
_, resp := m.Client.DeletePost(postId)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) EditMessage(postId string, text string) (string, error) { //nolint:golint
|
||||||
|
post := &model.Post{Message: text}
|
||||||
|
res, resp := m.Client.UpdatePost(postId, post)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return "", resp.Error
|
||||||
|
}
|
||||||
|
return res.Id, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetFileLinks(filenames []string) []string {
|
||||||
|
uriScheme := "https://"
|
||||||
|
if m.NoTLS {
|
||||||
|
uriScheme = "http://"
|
||||||
|
}
|
||||||
|
|
||||||
|
var output []string
|
||||||
|
for _, f := range filenames {
|
||||||
|
res, resp := m.Client.GetFileLink(f)
|
||||||
|
if resp.Error != nil {
|
||||||
|
// public links is probably disabled, create the link ourselves
|
||||||
|
output = append(output, uriScheme+m.Credentials.Server+model.API_URL_SUFFIX_V4+"/files/"+f)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
output = append(output, res)
|
||||||
|
}
|
||||||
|
return output
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetPosts(channelId string, limit int) *model.PostList { //nolint:golint
|
||||||
|
res, resp := m.Client.GetPostsForChannel(channelId, 0, limit, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetPostsSince(channelId string, time int64) *model.PostList { //nolint:golint
|
||||||
|
res, resp := m.Client.GetPostsSince(channelId, time)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetPublicLink(filename string) string {
|
||||||
|
res, resp := m.Client.GetFileLink(filename)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetPublicLinks(filenames []string) []string {
|
||||||
|
var output []string
|
||||||
|
for _, f := range filenames {
|
||||||
|
res, resp := m.Client.GetFileLink(f)
|
||||||
|
if resp.Error != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
output = append(output, res)
|
||||||
|
}
|
||||||
|
return output
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) PostMessage(channelId string, text string, rootId string) (string, error) { //nolint:golint
|
||||||
|
post := &model.Post{ChannelId: channelId, Message: text, RootId: rootId}
|
||||||
|
res, resp := m.Client.CreatePost(post)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return "", resp.Error
|
||||||
|
}
|
||||||
|
return res.Id, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) PostMessageWithFiles(channelId string, text string, rootId string, fileIds []string) (string, error) { //nolint:golint
|
||||||
|
post := &model.Post{ChannelId: channelId, Message: text, RootId: rootId, FileIds: fileIds}
|
||||||
|
res, resp := m.Client.CreatePost(post)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return "", resp.Error
|
||||||
|
}
|
||||||
|
return res.Id, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) SearchPosts(query string) *model.PostList {
|
||||||
|
res, resp := m.Client.SearchPosts(m.Team.Id, query, false)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendDirectMessage sends a direct message to specified user
|
||||||
|
func (m *MMClient) SendDirectMessage(toUserId string, msg string, rootId string) { //nolint:golint
|
||||||
|
m.SendDirectMessageProps(toUserId, msg, rootId, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) SendDirectMessageProps(toUserId string, msg string, rootId string, props map[string]interface{}) { //nolint:golint
|
||||||
|
m.logger.Debugf("SendDirectMessage to %s, msg %s", toUserId, msg)
|
||||||
|
// create DM channel (only happens on first message)
|
||||||
|
_, resp := m.Client.CreateDirectChannel(m.User.Id, toUserId)
|
||||||
|
if resp.Error != nil {
|
||||||
|
m.logger.Debugf("SendDirectMessage to %#v failed: %s", toUserId, resp.Error)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
channelName := model.GetDMNameFromIds(toUserId, m.User.Id)
|
||||||
|
|
||||||
|
// update our channels
|
||||||
|
if err := m.UpdateChannels(); err != nil {
|
||||||
|
m.logger.Errorf("failed to update channels: %#v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// build & send the message
|
||||||
|
msg = strings.Replace(msg, "\r", "", -1)
|
||||||
|
post := &model.Post{ChannelId: m.GetChannelId(channelName, m.Team.Id), Message: msg, RootId: rootId, Props: props}
|
||||||
|
m.Client.CreatePost(post)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UploadFile(data []byte, channelId string, filename string) (string, error) { //nolint:golint
|
||||||
|
f, resp := m.Client.UploadFile(data, channelId, filename)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return "", resp.Error
|
||||||
|
}
|
||||||
|
return f.FileInfos[0].Id, nil
|
||||||
|
}
|
154
matterclient/users.go
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
package matterclient
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"github.com/mattermost/mattermost-server/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (m *MMClient) GetNickName(userId string) string { //nolint:golint
|
||||||
|
user := m.GetUser(userId)
|
||||||
|
if user != nil {
|
||||||
|
return user.Nickname
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetStatus(userId string) string { //nolint:golint
|
||||||
|
res, resp := m.Client.GetUserStatus(userId, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
if res.Status == model.STATUS_AWAY {
|
||||||
|
return "away"
|
||||||
|
}
|
||||||
|
if res.Status == model.STATUS_ONLINE {
|
||||||
|
return "online"
|
||||||
|
}
|
||||||
|
return "offline"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetStatuses() map[string]string {
|
||||||
|
var ids []string
|
||||||
|
statuses := make(map[string]string)
|
||||||
|
for id := range m.Users {
|
||||||
|
ids = append(ids, id)
|
||||||
|
}
|
||||||
|
res, resp := m.Client.GetUsersStatusesByIds(ids)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return statuses
|
||||||
|
}
|
||||||
|
for _, status := range res {
|
||||||
|
statuses[status.UserId] = "offline"
|
||||||
|
if status.Status == model.STATUS_AWAY {
|
||||||
|
statuses[status.UserId] = "away"
|
||||||
|
}
|
||||||
|
if status.Status == model.STATUS_ONLINE {
|
||||||
|
statuses[status.UserId] = "online"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return statuses
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetTeamId() string { //nolint:golint
|
||||||
|
return m.Team.Id
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetTeamName returns the name of the specified teamId
|
||||||
|
func (m *MMClient) GetTeamName(teamId string) string { //nolint:golint
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
for _, t := range m.OtherTeams {
|
||||||
|
if t.Id == teamId {
|
||||||
|
return t.Team.Name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetUser(userId string) *model.User { //nolint:golint
|
||||||
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
|
_, ok := m.Users[userId]
|
||||||
|
if !ok {
|
||||||
|
res, resp := m.Client.GetUser(userId, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
m.Users[userId] = res
|
||||||
|
}
|
||||||
|
return m.Users[userId]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetUserName(userId string) string { //nolint:golint
|
||||||
|
user := m.GetUser(userId)
|
||||||
|
if user != nil {
|
||||||
|
return user.Username
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) GetUsers() map[string]*model.User {
|
||||||
|
users := make(map[string]*model.User)
|
||||||
|
m.RLock()
|
||||||
|
defer m.RUnlock()
|
||||||
|
for k, v := range m.Users {
|
||||||
|
users[k] = v
|
||||||
|
}
|
||||||
|
return users
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UpdateUsers() error {
|
||||||
|
mmusers, resp := m.Client.GetUsers(0, 50000, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return errors.New(resp.Error.DetailedError)
|
||||||
|
}
|
||||||
|
m.Lock()
|
||||||
|
for _, user := range mmusers {
|
||||||
|
m.Users[user.Id] = user
|
||||||
|
}
|
||||||
|
m.Unlock()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UpdateUserNick(nick string) error {
|
||||||
|
user := m.User
|
||||||
|
user.Nickname = nick
|
||||||
|
_, resp := m.Client.UpdateUser(user)
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UsernamesInChannel(channelId string) []string { //nolint:golint
|
||||||
|
res, resp := m.Client.GetChannelMembers(channelId, 0, 50000, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
m.logger.Errorf("UsernamesInChannel(%s) failed: %s", channelId, resp.Error)
|
||||||
|
return []string{}
|
||||||
|
}
|
||||||
|
allusers := m.GetUsers()
|
||||||
|
result := []string{}
|
||||||
|
for _, member := range *res {
|
||||||
|
result = append(result, allusers[member.UserId].Nickname)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UpdateStatus(userId string, status string) error { //nolint:golint
|
||||||
|
_, resp := m.Client.UpdateUserStatus(userId, &model.Status{Status: status})
|
||||||
|
if resp.Error != nil {
|
||||||
|
return resp.Error
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MMClient) UpdateUser(userId string) { //nolint:golint
|
||||||
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
|
res, resp := m.Client.GetUser(userId, "")
|
||||||
|
if resp.Error != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
m.Users[userId] = res
|
||||||
|
}
|
@ -6,23 +6,27 @@ import (
|
|||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/gorilla/schema"
|
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gorilla/schema"
|
||||||
|
"github.com/nlopes/slack"
|
||||||
)
|
)
|
||||||
|
|
||||||
// OMessage for mattermost incoming webhook. (send to mattermost)
|
// OMessage for mattermost incoming webhook. (send to mattermost)
|
||||||
type OMessage struct {
|
type OMessage struct {
|
||||||
Channel string `json:"channel,omitempty"`
|
Channel string `json:"channel,omitempty"`
|
||||||
IconURL string `json:"icon_url,omitempty"`
|
IconURL string `json:"icon_url,omitempty"`
|
||||||
IconEmoji string `json:"icon_emoji,omitempty"`
|
IconEmoji string `json:"icon_emoji,omitempty"`
|
||||||
UserName string `json:"username,omitempty"`
|
UserName string `json:"username,omitempty"`
|
||||||
Text string `json:"text"`
|
Text string `json:"text"`
|
||||||
Attachments interface{} `json:"attachments,omitempty"`
|
Attachments []slack.Attachment `json:"attachments,omitempty"`
|
||||||
Type string `json:"type,omitempty"`
|
Type string `json:"type,omitempty"`
|
||||||
|
Props map[string]interface{} `json:"props"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// IMessage for mattermost outgoing webhook. (received from mattermost)
|
// IMessage for mattermost outgoing webhook. (received from mattermost)
|
||||||
@ -37,16 +41,18 @@ type IMessage struct {
|
|||||||
Timestamp string `schema:"timestamp"`
|
Timestamp string `schema:"timestamp"`
|
||||||
UserID string `schema:"user_id"`
|
UserID string `schema:"user_id"`
|
||||||
UserName string `schema:"user_name"`
|
UserName string `schema:"user_name"`
|
||||||
PostId string `schema:"post_id"`
|
PostId string `schema:"post_id"` //nolint:golint
|
||||||
RawText string `schema:"raw_text"`
|
RawText string `schema:"raw_text"`
|
||||||
ServiceId string `schema:"service_id"`
|
ServiceId string `schema:"service_id"` //nolint:golint
|
||||||
Text string `schema:"text"`
|
Text string `schema:"text"`
|
||||||
TriggerWord string `schema:"trigger_word"`
|
TriggerWord string `schema:"trigger_word"`
|
||||||
|
FileIDs string `schema:"file_ids"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// Client for Mattermost.
|
// Client for Mattermost.
|
||||||
type Client struct {
|
type Client struct {
|
||||||
Url string // URL for incoming webhooks on mattermost.
|
// URL for incoming webhooks on mattermost.
|
||||||
|
Url string // nolint:golint
|
||||||
In chan IMessage
|
In chan IMessage
|
||||||
Out chan OMessage
|
Out chan OMessage
|
||||||
httpclient *http.Client
|
httpclient *http.Client
|
||||||
@ -65,7 +71,7 @@ type Config struct {
|
|||||||
func New(url string, config Config) *Client {
|
func New(url string, config Config) *Client {
|
||||||
c := &Client{Url: url, In: make(chan IMessage), Out: make(chan OMessage), Config: config}
|
c := &Client{Url: url, In: make(chan IMessage), Out: make(chan OMessage), Config: config}
|
||||||
tr := &http.Transport{
|
tr := &http.Transport{
|
||||||
TLSClientConfig: &tls.Config{InsecureSkipVerify: config.InsecureSkipVerify},
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: config.InsecureSkipVerify}, //nolint:gosec
|
||||||
}
|
}
|
||||||
c.httpclient = &http.Client{Transport: tr}
|
c.httpclient = &http.Client{Transport: tr}
|
||||||
if !c.DisableServer {
|
if !c.DisableServer {
|
||||||
@ -82,8 +88,14 @@ func New(url string, config Config) *Client {
|
|||||||
func (c *Client) StartServer() {
|
func (c *Client) StartServer() {
|
||||||
mux := http.NewServeMux()
|
mux := http.NewServeMux()
|
||||||
mux.Handle("/", c)
|
mux.Handle("/", c)
|
||||||
|
srv := &http.Server{
|
||||||
|
ReadTimeout: 5 * time.Second,
|
||||||
|
WriteTimeout: 10 * time.Second,
|
||||||
|
Handler: mux,
|
||||||
|
Addr: c.BindAddress,
|
||||||
|
}
|
||||||
log.Printf("Listening on http://%v...\n", c.BindAddress)
|
log.Printf("Listening on http://%v...\n", c.BindAddress)
|
||||||
if err := http.ListenAndServe(c.BindAddress, mux); err != nil {
|
if err := srv.ListenAndServe(); err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -127,12 +139,11 @@ func (c *Client) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
// Receive returns an incoming message from mattermost outgoing webhooks URL.
|
// Receive returns an incoming message from mattermost outgoing webhooks URL.
|
||||||
func (c *Client) Receive() IMessage {
|
func (c *Client) Receive() IMessage {
|
||||||
for {
|
var msg IMessage
|
||||||
select {
|
for msg := range c.In {
|
||||||
case msg := <-c.In:
|
return msg
|
||||||
return msg
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
return msg
|
||||||
}
|
}
|
||||||
|
|
||||||
// Send sends a msg to mattermost incoming webhooks URL.
|
// Send sends a msg to mattermost incoming webhooks URL.
|
||||||
|
50
migration.md
@ -1,50 +0,0 @@
|
|||||||
# Breaking changes from 0.4 to 0.5 for matterbridge (webhooks version)
|
|
||||||
## IRC section
|
|
||||||
### Server
|
|
||||||
Port removed, added to server
|
|
||||||
```
|
|
||||||
server="irc.freenode.net"
|
|
||||||
port=6667
|
|
||||||
```
|
|
||||||
changed to
|
|
||||||
```
|
|
||||||
server="irc.freenode.net:6667"
|
|
||||||
```
|
|
||||||
### Channel
|
|
||||||
Removed see Channels section below
|
|
||||||
|
|
||||||
### UseSlackCircumfix=true
|
|
||||||
Removed, can be done by using ```RemoteNickFormat="<{NICK}> "```
|
|
||||||
|
|
||||||
## Mattermost section
|
|
||||||
### BindAddress
|
|
||||||
Port removed, added to BindAddress
|
|
||||||
|
|
||||||
```
|
|
||||||
BindAddress="0.0.0.0"
|
|
||||||
port=9999
|
|
||||||
```
|
|
||||||
|
|
||||||
changed to
|
|
||||||
|
|
||||||
```
|
|
||||||
BindAddress="0.0.0.0:9999"
|
|
||||||
```
|
|
||||||
|
|
||||||
### Token
|
|
||||||
Removed
|
|
||||||
|
|
||||||
## Channels section
|
|
||||||
```
|
|
||||||
[Token "outgoingwebhooktoken1"]
|
|
||||||
IRCChannel="#off-topic"
|
|
||||||
MMChannel="off-topic"
|
|
||||||
```
|
|
||||||
|
|
||||||
changed to
|
|
||||||
|
|
||||||
```
|
|
||||||
[Channel "channelnameofchoice"]
|
|
||||||
IRC="#off-topic"
|
|
||||||
Mattermost="off-topic"
|
|
||||||
```
|
|
3
vendor/github.com/42wim/go-gitter/.gitignore
generated
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
.idea
|
||||||
|
/test
|
||||||
|
app.yaml
|
154
vendor/github.com/42wim/go-gitter/README.md
generated
vendored
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
# gitter
|
||||||
|
Gitter API in Go
|
||||||
|
https://developer.gitter.im
|
||||||
|
|
||||||
|
#### Install
|
||||||
|
|
||||||
|
`go get github.com/sromku/go-gitter`
|
||||||
|
|
||||||
|
- [Initialize](#initialize)
|
||||||
|
- [Users](#users)
|
||||||
|
- [Rooms](#rooms)
|
||||||
|
- [Messages](#messages)
|
||||||
|
- [Stream](#stream)
|
||||||
|
- [Faye (Experimental)](#faye-experimental)
|
||||||
|
- [Debug](#debug)
|
||||||
|
- [App Engine](#app-engine)
|
||||||
|
|
||||||
|
##### Initialize
|
||||||
|
``` Go
|
||||||
|
api := gitter.New("YOUR_ACCESS_TOKEN")
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Users
|
||||||
|
|
||||||
|
- Get current user
|
||||||
|
|
||||||
|
``` Go
|
||||||
|
user, err := api.GetUser()
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Rooms
|
||||||
|
|
||||||
|
- Get all rooms
|
||||||
|
``` Go
|
||||||
|
rooms, err := api.GetRooms()
|
||||||
|
```
|
||||||
|
|
||||||
|
- Get room by id
|
||||||
|
``` Go
|
||||||
|
room, err := api.GetRoom("roomID")
|
||||||
|
```
|
||||||
|
|
||||||
|
- Get rooms of some user
|
||||||
|
``` Go
|
||||||
|
rooms, err := api.GetRooms("userID")
|
||||||
|
```
|
||||||
|
|
||||||
|
- Join room
|
||||||
|
``` Go
|
||||||
|
room, err := api.JoinRoom("roomID", "userID")
|
||||||
|
```
|
||||||
|
|
||||||
|
- Leave room
|
||||||
|
``` Go
|
||||||
|
room, err := api.LeaveRoom("roomID", "userID")
|
||||||
|
```
|
||||||
|
|
||||||
|
- Get room id
|
||||||
|
``` Go
|
||||||
|
id, err := api.GetRoomId("room/uri")
|
||||||
|
```
|
||||||
|
|
||||||
|
- Search gitter rooms
|
||||||
|
``` Go
|
||||||
|
rooms, err := api.SearchRooms("search/string")
|
||||||
|
```
|
||||||
|
##### Messages
|
||||||
|
|
||||||
|
- Get messages of room
|
||||||
|
``` Go
|
||||||
|
messages, err := api.GetMessages("roomID", nil)
|
||||||
|
```
|
||||||
|
|
||||||
|
- Get one message
|
||||||
|
``` Go
|
||||||
|
message, err := api.GetMessage("roomID", "messageID")
|
||||||
|
```
|
||||||
|
|
||||||
|
- Send message
|
||||||
|
``` Go
|
||||||
|
err := api.SendMessage("roomID", "free chat text")
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Stream
|
||||||
|
|
||||||
|
Create stream to the room and start listening to incoming messages
|
||||||
|
|
||||||
|
``` Go
|
||||||
|
stream := api.Stream(room.Id)
|
||||||
|
go api.Listen(stream)
|
||||||
|
|
||||||
|
for {
|
||||||
|
event := <-stream.Event
|
||||||
|
switch ev := event.Data.(type) {
|
||||||
|
case *gitter.MessageReceived:
|
||||||
|
fmt.Println(ev.Message.From.Username + ": " + ev.Message.Text)
|
||||||
|
case *gitter.GitterConnectionClosed:
|
||||||
|
// connection was closed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Close stream connection
|
||||||
|
|
||||||
|
``` Go
|
||||||
|
stream.Close()
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Faye (Experimental)
|
||||||
|
|
||||||
|
``` Go
|
||||||
|
faye := api.Faye(room.ID)
|
||||||
|
go faye.Listen()
|
||||||
|
|
||||||
|
for {
|
||||||
|
event := <-faye.Event
|
||||||
|
switch ev := event.Data.(type) {
|
||||||
|
case *gitter.MessageReceived:
|
||||||
|
fmt.Println(ev.Message.From.Username + ": " + ev.Message.Text)
|
||||||
|
case *gitter.GitterConnectionClosed: //this one is never called in Faye
|
||||||
|
// connection was closed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Debug
|
||||||
|
|
||||||
|
You can print the internal errors by enabling debug to true
|
||||||
|
|
||||||
|
``` Go
|
||||||
|
api.SetDebug(true, nil)
|
||||||
|
```
|
||||||
|
|
||||||
|
You can also define your own `io.Writer` in case you want to persist the logs somewhere.
|
||||||
|
For example keeping the errors on file
|
||||||
|
|
||||||
|
``` Go
|
||||||
|
logFile, err := os.Create("gitter.log")
|
||||||
|
api.SetDebug(true, logFile)
|
||||||
|
```
|
||||||
|
|
||||||
|
##### App Engine
|
||||||
|
|
||||||
|
Initialize app engine client and continue as usual
|
||||||
|
|
||||||
|
``` Go
|
||||||
|
c := appengine.NewContext(r)
|
||||||
|
client := urlfetch.Client(c)
|
||||||
|
|
||||||
|
api := gitter.New("YOUR_ACCESS_TOKEN")
|
||||||
|
api.SetClient(client)
|
||||||
|
```
|
||||||
|
|
||||||
|
[Documentation](https://godoc.org/github.com/sromku/go-gitter)
|
107
vendor/github.com/42wim/go-gitter/gitter.go
generated
vendored
@ -127,7 +127,6 @@ func (gitter *Gitter) GetRooms() ([]Room, error) {
|
|||||||
|
|
||||||
// GetUsersInRoom returns the users in the room with the passed id
|
// GetUsersInRoom returns the users in the room with the passed id
|
||||||
func (gitter *Gitter) GetUsersInRoom(roomID string) ([]User, error) {
|
func (gitter *Gitter) GetUsersInRoom(roomID string) ([]User, error) {
|
||||||
|
|
||||||
var users []User
|
var users []User
|
||||||
response, err := gitter.get(gitter.config.apiBaseURL + "rooms/" + roomID + "/users")
|
response, err := gitter.get(gitter.config.apiBaseURL + "rooms/" + roomID + "/users")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -206,17 +205,43 @@ func (gitter *Gitter) GetMessage(roomID, messageID string) (*Message, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// SendMessage sends a message to a room
|
// SendMessage sends a message to a room
|
||||||
func (gitter *Gitter) SendMessage(roomID, text string) error {
|
func (gitter *Gitter) SendMessage(roomID, text string) (*Message, error) {
|
||||||
|
|
||||||
message := Message{Text: text}
|
message := Message{Text: text}
|
||||||
body, _ := json.Marshal(message)
|
body, _ := json.Marshal(message)
|
||||||
_, err := gitter.post(gitter.config.apiBaseURL+"rooms/"+roomID+"/chatMessages", body)
|
response, err := gitter.post(gitter.config.apiBaseURL+"rooms/"+roomID+"/chatMessages", body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
gitter.log(err)
|
gitter.log(err)
|
||||||
return err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
err = json.Unmarshal(response, &message)
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return &message, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// UpdateMessage updates a message in a room
|
||||||
|
func (gitter *Gitter) UpdateMessage(roomID, msgID, text string) (*Message, error) {
|
||||||
|
|
||||||
|
message := Message{Text: text}
|
||||||
|
body, _ := json.Marshal(message)
|
||||||
|
response, err := gitter.put(gitter.config.apiBaseURL+"rooms/"+roomID+"/chatMessages/"+msgID, body)
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = json.Unmarshal(response, &message)
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return &message, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// JoinRoom joins a room
|
// JoinRoom joins a room
|
||||||
@ -259,6 +284,45 @@ func (gitter *Gitter) SetDebug(debug bool, logWriter io.Writer) {
|
|||||||
gitter.logWriter = logWriter
|
gitter.logWriter = logWriter
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SearchRooms queries the Rooms resources of gitter API
|
||||||
|
func (gitter *Gitter) SearchRooms(room string) ([]Room, error) {
|
||||||
|
|
||||||
|
var rooms struct {
|
||||||
|
Results []Room `json:"results"`
|
||||||
|
}
|
||||||
|
|
||||||
|
response, err := gitter.get(gitter.config.apiBaseURL + "rooms?q=" + room)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = json.Unmarshal(response, &rooms)
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return rooms.Results, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetRoomId returns the room ID of a given URI
|
||||||
|
func (gitter *Gitter) GetRoomId(uri string) (string, error) {
|
||||||
|
|
||||||
|
rooms, err := gitter.SearchRooms(uri)
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, element := range rooms {
|
||||||
|
if element.URI == uri {
|
||||||
|
return element.ID, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return "", APIError{What: "Room not found."}
|
||||||
|
}
|
||||||
|
|
||||||
// Pagination params
|
// Pagination params
|
||||||
type Pagination struct {
|
type Pagination struct {
|
||||||
|
|
||||||
@ -376,6 +440,39 @@ func (gitter *Gitter) post(url string, body []byte) ([]byte, error) {
|
|||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (gitter *Gitter) put(url string, body []byte) ([]byte, error) {
|
||||||
|
r, err := http.NewRequest("PUT", url, bytes.NewBuffer(body))
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
r.Header.Set("Content-Type", "application/json")
|
||||||
|
r.Header.Set("Accept", "application/json")
|
||||||
|
r.Header.Set("Authorization", "Bearer "+gitter.config.token)
|
||||||
|
|
||||||
|
resp, err := gitter.config.client.Do(r)
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
err = APIError{What: fmt.Sprintf("Status code: %v", resp.StatusCode)}
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := ioutil.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
gitter.log(err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (gitter *Gitter) delete(url string) ([]byte, error) {
|
func (gitter *Gitter) delete(url string) ([]byte, error) {
|
||||||
r, err := http.NewRequest("delete", url, nil)
|
r, err := http.NewRequest("delete", url, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
20
vendor/github.com/42wim/go-gitter/stream.go
generated
vendored
@ -47,13 +47,13 @@ Loop:
|
|||||||
}
|
}
|
||||||
break Loop
|
break Loop
|
||||||
}
|
}
|
||||||
|
|
||||||
resp := stream.getResponse()
|
resp := stream.getResponse()
|
||||||
if resp.StatusCode != 200 {
|
if resp.StatusCode != 200 {
|
||||||
gitter.log(fmt.Sprintf("Unexpected response code %v", resp.StatusCode))
|
gitter.log(fmt.Sprintf("Unexpected response code %v", resp.StatusCode))
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
//"The JSON stream returns messages as JSON objects that are delimited by carriage return (\r)" <- Not true crap it's (\n) only
|
//"The JSON stream returns messages as JSON objects that are delimited by carriage return (\r)" <- Not true crap it's (\n) only
|
||||||
reader = bufio.NewReader(resp.Body)
|
reader = bufio.NewReader(resp.Body)
|
||||||
line, err := reader.ReadBytes('\n')
|
line, err := reader.ReadBytes('\n')
|
||||||
@ -112,6 +112,7 @@ type Stream struct {
|
|||||||
|
|
||||||
func (stream *Stream) destroy() {
|
func (stream *Stream) destroy() {
|
||||||
close(stream.Event)
|
close(stream.Event)
|
||||||
|
stream.streamConnection.currentRetries = 0
|
||||||
}
|
}
|
||||||
|
|
||||||
type Event struct {
|
type Event struct {
|
||||||
@ -135,10 +136,11 @@ func (stream *Stream) connect() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
res, err := stream.gitter.getResponse(stream.url, stream)
|
res, err := stream.gitter.getResponse(stream.url, stream)
|
||||||
if stream.streamConnection.canceled {
|
if err != nil || res.StatusCode != 200 {
|
||||||
// do nothing
|
stream.gitter.log("Failed to get response, trying reconnect")
|
||||||
} else if err != nil || res.StatusCode != 200 {
|
if res != nil {
|
||||||
stream.gitter.log("Failed to get response, trying reconnect ")
|
stream.gitter.log(fmt.Sprintf("Status code: %v", res.StatusCode))
|
||||||
|
}
|
||||||
stream.gitter.log(err)
|
stream.gitter.log(err)
|
||||||
|
|
||||||
// sleep and wait
|
// sleep and wait
|
||||||
@ -161,9 +163,6 @@ type streamConnection struct {
|
|||||||
// connection was closed
|
// connection was closed
|
||||||
closed bool
|
closed bool
|
||||||
|
|
||||||
// canceled
|
|
||||||
canceled bool
|
|
||||||
|
|
||||||
// wait time till next try
|
// wait time till next try
|
||||||
wait time.Duration
|
wait time.Duration
|
||||||
|
|
||||||
@ -192,13 +191,10 @@ func (stream *Stream) Close() {
|
|||||||
stream.gitter.log("Stream connection close request")
|
stream.gitter.log("Stream connection close request")
|
||||||
switch transport := stream.gitter.config.client.Transport.(type) {
|
switch transport := stream.gitter.config.client.Transport.(type) {
|
||||||
case *httpclient.Transport:
|
case *httpclient.Transport:
|
||||||
stream.streamConnection.canceled = true
|
|
||||||
transport.CancelRequest(conn.request)
|
transport.CancelRequest(conn.request)
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
conn.currentRetries = 0
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (stream *Stream) isClosed() bool {
|
func (stream *Stream) isClosed() bool {
|
||||||
|
434
vendor/github.com/42wim/matterbridge-plus/bridge/bridge.go
generated
vendored
@ -1,434 +0,0 @@
|
|||||||
package bridge
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto/tls"
|
|
||||||
"github.com/42wim/matterbridge-plus/matterclient"
|
|
||||||
"github.com/42wim/matterbridge/matterhook"
|
|
||||||
log "github.com/Sirupsen/logrus"
|
|
||||||
"github.com/peterhellberg/giphy"
|
|
||||||
ircm "github.com/sorcix/irc"
|
|
||||||
"github.com/thoj/go-ircevent"
|
|
||||||
"regexp"
|
|
||||||
"sort"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
//type Bridge struct {
|
|
||||||
type MMhook struct {
|
|
||||||
mh *matterhook.Client
|
|
||||||
}
|
|
||||||
|
|
||||||
type MMapi struct {
|
|
||||||
mc *matterclient.MMClient
|
|
||||||
mmMap map[string]string
|
|
||||||
mmIgnoreNicks []string
|
|
||||||
}
|
|
||||||
|
|
||||||
type MMirc struct {
|
|
||||||
i *irc.Connection
|
|
||||||
ircNick string
|
|
||||||
ircMap map[string]string
|
|
||||||
names map[string][]string
|
|
||||||
ircIgnoreNicks []string
|
|
||||||
}
|
|
||||||
|
|
||||||
type MMMessage struct {
|
|
||||||
Text string
|
|
||||||
Channel string
|
|
||||||
Username string
|
|
||||||
}
|
|
||||||
|
|
||||||
type Bridge struct {
|
|
||||||
MMhook
|
|
||||||
MMapi
|
|
||||||
MMirc
|
|
||||||
*Config
|
|
||||||
kind string
|
|
||||||
}
|
|
||||||
|
|
||||||
type FancyLog struct {
|
|
||||||
irc *log.Entry
|
|
||||||
mm *log.Entry
|
|
||||||
}
|
|
||||||
|
|
||||||
var flog FancyLog
|
|
||||||
|
|
||||||
const Legacy = "legacy"
|
|
||||||
|
|
||||||
func initFLog() {
|
|
||||||
flog.irc = log.WithFields(log.Fields{"module": "irc"})
|
|
||||||
flog.mm = log.WithFields(log.Fields{"module": "mattermost"})
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewBridge(name string, config *Config, kind string) *Bridge {
|
|
||||||
initFLog()
|
|
||||||
b := &Bridge{}
|
|
||||||
b.Config = config
|
|
||||||
b.kind = kind
|
|
||||||
b.ircNick = b.Config.IRC.Nick
|
|
||||||
b.ircMap = make(map[string]string)
|
|
||||||
b.MMirc.names = make(map[string][]string)
|
|
||||||
b.ircIgnoreNicks = strings.Fields(b.Config.IRC.IgnoreNicks)
|
|
||||||
b.mmIgnoreNicks = strings.Fields(b.Config.Mattermost.IgnoreNicks)
|
|
||||||
if kind == Legacy {
|
|
||||||
if len(b.Config.Token) > 0 {
|
|
||||||
for _, val := range b.Config.Token {
|
|
||||||
b.ircMap[val.IRCChannel] = val.MMChannel
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
b.mh = matterhook.New(b.Config.Mattermost.URL,
|
|
||||||
matterhook.Config{Port: b.Config.Mattermost.Port, Token: b.Config.Mattermost.Token,
|
|
||||||
InsecureSkipVerify: b.Config.Mattermost.SkipTLSVerify,
|
|
||||||
BindAddress: b.Config.Mattermost.BindAddress})
|
|
||||||
} else {
|
|
||||||
b.mmMap = make(map[string]string)
|
|
||||||
if len(b.Config.Channel) > 0 {
|
|
||||||
for _, val := range b.Config.Channel {
|
|
||||||
b.ircMap[val.IRC] = val.Mattermost
|
|
||||||
b.mmMap[val.Mattermost] = val.IRC
|
|
||||||
}
|
|
||||||
}
|
|
||||||
b.mc = matterclient.New(b.Config.Mattermost.Login, b.Config.Mattermost.Password,
|
|
||||||
b.Config.Mattermost.Team, b.Config.Mattermost.Server)
|
|
||||||
b.mc.SkipTLSVerify = b.Config.Mattermost.SkipTLSVerify
|
|
||||||
b.mc.NoTLS = b.Config.Mattermost.NoTLS
|
|
||||||
flog.mm.Infof("Trying login %s (team: %s) on %s", b.Config.Mattermost.Login, b.Config.Mattermost.Team, b.Config.Mattermost.Server)
|
|
||||||
err := b.mc.Login()
|
|
||||||
if err != nil {
|
|
||||||
flog.mm.Fatal("Can not connect", err)
|
|
||||||
}
|
|
||||||
flog.mm.Info("Login ok")
|
|
||||||
b.mc.JoinChannel(b.Config.Mattermost.Channel)
|
|
||||||
if len(b.Config.Channel) > 0 {
|
|
||||||
for _, val := range b.Config.Channel {
|
|
||||||
b.mc.JoinChannel(val.Mattermost)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
go b.mc.WsReceiver()
|
|
||||||
}
|
|
||||||
flog.irc.Info("Trying IRC connection")
|
|
||||||
b.i = b.createIRC(name)
|
|
||||||
flog.irc.Info("Connection succeeded")
|
|
||||||
go b.handleMatter()
|
|
||||||
return b
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) createIRC(name string) *irc.Connection {
|
|
||||||
i := irc.IRC(b.Config.IRC.Nick, b.Config.IRC.Nick)
|
|
||||||
i.UseTLS = b.Config.IRC.UseTLS
|
|
||||||
i.TLSConfig = &tls.Config{InsecureSkipVerify: b.Config.IRC.SkipTLSVerify}
|
|
||||||
if b.Config.IRC.Password != "" {
|
|
||||||
i.Password = b.Config.IRC.Password
|
|
||||||
}
|
|
||||||
i.AddCallback(ircm.RPL_WELCOME, b.handleNewConnection)
|
|
||||||
i.Connect(b.Config.IRC.Server + ":" + strconv.Itoa(b.Config.IRC.Port))
|
|
||||||
return i
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleNewConnection(event *irc.Event) {
|
|
||||||
flog.irc.Info("Registering callbacks")
|
|
||||||
i := b.i
|
|
||||||
b.ircNick = event.Arguments[0]
|
|
||||||
i.AddCallback("PRIVMSG", b.handlePrivMsg)
|
|
||||||
i.AddCallback("CTCP_ACTION", b.handlePrivMsg)
|
|
||||||
i.AddCallback(ircm.RPL_ENDOFNAMES, b.endNames)
|
|
||||||
i.AddCallback(ircm.RPL_NAMREPLY, b.storeNames)
|
|
||||||
i.AddCallback(ircm.RPL_TOPICWHOTIME, b.handleTopicWhoTime)
|
|
||||||
i.AddCallback(ircm.NOTICE, b.handleNotice)
|
|
||||||
i.AddCallback(ircm.RPL_MYINFO, func(e *irc.Event) { flog.irc.Infof("%s: %s", e.Code, strings.Join(e.Arguments[1:], " ")) })
|
|
||||||
i.AddCallback("PING", func(e *irc.Event) {
|
|
||||||
i.SendRaw("PONG :" + e.Message())
|
|
||||||
flog.irc.Debugf("PING/PONG")
|
|
||||||
})
|
|
||||||
if b.Config.Mattermost.ShowJoinPart {
|
|
||||||
i.AddCallback("JOIN", b.handleJoinPart)
|
|
||||||
i.AddCallback("PART", b.handleJoinPart)
|
|
||||||
}
|
|
||||||
i.AddCallback("*", b.handleOther)
|
|
||||||
b.setupChannels()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) setupChannels() {
|
|
||||||
i := b.i
|
|
||||||
if b.Config.IRC.Channel != "" {
|
|
||||||
flog.irc.Infof("Joining %s as %s", b.Config.IRC.Channel, b.ircNick)
|
|
||||||
i.Join(b.Config.IRC.Channel)
|
|
||||||
}
|
|
||||||
if b.kind == Legacy {
|
|
||||||
for _, val := range b.Config.Token {
|
|
||||||
flog.irc.Infof("Joining %s as %s", val.IRCChannel, b.ircNick)
|
|
||||||
i.Join(val.IRCChannel)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
for _, val := range b.Config.Channel {
|
|
||||||
flog.irc.Infof("Joining %s as %s", val.IRC, b.ircNick)
|
|
||||||
i.Join(val.IRC)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleIrcBotCommand(event *irc.Event) bool {
|
|
||||||
parts := strings.Fields(event.Message())
|
|
||||||
exp, _ := regexp.Compile("[:,]+$")
|
|
||||||
channel := event.Arguments[0]
|
|
||||||
command := ""
|
|
||||||
if len(parts) == 2 {
|
|
||||||
command = parts[1]
|
|
||||||
}
|
|
||||||
if exp.ReplaceAllString(parts[0], "") == b.ircNick {
|
|
||||||
switch command {
|
|
||||||
case "users":
|
|
||||||
usernames := b.mc.UsernamesInChannel(b.getMMChannel(channel))
|
|
||||||
sort.Strings(usernames)
|
|
||||||
b.i.Privmsg(channel, "Users on Mattermost: "+strings.Join(usernames, ", "))
|
|
||||||
default:
|
|
||||||
b.i.Privmsg(channel, "Valid commands are: [users, help]")
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) ircNickFormat(nick string) string {
|
|
||||||
if nick == b.ircNick {
|
|
||||||
return nick
|
|
||||||
}
|
|
||||||
if b.Config.Mattermost.RemoteNickFormat == nil {
|
|
||||||
return "irc-" + nick
|
|
||||||
}
|
|
||||||
return strings.Replace(*b.Config.Mattermost.RemoteNickFormat, "{NICK}", nick, -1)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handlePrivMsg(event *irc.Event) {
|
|
||||||
if b.ignoreMessage(event.Nick, event.Message(), "irc") {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if b.handleIrcBotCommand(event) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
msg := ""
|
|
||||||
if event.Code == "CTCP_ACTION" {
|
|
||||||
msg = event.Nick + " "
|
|
||||||
}
|
|
||||||
msg += event.Message()
|
|
||||||
b.Send(b.ircNickFormat(event.Nick), msg, b.getMMChannel(event.Arguments[0]))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleJoinPart(event *irc.Event) {
|
|
||||||
b.Send(b.ircNick, b.ircNickFormat(event.Nick)+" "+strings.ToLower(event.Code)+"s "+event.Message(), b.getMMChannel(event.Arguments[0]))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleNotice(event *irc.Event) {
|
|
||||||
if strings.Contains(event.Message(), "This nickname is registered") {
|
|
||||||
b.i.Privmsg(b.Config.IRC.NickServNick, "IDENTIFY "+b.Config.IRC.NickServPassword)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) nicksPerRow() int {
|
|
||||||
if b.Config.Mattermost.NicksPerRow < 1 {
|
|
||||||
return 4
|
|
||||||
}
|
|
||||||
return b.Config.Mattermost.NicksPerRow
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) formatnicks(nicks []string, continued bool) string {
|
|
||||||
switch b.Config.Mattermost.NickFormatter {
|
|
||||||
case "table":
|
|
||||||
return tableformatter(nicks, b.nicksPerRow(), continued)
|
|
||||||
default:
|
|
||||||
return plainformatter(nicks, b.nicksPerRow())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) storeNames(event *irc.Event) {
|
|
||||||
channel := event.Arguments[2]
|
|
||||||
b.MMirc.names[channel] = append(
|
|
||||||
b.MMirc.names[channel],
|
|
||||||
strings.Split(strings.TrimSpace(event.Message()), " ")...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) endNames(event *irc.Event) {
|
|
||||||
channel := event.Arguments[1]
|
|
||||||
sort.Strings(b.MMirc.names[channel])
|
|
||||||
maxNamesPerPost := (300 / b.nicksPerRow()) * b.nicksPerRow()
|
|
||||||
continued := false
|
|
||||||
for len(b.MMirc.names[channel]) > maxNamesPerPost {
|
|
||||||
b.Send(
|
|
||||||
b.ircNick,
|
|
||||||
b.formatnicks(b.MMirc.names[channel][0:maxNamesPerPost], continued),
|
|
||||||
b.getMMChannel(channel))
|
|
||||||
b.MMirc.names[channel] = b.MMirc.names[channel][maxNamesPerPost:]
|
|
||||||
continued = true
|
|
||||||
}
|
|
||||||
b.Send(b.ircNick, b.formatnicks(b.MMirc.names[channel], continued), b.getMMChannel(channel))
|
|
||||||
b.MMirc.names[channel] = nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleTopicWhoTime(event *irc.Event) {
|
|
||||||
parts := strings.Split(event.Arguments[2], "!")
|
|
||||||
t, err := strconv.ParseInt(event.Arguments[3], 10, 64)
|
|
||||||
if err != nil {
|
|
||||||
flog.irc.Errorf("Invalid time stamp: %s", event.Arguments[3])
|
|
||||||
}
|
|
||||||
user := parts[0]
|
|
||||||
if len(parts) > 1 {
|
|
||||||
user += " [" + parts[1] + "]"
|
|
||||||
}
|
|
||||||
flog.irc.Infof("%s: Topic set by %s [%s]", event.Code, user, time.Unix(t, 0))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleOther(event *irc.Event) {
|
|
||||||
flog.irc.Debugf("%#v", event)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) Send(nick string, message string, channel string) error {
|
|
||||||
return b.SendType(nick, message, channel, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) SendType(nick string, message string, channel string, mtype string) error {
|
|
||||||
if b.Config.Mattermost.PrefixMessagesWithNick {
|
|
||||||
if IsMarkup(message) {
|
|
||||||
message = nick + "\n\n" + message
|
|
||||||
} else {
|
|
||||||
message = nick + " " + message
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if b.kind == Legacy {
|
|
||||||
matterMessage := matterhook.OMessage{IconURL: b.Config.Mattermost.IconURL}
|
|
||||||
matterMessage.Channel = channel
|
|
||||||
matterMessage.UserName = nick
|
|
||||||
matterMessage.Type = mtype
|
|
||||||
matterMessage.Text = message
|
|
||||||
err := b.mh.Send(matterMessage)
|
|
||||||
if err != nil {
|
|
||||||
flog.mm.Info(err)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
flog.mm.Debug("->mattermost channel: ", channel, " ", message)
|
|
||||||
b.mc.PostMessage(channel, message)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleMatterHook(mchan chan *MMMessage) {
|
|
||||||
for {
|
|
||||||
message := b.mh.Receive()
|
|
||||||
m := &MMMessage{}
|
|
||||||
m.Username = message.UserName
|
|
||||||
m.Text = message.Text
|
|
||||||
m.Channel = message.Token
|
|
||||||
mchan <- m
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleMatterClient(mchan chan *MMMessage) {
|
|
||||||
for message := range b.mc.MessageChan {
|
|
||||||
// do not post our own messages back to irc
|
|
||||||
if message.Raw.Action == "posted" && b.mc.User.Username != message.Username {
|
|
||||||
m := &MMMessage{}
|
|
||||||
m.Username = message.Username
|
|
||||||
m.Channel = message.Channel
|
|
||||||
m.Text = message.Text
|
|
||||||
flog.mm.Debugf("<-mattermost channel: %s %#v %#v", message.Channel, message.Post, message.Raw)
|
|
||||||
mchan <- m
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) handleMatter() {
|
|
||||||
flog.mm.Infof("Choosing Mattermost connection type %s", b.kind)
|
|
||||||
mchan := make(chan *MMMessage)
|
|
||||||
if b.kind == Legacy {
|
|
||||||
go b.handleMatterHook(mchan)
|
|
||||||
} else {
|
|
||||||
go b.handleMatterClient(mchan)
|
|
||||||
}
|
|
||||||
flog.mm.Info("Start listening for Mattermost messages")
|
|
||||||
for message := range mchan {
|
|
||||||
var username string
|
|
||||||
if b.ignoreMessage(message.Username, message.Text, "mattermost") {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
username = message.Username + ": "
|
|
||||||
if b.Config.IRC.RemoteNickFormat != "" {
|
|
||||||
username = strings.Replace(b.Config.IRC.RemoteNickFormat, "{NICK}", message.Username, -1)
|
|
||||||
} else if b.Config.IRC.UseSlackCircumfix {
|
|
||||||
username = "<" + message.Username + "> "
|
|
||||||
}
|
|
||||||
cmds := strings.Fields(message.Text)
|
|
||||||
// empty message
|
|
||||||
if len(cmds) == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
cmd := cmds[0]
|
|
||||||
switch cmd {
|
|
||||||
case "!users":
|
|
||||||
flog.mm.Info("Received !users from ", message.Username)
|
|
||||||
b.i.SendRaw("NAMES " + b.getIRCChannel(message.Channel))
|
|
||||||
continue
|
|
||||||
case "!gif":
|
|
||||||
message.Text = b.giphyRandom(strings.Fields(strings.Replace(message.Text, "!gif ", "", 1)))
|
|
||||||
b.Send(b.ircNick, message.Text, b.getIRCChannel(message.Channel))
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
texts := strings.Split(message.Text, "\n")
|
|
||||||
for _, text := range texts {
|
|
||||||
flog.mm.Debug("Sending message from " + message.Username + " to " + message.Channel)
|
|
||||||
b.i.Privmsg(b.getIRCChannel(message.Channel), username+text)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) giphyRandom(query []string) string {
|
|
||||||
g := giphy.DefaultClient
|
|
||||||
if b.Config.General.GiphyAPIKey != "" {
|
|
||||||
g.APIKey = b.Config.General.GiphyAPIKey
|
|
||||||
}
|
|
||||||
res, err := g.Random(query)
|
|
||||||
if err != nil {
|
|
||||||
return "error"
|
|
||||||
}
|
|
||||||
return res.Data.FixedHeightDownsampledURL
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) getMMChannel(ircChannel string) string {
|
|
||||||
mmchannel, ok := b.ircMap[ircChannel]
|
|
||||||
if !ok {
|
|
||||||
mmchannel = b.Config.Mattermost.Channel
|
|
||||||
}
|
|
||||||
return mmchannel
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) getIRCChannel(channel string) string {
|
|
||||||
if b.kind == Legacy {
|
|
||||||
ircchannel := b.Config.IRC.Channel
|
|
||||||
_, ok := b.Config.Token[channel]
|
|
||||||
if ok {
|
|
||||||
ircchannel = b.Config.Token[channel].IRCChannel
|
|
||||||
}
|
|
||||||
return ircchannel
|
|
||||||
}
|
|
||||||
ircchannel, ok := b.mmMap[channel]
|
|
||||||
if !ok {
|
|
||||||
ircchannel = b.Config.IRC.Channel
|
|
||||||
}
|
|
||||||
return ircchannel
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *Bridge) ignoreMessage(nick string, message string, protocol string) bool {
|
|
||||||
var ignoreNicks = b.mmIgnoreNicks
|
|
||||||
if protocol == "irc" {
|
|
||||||
ignoreNicks = b.ircIgnoreNicks
|
|
||||||
}
|
|
||||||
// should we discard messages ?
|
|
||||||
for _, entry := range ignoreNicks {
|
|
||||||
if nick == entry {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
68
vendor/github.com/42wim/matterbridge-plus/bridge/config.go
generated
vendored
@ -1,68 +0,0 @@
|
|||||||
package bridge
|
|
||||||
|
|
||||||
import (
|
|
||||||
"gopkg.in/gcfg.v1"
|
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Config struct {
|
|
||||||
IRC struct {
|
|
||||||
UseTLS bool
|
|
||||||
SkipTLSVerify bool
|
|
||||||
Server string
|
|
||||||
Port int
|
|
||||||
Nick string
|
|
||||||
Password string
|
|
||||||
Channel string
|
|
||||||
UseSlackCircumfix bool
|
|
||||||
NickServNick string
|
|
||||||
NickServPassword string
|
|
||||||
RemoteNickFormat string
|
|
||||||
IgnoreNicks string
|
|
||||||
}
|
|
||||||
Mattermost struct {
|
|
||||||
URL string
|
|
||||||
Port int
|
|
||||||
ShowJoinPart bool
|
|
||||||
Token string
|
|
||||||
IconURL string
|
|
||||||
SkipTLSVerify bool
|
|
||||||
BindAddress string
|
|
||||||
Channel string
|
|
||||||
PrefixMessagesWithNick bool
|
|
||||||
NicksPerRow int
|
|
||||||
NickFormatter string
|
|
||||||
Server string
|
|
||||||
Team string
|
|
||||||
Login string
|
|
||||||
Password string
|
|
||||||
RemoteNickFormat *string
|
|
||||||
IgnoreNicks string
|
|
||||||
NoTLS bool
|
|
||||||
}
|
|
||||||
Token map[string]*struct {
|
|
||||||
IRCChannel string
|
|
||||||
MMChannel string
|
|
||||||
}
|
|
||||||
Channel map[string]*struct {
|
|
||||||
IRC string
|
|
||||||
Mattermost string
|
|
||||||
}
|
|
||||||
General struct {
|
|
||||||
GiphyAPIKey string
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewConfig(cfgfile string) *Config {
|
|
||||||
var cfg Config
|
|
||||||
content, err := ioutil.ReadFile(cfgfile)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
err = gcfg.ReadStringInto(&cfg, string(content))
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal("Failed to parse "+cfgfile+":", err)
|
|
||||||
}
|
|
||||||
return &cfg
|
|
||||||
}
|
|
59
vendor/github.com/42wim/matterbridge-plus/bridge/helper.go
generated
vendored
@ -1,59 +0,0 @@
|
|||||||
package bridge
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
func tableformatter(nicks []string, nicksPerRow int, continued bool) string {
|
|
||||||
result := "|IRC users"
|
|
||||||
if continued {
|
|
||||||
result = "|(continued)"
|
|
||||||
}
|
|
||||||
for i := 0; i < 2; i++ {
|
|
||||||
for j := 1; j <= nicksPerRow && j <= len(nicks); j++ {
|
|
||||||
if i == 0 {
|
|
||||||
result += "|"
|
|
||||||
} else {
|
|
||||||
result += ":-|"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
result += "\r\n|"
|
|
||||||
}
|
|
||||||
result += nicks[0] + "|"
|
|
||||||
for i := 1; i < len(nicks); i++ {
|
|
||||||
if i%nicksPerRow == 0 {
|
|
||||||
result += "\r\n|" + nicks[i] + "|"
|
|
||||||
} else {
|
|
||||||
result += nicks[i] + "|"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
func plainformatter(nicks []string, nicksPerRow int) string {
|
|
||||||
return strings.Join(nicks, ", ") + " currently on IRC"
|
|
||||||
}
|
|
||||||
|
|
||||||
func IsMarkup(message string) bool {
|
|
||||||
switch message[0] {
|
|
||||||
case '|':
|
|
||||||
fallthrough
|
|
||||||
case '#':
|
|
||||||
fallthrough
|
|
||||||
case '_':
|
|
||||||
fallthrough
|
|
||||||
case '*':
|
|
||||||
fallthrough
|
|
||||||
case '~':
|
|
||||||
fallthrough
|
|
||||||
case '-':
|
|
||||||
fallthrough
|
|
||||||
case ':':
|
|
||||||
fallthrough
|
|
||||||
case '>':
|
|
||||||
fallthrough
|
|
||||||
case '=':
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
441
vendor/github.com/42wim/matterbridge-plus/matterclient/matterclient.go
generated
vendored
@ -1,441 +0,0 @@
|
|||||||
package matterclient
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto/tls"
|
|
||||||
"errors"
|
|
||||||
log "github.com/Sirupsen/logrus"
|
|
||||||
"net/http"
|
|
||||||
"net/http/cookiejar"
|
|
||||||
"net/url"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/gorilla/websocket"
|
|
||||||
"github.com/jpillora/backoff"
|
|
||||||
"github.com/mattermost/platform/model"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Credentials struct {
|
|
||||||
Login string
|
|
||||||
Team string
|
|
||||||
Pass string
|
|
||||||
Server string
|
|
||||||
NoTLS bool
|
|
||||||
SkipTLSVerify bool
|
|
||||||
}
|
|
||||||
|
|
||||||
type Message struct {
|
|
||||||
Raw *model.Message
|
|
||||||
Post *model.Post
|
|
||||||
Team string
|
|
||||||
Channel string
|
|
||||||
Username string
|
|
||||||
Text string
|
|
||||||
}
|
|
||||||
|
|
||||||
type MMClient struct {
|
|
||||||
*Credentials
|
|
||||||
Client *model.Client
|
|
||||||
WsClient *websocket.Conn
|
|
||||||
WsQuit bool
|
|
||||||
WsAway bool
|
|
||||||
Channels *model.ChannelList
|
|
||||||
MoreChannels *model.ChannelList
|
|
||||||
User *model.User
|
|
||||||
Users map[string]*model.User
|
|
||||||
MessageChan chan *Message
|
|
||||||
Team *model.Team
|
|
||||||
log *log.Entry
|
|
||||||
}
|
|
||||||
|
|
||||||
func New(login, pass, team, server string) *MMClient {
|
|
||||||
cred := &Credentials{Login: login, Pass: pass, Team: team, Server: server}
|
|
||||||
mmclient := &MMClient{Credentials: cred, MessageChan: make(chan *Message, 100)}
|
|
||||||
mmclient.log = log.WithFields(log.Fields{"module": "matterclient"})
|
|
||||||
log.SetFormatter(&log.TextFormatter{FullTimestamp: true})
|
|
||||||
return mmclient
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) SetLogLevel(level string) {
|
|
||||||
l, err := log.ParseLevel(level)
|
|
||||||
if err != nil {
|
|
||||||
log.SetLevel(log.InfoLevel)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
log.SetLevel(l)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) Login() error {
|
|
||||||
if m.WsQuit {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
b := &backoff.Backoff{
|
|
||||||
Min: time.Second,
|
|
||||||
Max: 5 * time.Minute,
|
|
||||||
Jitter: true,
|
|
||||||
}
|
|
||||||
uriScheme := "https://"
|
|
||||||
wsScheme := "wss://"
|
|
||||||
if m.NoTLS {
|
|
||||||
uriScheme = "http://"
|
|
||||||
wsScheme = "ws://"
|
|
||||||
}
|
|
||||||
// login to mattermost
|
|
||||||
m.Client = model.NewClient(uriScheme + m.Credentials.Server)
|
|
||||||
m.Client.HttpClient.Transport = &http.Transport{TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}}
|
|
||||||
var myinfo *model.Result
|
|
||||||
var appErr *model.AppError
|
|
||||||
var logmsg = "trying login"
|
|
||||||
for {
|
|
||||||
m.log.Debugf("%s %s %s %s", logmsg, m.Credentials.Team, m.Credentials.Login, m.Credentials.Server)
|
|
||||||
if strings.Contains(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN) {
|
|
||||||
m.log.Debugf(logmsg+" with ", model.SESSION_COOKIE_TOKEN)
|
|
||||||
token := strings.Split(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN+"=")
|
|
||||||
m.Client.HttpClient.Jar = m.createCookieJar(token[1])
|
|
||||||
m.Client.MockSession(token[1])
|
|
||||||
myinfo, appErr = m.Client.GetMe("")
|
|
||||||
if myinfo.Data.(*model.User) == nil {
|
|
||||||
m.log.Errorf("LOGIN TOKEN: %s is invalid", m.Credentials.Pass)
|
|
||||||
return errors.New("invalid " + model.SESSION_COOKIE_TOKEN)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
myinfo, appErr = m.Client.Login(m.Credentials.Login, m.Credentials.Pass)
|
|
||||||
}
|
|
||||||
if appErr != nil {
|
|
||||||
d := b.Duration()
|
|
||||||
m.log.Debug(appErr.DetailedError)
|
|
||||||
if !strings.Contains(appErr.DetailedError, "connection refused") &&
|
|
||||||
!strings.Contains(appErr.DetailedError, "invalid character") {
|
|
||||||
if appErr.Message == "" {
|
|
||||||
return errors.New(appErr.DetailedError)
|
|
||||||
}
|
|
||||||
return errors.New(appErr.Message)
|
|
||||||
}
|
|
||||||
m.log.Debugf("LOGIN: %s, reconnecting in %s", appErr, d)
|
|
||||||
time.Sleep(d)
|
|
||||||
logmsg = "retrying login"
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
// reset timer
|
|
||||||
b.Reset()
|
|
||||||
|
|
||||||
initLoad, _ := m.Client.GetInitialLoad()
|
|
||||||
initData := initLoad.Data.(*model.InitialLoad)
|
|
||||||
m.User = initData.User
|
|
||||||
for _, v := range initData.Teams {
|
|
||||||
m.log.Debugf("trying %s (id: %s)", v.Name, v.Id)
|
|
||||||
if v.Name == m.Credentials.Team {
|
|
||||||
m.Client.SetTeamId(v.Id)
|
|
||||||
m.Team = v
|
|
||||||
m.log.Debugf("GetallTeamListings: found id %s for team %s", v.Id, v.Name)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if m.Team == nil {
|
|
||||||
return errors.New("team not found")
|
|
||||||
}
|
|
||||||
|
|
||||||
// setup websocket connection
|
|
||||||
wsurl := wsScheme + m.Credentials.Server + "/api/v3/users/websocket"
|
|
||||||
header := http.Header{}
|
|
||||||
header.Set(model.HEADER_AUTH, "BEARER "+m.Client.AuthToken)
|
|
||||||
|
|
||||||
m.log.Debug("WsClient: making connection")
|
|
||||||
var err error
|
|
||||||
for {
|
|
||||||
wsDialer := &websocket.Dialer{Proxy: http.ProxyFromEnvironment, TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}}
|
|
||||||
m.WsClient, _, err = wsDialer.Dial(wsurl, header)
|
|
||||||
if err != nil {
|
|
||||||
d := b.Duration()
|
|
||||||
m.log.Debugf("WSS: %s, reconnecting in %s", err, d)
|
|
||||||
time.Sleep(d)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
b.Reset()
|
|
||||||
|
|
||||||
// populating users
|
|
||||||
m.UpdateUsers()
|
|
||||||
|
|
||||||
// populating channels
|
|
||||||
m.UpdateChannels()
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) WsReceiver() {
|
|
||||||
var rmsg model.Message
|
|
||||||
for {
|
|
||||||
if m.WsQuit {
|
|
||||||
m.log.Debug("exiting WsReceiver")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if err := m.WsClient.ReadJSON(&rmsg); err != nil {
|
|
||||||
m.log.Error("error:", err)
|
|
||||||
// reconnect
|
|
||||||
m.Login()
|
|
||||||
}
|
|
||||||
if rmsg.Action == "ping" {
|
|
||||||
m.handleWsPing()
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
msg := &Message{Raw: &rmsg, Team: m.Credentials.Team}
|
|
||||||
m.parseMessage(msg)
|
|
||||||
m.MessageChan <- msg
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) handleWsPing() {
|
|
||||||
m.log.Debug("Ws PING")
|
|
||||||
if !m.WsQuit && !m.WsAway {
|
|
||||||
m.log.Debug("Ws PONG")
|
|
||||||
m.WsClient.WriteMessage(websocket.PongMessage, []byte{})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) parseMessage(rmsg *Message) {
|
|
||||||
switch rmsg.Raw.Action {
|
|
||||||
case model.ACTION_POSTED:
|
|
||||||
m.parseActionPost(rmsg)
|
|
||||||
/*
|
|
||||||
case model.ACTION_USER_REMOVED:
|
|
||||||
m.handleWsActionUserRemoved(&rmsg)
|
|
||||||
case model.ACTION_USER_ADDED:
|
|
||||||
m.handleWsActionUserAdded(&rmsg)
|
|
||||||
*/
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) parseActionPost(rmsg *Message) {
|
|
||||||
data := model.PostFromJson(strings.NewReader(rmsg.Raw.Props["post"]))
|
|
||||||
// log.Println("receiving userid", data.UserId)
|
|
||||||
// we don't have the user, refresh the userlist
|
|
||||||
if m.Users[data.UserId] == nil {
|
|
||||||
m.UpdateUsers()
|
|
||||||
}
|
|
||||||
rmsg.Username = m.Users[data.UserId].Username
|
|
||||||
rmsg.Channel = m.GetChannelName(data.ChannelId)
|
|
||||||
// direct message
|
|
||||||
if strings.Contains(rmsg.Channel, "__") {
|
|
||||||
//log.Println("direct message")
|
|
||||||
rcvusers := strings.Split(rmsg.Channel, "__")
|
|
||||||
if rcvusers[0] != m.User.Id {
|
|
||||||
rmsg.Channel = m.Users[rcvusers[0]].Username
|
|
||||||
} else {
|
|
||||||
rmsg.Channel = m.Users[rcvusers[1]].Username
|
|
||||||
}
|
|
||||||
}
|
|
||||||
rmsg.Text = data.Message
|
|
||||||
rmsg.Post = data
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UpdateUsers() error {
|
|
||||||
mmusers, _ := m.Client.GetProfilesForDirectMessageList(m.Team.Id)
|
|
||||||
m.Users = mmusers.Data.(map[string]*model.User)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UpdateChannels() error {
|
|
||||||
mmchannels, _ := m.Client.GetChannels("")
|
|
||||||
m.Channels = mmchannels.Data.(*model.ChannelList)
|
|
||||||
mmchannels, _ = m.Client.GetMoreChannels("")
|
|
||||||
m.MoreChannels = mmchannels.Data.(*model.ChannelList)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetChannelName(id string) string {
|
|
||||||
for _, channel := range append(m.Channels.Channels, m.MoreChannels.Channels...) {
|
|
||||||
if channel.Id == id {
|
|
||||||
return channel.Name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// not found? could be a new direct message from mattermost. Try to update and check again
|
|
||||||
m.UpdateChannels()
|
|
||||||
for _, channel := range append(m.Channels.Channels, m.MoreChannels.Channels...) {
|
|
||||||
if channel.Id == id {
|
|
||||||
return channel.Name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetChannelId(name string) string {
|
|
||||||
for _, channel := range append(m.Channels.Channels, m.MoreChannels.Channels...) {
|
|
||||||
if channel.Name == name {
|
|
||||||
return channel.Id
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetChannelHeader(id string) string {
|
|
||||||
for _, channel := range append(m.Channels.Channels, m.MoreChannels.Channels...) {
|
|
||||||
if channel.Id == id {
|
|
||||||
return channel.Header
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) PostMessage(channel string, text string) {
|
|
||||||
post := &model.Post{ChannelId: m.GetChannelId(channel), Message: text}
|
|
||||||
m.Client.CreatePost(post)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) JoinChannel(channel string) error {
|
|
||||||
cleanChan := strings.Replace(channel, "#", "", 1)
|
|
||||||
if m.GetChannelId(cleanChan) == "" {
|
|
||||||
return errors.New("failed to join")
|
|
||||||
}
|
|
||||||
for _, c := range m.Channels.Channels {
|
|
||||||
if c.Name == cleanChan {
|
|
||||||
m.log.Debug("Not joining ", cleanChan, " already joined.")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
m.log.Debug("Joining ", cleanChan)
|
|
||||||
_, err := m.Client.JoinChannel(m.GetChannelId(cleanChan))
|
|
||||||
if err != nil {
|
|
||||||
return errors.New("failed to join")
|
|
||||||
}
|
|
||||||
// m.SyncChannel(m.getMMChannelId(strings.Replace(channel, "#", "", 1)), strings.Replace(channel, "#", "", 1))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetPostsSince(channelId string, time int64) *model.PostList {
|
|
||||||
res, err := m.Client.GetPostsSince(channelId, time)
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return res.Data.(*model.PostList)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) SearchPosts(query string) *model.PostList {
|
|
||||||
res, err := m.Client.SearchPosts(query, false)
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return res.Data.(*model.PostList)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetPosts(channelId string, limit int) *model.PostList {
|
|
||||||
res, err := m.Client.GetPosts(channelId, 0, limit, "")
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return res.Data.(*model.PostList)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetPublicLink(filename string) string {
|
|
||||||
res, err := m.Client.GetPublicLink(filename)
|
|
||||||
if err != nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
return res.Data.(string)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetPublicLinks(filenames []string) []string {
|
|
||||||
var output []string
|
|
||||||
for _, f := range filenames {
|
|
||||||
res, err := m.Client.GetPublicLink(f)
|
|
||||||
if err != nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
output = append(output, res.Data.(string))
|
|
||||||
}
|
|
||||||
return output
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UpdateChannelHeader(channelId string, header string) {
|
|
||||||
data := make(map[string]string)
|
|
||||||
data["channel_id"] = channelId
|
|
||||||
data["channel_header"] = header
|
|
||||||
m.log.Debugf("updating channelheader %#v, %#v", channelId, header)
|
|
||||||
_, err := m.Client.UpdateChannelHeader(data)
|
|
||||||
if err != nil {
|
|
||||||
log.Error(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UpdateLastViewed(channelId string) {
|
|
||||||
m.log.Debugf("posting lastview %#v", channelId)
|
|
||||||
_, err := m.Client.UpdateLastViewedAt(channelId)
|
|
||||||
if err != nil {
|
|
||||||
m.log.Error(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) UsernamesInChannel(channelName string) []string {
|
|
||||||
ceiRes, err := m.Client.GetChannelExtraInfo(m.GetChannelId(channelName), 5000, "")
|
|
||||||
if err != nil {
|
|
||||||
m.log.Errorf("UsernamesInChannel(%s) failed: %s", channelName, err)
|
|
||||||
return []string{}
|
|
||||||
}
|
|
||||||
extra := ceiRes.Data.(*model.ChannelExtra)
|
|
||||||
result := []string{}
|
|
||||||
for _, member := range extra.Members {
|
|
||||||
result = append(result, member.Username)
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) createCookieJar(token string) *cookiejar.Jar {
|
|
||||||
var cookies []*http.Cookie
|
|
||||||
jar, _ := cookiejar.New(nil)
|
|
||||||
firstCookie := &http.Cookie{
|
|
||||||
Name: "MMAUTHTOKEN",
|
|
||||||
Value: token,
|
|
||||||
Path: "/",
|
|
||||||
Domain: m.Credentials.Server,
|
|
||||||
}
|
|
||||||
cookies = append(cookies, firstCookie)
|
|
||||||
cookieURL, _ := url.Parse("https://" + m.Credentials.Server)
|
|
||||||
jar.SetCookies(cookieURL, cookies)
|
|
||||||
return jar
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) GetOtherUserDM(channel string) *model.User {
|
|
||||||
m.UpdateUsers()
|
|
||||||
var rcvuser *model.User
|
|
||||||
if strings.Contains(channel, "__") {
|
|
||||||
rcvusers := strings.Split(channel, "__")
|
|
||||||
if rcvusers[0] != m.User.Id {
|
|
||||||
rcvuser = m.Users[rcvusers[0]]
|
|
||||||
} else {
|
|
||||||
rcvuser = m.Users[rcvusers[1]]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return rcvuser
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *MMClient) SendDirectMessage(toUserId string, msg string) {
|
|
||||||
m.log.Debugf("SendDirectMessage to %s, msg %s", toUserId, msg)
|
|
||||||
var channel string
|
|
||||||
// We don't have a DM with this user yet.
|
|
||||||
if m.GetChannelId(toUserId+"__"+m.User.Id) == "" && m.GetChannelId(m.User.Id+"__"+toUserId) == "" {
|
|
||||||
// create DM channel
|
|
||||||
_, err := m.Client.CreateDirectChannel(toUserId)
|
|
||||||
if err != nil {
|
|
||||||
m.log.Debugf("SendDirectMessage to %#v failed: %s", toUserId, err)
|
|
||||||
}
|
|
||||||
// update our channels
|
|
||||||
mmchannels, _ := m.Client.GetChannels("")
|
|
||||||
m.Channels = mmchannels.Data.(*model.ChannelList)
|
|
||||||
}
|
|
||||||
|
|
||||||
// build the channel name
|
|
||||||
if toUserId > m.User.Id {
|
|
||||||
channel = m.User.Id + "__" + toUserId
|
|
||||||
} else {
|
|
||||||
channel = toUserId + "__" + m.User.Id
|
|
||||||
}
|
|
||||||
// build & send the message
|
|
||||||
msg = strings.Replace(msg, "\r", "", -1)
|
|
||||||
post := &model.Post{ChannelId: m.GetChannelId(channel), Message: msg}
|
|
||||||
m.Client.CreatePost(post)
|
|
||||||
}
|
|
24
vendor/github.com/Baozisoftware/qrcode-terminal-go/.gitignore
generated
vendored
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
# Compiled Object files, Static and Dynamic libs (Shared Objects)
|
||||||
|
*.o
|
||||||
|
*.a
|
||||||
|
*.so
|
||||||
|
|
||||||
|
# Folders
|
||||||
|
_obj
|
||||||
|
_test
|
||||||
|
|
||||||
|
# Architecture specific extensions/prefixes
|
||||||
|
*.[568vq]
|
||||||
|
[568vq].out
|
||||||
|
|
||||||
|
*.cgo1.go
|
||||||
|
*.cgo2.c
|
||||||
|
_cgo_defun.c
|
||||||
|
_cgo_gotypes.go
|
||||||
|
_cgo_export.*
|
||||||
|
|
||||||
|
_testmain.go
|
||||||
|
|
||||||
|
*.exe
|
||||||
|
*.test
|
||||||
|
*.prof
|
29
vendor/github.com/Baozisoftware/qrcode-terminal-go/LICENSE
generated
vendored
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
BSD 3-Clause License
|
||||||
|
|
||||||
|
Copyright (c) 2017, Baozisoftware
|
||||||
|
All rights reserved.
|
||||||
|
|
||||||
|
Redistribution and use in source and binary forms, with or without
|
||||||
|
modification, are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
|
* Redistributions of source code must retain the above copyright notice, this
|
||||||
|
list of conditions and the following disclaimer.
|
||||||
|
|
||||||
|
* Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
this list of conditions and the following disclaimer in the documentation
|
||||||
|
and/or other materials provided with the distribution.
|
||||||
|
|
||||||
|
* Neither the name of the copyright holder nor the names of its
|
||||||
|
contributors may be used to endorse or promote products derived from
|
||||||
|
this software without specific prior written permission.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||||
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||||
|
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
39
vendor/github.com/Baozisoftware/qrcode-terminal-go/README.md
generated
vendored
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
|
||||||
|
# qrcode-terminal-go
|
||||||
|
QRCode terminal for golang.
|
||||||
|
|
||||||
|
# Example
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import "github.com/Baozisoftware/qrcode-terminal-go"
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
Test1()
|
||||||
|
Test2()
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test1(){
|
||||||
|
content := "Hello, 世界"
|
||||||
|
obj := qrcodeTerminal.New()
|
||||||
|
obj.Get(content).Print()
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test2(){
|
||||||
|
content := "https://github.com/Baozisoftware/qrcode-terminal-go"
|
||||||
|
obj := qrcodeTerminal.New2(qrcodeTerminal.ConsoleColors.BrightBlue,qrcodeTerminal.ConsoleColors.BrightGreen,qrcodeTerminal.QRCodeRecoveryLevels.Low)
|
||||||
|
obj.Get([]byte(content)).Print()
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
### Windows XP
|
||||||
|

|
||||||
|
### Windows 7
|
||||||
|

|
||||||
|
### Windows 10
|
||||||
|

|
||||||
|
### Ubuntu
|
||||||
|

|
||||||
|
### macOS
|
||||||
|

|
155
vendor/github.com/Baozisoftware/qrcode-terminal-go/qrcodeTerminal.go
generated
vendored
Normal file
@ -0,0 +1,155 @@
|
|||||||
|
package qrcodeTerminal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/skip2/go-qrcode"
|
||||||
|
"github.com/mattn/go-colorable"
|
||||||
|
"image/png"
|
||||||
|
nbytes "bytes"
|
||||||
|
)
|
||||||
|
|
||||||
|
type consoleColor string
|
||||||
|
type consoleColors struct {
|
||||||
|
NormalBlack consoleColor
|
||||||
|
NormalRed consoleColor
|
||||||
|
NormalGreen consoleColor
|
||||||
|
NormalYellow consoleColor
|
||||||
|
NormalBlue consoleColor
|
||||||
|
NormalMagenta consoleColor
|
||||||
|
NormalCyan consoleColor
|
||||||
|
NormalWhite consoleColor
|
||||||
|
BrightBlack consoleColor
|
||||||
|
BrightRed consoleColor
|
||||||
|
BrightGreen consoleColor
|
||||||
|
BrightYellow consoleColor
|
||||||
|
BrightBlue consoleColor
|
||||||
|
BrightMagenta consoleColor
|
||||||
|
BrightCyan consoleColor
|
||||||
|
BrightWhite consoleColor
|
||||||
|
}
|
||||||
|
type qrcodeRecoveryLevel qrcode.RecoveryLevel
|
||||||
|
type qrcodeRecoveryLevels struct {
|
||||||
|
Low qrcodeRecoveryLevel
|
||||||
|
Medium qrcodeRecoveryLevel
|
||||||
|
High qrcodeRecoveryLevel
|
||||||
|
Highest qrcodeRecoveryLevel
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
ConsoleColors consoleColors = consoleColors{
|
||||||
|
NormalBlack: "\033[38;5;0m \033[0m",
|
||||||
|
NormalRed: "\033[38;5;1m \033[0m",
|
||||||
|
NormalGreen: "\033[38;5;2m \033[0m",
|
||||||
|
NormalYellow: "\033[38;5;3m \033[0m",
|
||||||
|
NormalBlue: "\033[38;5;4m \033[0m",
|
||||||
|
NormalMagenta: "\033[38;5;5m \033[0m",
|
||||||
|
NormalCyan: "\033[38;5;6m \033[0m",
|
||||||
|
NormalWhite: "\033[38;5;7m \033[0m",
|
||||||
|
BrightBlack: "\033[48;5;0m \033[0m",
|
||||||
|
BrightRed: "\033[48;5;1m \033[0m",
|
||||||
|
BrightGreen: "\033[48;5;2m \033[0m",
|
||||||
|
BrightYellow: "\033[48;5;3m \033[0m",
|
||||||
|
BrightBlue: "\033[48;5;4m \033[0m",
|
||||||
|
BrightMagenta: "\033[48;5;5m \033[0m",
|
||||||
|
BrightCyan: "\033[48;5;6m \033[0m",
|
||||||
|
BrightWhite: "\033[48;5;7m \033[0m"}
|
||||||
|
QRCodeRecoveryLevels = qrcodeRecoveryLevels{
|
||||||
|
Low: qrcodeRecoveryLevel(qrcode.Low),
|
||||||
|
Medium: qrcodeRecoveryLevel(qrcode.Medium),
|
||||||
|
High: qrcodeRecoveryLevel(qrcode.High),
|
||||||
|
Highest: qrcodeRecoveryLevel(qrcode.Highest)}
|
||||||
|
)
|
||||||
|
|
||||||
|
type QRCodeString string
|
||||||
|
|
||||||
|
func (v *QRCodeString) Print() {
|
||||||
|
fmt.Fprint(outer, *v)
|
||||||
|
}
|
||||||
|
|
||||||
|
type qrcodeTerminal struct {
|
||||||
|
front consoleColor
|
||||||
|
back consoleColor
|
||||||
|
level qrcodeRecoveryLevel
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *qrcodeTerminal) Get(content interface{}) (result *QRCodeString) {
|
||||||
|
var qr *qrcode.QRCode
|
||||||
|
var err error
|
||||||
|
if t, ok := content.(string); ok {
|
||||||
|
qr, err = qrcode.New(t, qrcode.RecoveryLevel(v.level))
|
||||||
|
} else if t, ok := content.([]byte); ok {
|
||||||
|
qr, err = qrcode.New(string(t), qrcode.RecoveryLevel(v.level))
|
||||||
|
}
|
||||||
|
if qr != nil && err == nil {
|
||||||
|
data := qr.Bitmap()
|
||||||
|
result = v.getQRCodeString(data)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *qrcodeTerminal) Get2(bytes []byte) (result *QRCodeString) {
|
||||||
|
data, err := parseQR(bytes)
|
||||||
|
if err == nil {
|
||||||
|
result = v.getQRCodeString(data)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func New2(front, back consoleColor, level qrcodeRecoveryLevel) *qrcodeTerminal {
|
||||||
|
obj := qrcodeTerminal{front: front, back: back, level: level}
|
||||||
|
return &obj
|
||||||
|
}
|
||||||
|
|
||||||
|
func New() *qrcodeTerminal {
|
||||||
|
front, back, level := ConsoleColors.BrightBlack, ConsoleColors.BrightWhite, QRCodeRecoveryLevels.Medium
|
||||||
|
return New2(front, back, level)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *qrcodeTerminal) getQRCodeString(data [][]bool) (result *QRCodeString) {
|
||||||
|
str := ""
|
||||||
|
for ir, row := range data {
|
||||||
|
lr := len(row)
|
||||||
|
if ir == 0 || ir == 1 || ir == 2 ||
|
||||||
|
ir == lr-1 || ir == lr-2 || ir == lr-3 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
for ic, col := range row {
|
||||||
|
lc := len(data)
|
||||||
|
if ic == 0 || ic == 1 || ic == 2 ||
|
||||||
|
ic == lc-1 || ic == lc-2 || ic == lc-3 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if col {
|
||||||
|
str += fmt.Sprint(v.front)
|
||||||
|
} else {
|
||||||
|
str += fmt.Sprint(v.back)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
str += fmt.Sprintln()
|
||||||
|
}
|
||||||
|
obj := QRCodeString(str)
|
||||||
|
result = &obj
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseQR(bytes []byte) (data [][]bool, err error) {
|
||||||
|
r := nbytes.NewReader(bytes)
|
||||||
|
img, err := png.Decode(r)
|
||||||
|
if err == nil {
|
||||||
|
rect := img.Bounds()
|
||||||
|
mx, my := rect.Max.X, rect.Max.Y
|
||||||
|
data = make([][]bool, mx)
|
||||||
|
for x := 0; x < mx; x++ {
|
||||||
|
data[x] = make([]bool, my)
|
||||||
|
for y := 0; y < my; y++ {
|
||||||
|
c := img.At(x, y)
|
||||||
|
r, _, _, _ := c.RGBA()
|
||||||
|
data[x][y] = r == 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var outer = colorable.NewColorableStdout()
|
14
vendor/github.com/BurntSushi/toml/COPYING
generated
vendored
@ -1,14 +0,0 @@
|
|||||||
DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
|
|
||||||
Version 2, December 2004
|
|
||||||
|
|
||||||
Copyright (C) 2004 Sam Hocevar <sam@hocevar.net>
|
|
||||||
|
|
||||||
Everyone is permitted to copy and distribute verbatim or modified
|
|
||||||
copies of this license document, and changing it is allowed as long
|
|
||||||
as the name is changed.
|
|
||||||
|
|
||||||
DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
|
|
||||||
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
|
||||||
|
|
||||||
0. You just DO WHAT THE FUCK YOU WANT TO.
|
|
||||||
|
|
90
vendor/github.com/BurntSushi/toml/cmd/toml-test-decoder/main.go
generated
vendored
@ -1,90 +0,0 @@
|
|||||||
// Command toml-test-decoder satisfies the toml-test interface for testing
|
|
||||||
// TOML decoders. Namely, it accepts TOML on stdin and outputs JSON on stdout.
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"flag"
|
|
||||||
"fmt"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
"path"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/BurntSushi/toml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
log.SetFlags(0)
|
|
||||||
|
|
||||||
flag.Usage = usage
|
|
||||||
flag.Parse()
|
|
||||||
}
|
|
||||||
|
|
||||||
func usage() {
|
|
||||||
log.Printf("Usage: %s < toml-file\n", path.Base(os.Args[0]))
|
|
||||||
flag.PrintDefaults()
|
|
||||||
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
if flag.NArg() != 0 {
|
|
||||||
flag.Usage()
|
|
||||||
}
|
|
||||||
|
|
||||||
var tmp interface{}
|
|
||||||
if _, err := toml.DecodeReader(os.Stdin, &tmp); err != nil {
|
|
||||||
log.Fatalf("Error decoding TOML: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
typedTmp := translate(tmp)
|
|
||||||
if err := json.NewEncoder(os.Stdout).Encode(typedTmp); err != nil {
|
|
||||||
log.Fatalf("Error encoding JSON: %s", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func translate(tomlData interface{}) interface{} {
|
|
||||||
switch orig := tomlData.(type) {
|
|
||||||
case map[string]interface{}:
|
|
||||||
typed := make(map[string]interface{}, len(orig))
|
|
||||||
for k, v := range orig {
|
|
||||||
typed[k] = translate(v)
|
|
||||||
}
|
|
||||||
return typed
|
|
||||||
case []map[string]interface{}:
|
|
||||||
typed := make([]map[string]interface{}, len(orig))
|
|
||||||
for i, v := range orig {
|
|
||||||
typed[i] = translate(v).(map[string]interface{})
|
|
||||||
}
|
|
||||||
return typed
|
|
||||||
case []interface{}:
|
|
||||||
typed := make([]interface{}, len(orig))
|
|
||||||
for i, v := range orig {
|
|
||||||
typed[i] = translate(v)
|
|
||||||
}
|
|
||||||
|
|
||||||
// We don't really need to tag arrays, but let's be future proof.
|
|
||||||
// (If TOML ever supports tuples, we'll need this.)
|
|
||||||
return tag("array", typed)
|
|
||||||
case time.Time:
|
|
||||||
return tag("datetime", orig.Format("2006-01-02T15:04:05Z"))
|
|
||||||
case bool:
|
|
||||||
return tag("bool", fmt.Sprintf("%v", orig))
|
|
||||||
case int64:
|
|
||||||
return tag("integer", fmt.Sprintf("%d", orig))
|
|
||||||
case float64:
|
|
||||||
return tag("float", fmt.Sprintf("%v", orig))
|
|
||||||
case string:
|
|
||||||
return tag("string", orig)
|
|
||||||
}
|
|
||||||
|
|
||||||
panic(fmt.Sprintf("Unknown type: %T", tomlData))
|
|
||||||
}
|
|
||||||
|
|
||||||
func tag(typeName string, data interface{}) map[string]interface{} {
|
|
||||||
return map[string]interface{}{
|
|
||||||
"type": typeName,
|
|
||||||
"value": data,
|
|
||||||
}
|
|
||||||
}
|
|
131
vendor/github.com/BurntSushi/toml/cmd/toml-test-encoder/main.go
generated
vendored
@ -1,131 +0,0 @@
|
|||||||
// Command toml-test-encoder satisfies the toml-test interface for testing
|
|
||||||
// TOML encoders. Namely, it accepts JSON on stdin and outputs TOML on stdout.
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"flag"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
"path"
|
|
||||||
"strconv"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/BurntSushi/toml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
log.SetFlags(0)
|
|
||||||
|
|
||||||
flag.Usage = usage
|
|
||||||
flag.Parse()
|
|
||||||
}
|
|
||||||
|
|
||||||
func usage() {
|
|
||||||
log.Printf("Usage: %s < json-file\n", path.Base(os.Args[0]))
|
|
||||||
flag.PrintDefaults()
|
|
||||||
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
if flag.NArg() != 0 {
|
|
||||||
flag.Usage()
|
|
||||||
}
|
|
||||||
|
|
||||||
var tmp interface{}
|
|
||||||
if err := json.NewDecoder(os.Stdin).Decode(&tmp); err != nil {
|
|
||||||
log.Fatalf("Error decoding JSON: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
tomlData := translate(tmp)
|
|
||||||
if err := toml.NewEncoder(os.Stdout).Encode(tomlData); err != nil {
|
|
||||||
log.Fatalf("Error encoding TOML: %s", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func translate(typedJson interface{}) interface{} {
|
|
||||||
switch v := typedJson.(type) {
|
|
||||||
case map[string]interface{}:
|
|
||||||
if len(v) == 2 && in("type", v) && in("value", v) {
|
|
||||||
return untag(v)
|
|
||||||
}
|
|
||||||
m := make(map[string]interface{}, len(v))
|
|
||||||
for k, v2 := range v {
|
|
||||||
m[k] = translate(v2)
|
|
||||||
}
|
|
||||||
return m
|
|
||||||
case []interface{}:
|
|
||||||
tabArray := make([]map[string]interface{}, len(v))
|
|
||||||
for i := range v {
|
|
||||||
if m, ok := translate(v[i]).(map[string]interface{}); ok {
|
|
||||||
tabArray[i] = m
|
|
||||||
} else {
|
|
||||||
log.Fatalf("JSON arrays may only contain objects. This " +
|
|
||||||
"corresponds to only tables being allowed in " +
|
|
||||||
"TOML table arrays.")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return tabArray
|
|
||||||
}
|
|
||||||
log.Fatalf("Unrecognized JSON format '%T'.", typedJson)
|
|
||||||
panic("unreachable")
|
|
||||||
}
|
|
||||||
|
|
||||||
func untag(typed map[string]interface{}) interface{} {
|
|
||||||
t := typed["type"].(string)
|
|
||||||
v := typed["value"]
|
|
||||||
switch t {
|
|
||||||
case "string":
|
|
||||||
return v.(string)
|
|
||||||
case "integer":
|
|
||||||
v := v.(string)
|
|
||||||
n, err := strconv.Atoi(v)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Could not parse '%s' as integer: %s", v, err)
|
|
||||||
}
|
|
||||||
return n
|
|
||||||
case "float":
|
|
||||||
v := v.(string)
|
|
||||||
f, err := strconv.ParseFloat(v, 64)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Could not parse '%s' as float64: %s", v, err)
|
|
||||||
}
|
|
||||||
return f
|
|
||||||
case "datetime":
|
|
||||||
v := v.(string)
|
|
||||||
t, err := time.Parse("2006-01-02T15:04:05Z", v)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Could not parse '%s' as a datetime: %s", v, err)
|
|
||||||
}
|
|
||||||
return t
|
|
||||||
case "bool":
|
|
||||||
v := v.(string)
|
|
||||||
switch v {
|
|
||||||
case "true":
|
|
||||||
return true
|
|
||||||
case "false":
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
log.Fatalf("Could not parse '%s' as a boolean.", v)
|
|
||||||
case "array":
|
|
||||||
v := v.([]interface{})
|
|
||||||
array := make([]interface{}, len(v))
|
|
||||||
for i := range v {
|
|
||||||
if m, ok := v[i].(map[string]interface{}); ok {
|
|
||||||
array[i] = untag(m)
|
|
||||||
} else {
|
|
||||||
log.Fatalf("Arrays may only contain other arrays or "+
|
|
||||||
"primitive values, but found a '%T'.", m)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return array
|
|
||||||
}
|
|
||||||
log.Fatalf("Unrecognized tag type '%s'.", t)
|
|
||||||
panic("unreachable")
|
|
||||||
}
|
|
||||||
|
|
||||||
func in(key string, m map[string]interface{}) bool {
|
|
||||||
_, ok := m[key]
|
|
||||||
return ok
|
|
||||||
}
|
|
61
vendor/github.com/BurntSushi/toml/cmd/tomlv/main.go
generated
vendored
@ -1,61 +0,0 @@
|
|||||||
// Command tomlv validates TOML documents and prints each key's type.
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"flag"
|
|
||||||
"fmt"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
"path"
|
|
||||||
"strings"
|
|
||||||
"text/tabwriter"
|
|
||||||
|
|
||||||
"github.com/BurntSushi/toml"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
flagTypes = false
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
log.SetFlags(0)
|
|
||||||
|
|
||||||
flag.BoolVar(&flagTypes, "types", flagTypes,
|
|
||||||
"When set, the types of every defined key will be shown.")
|
|
||||||
|
|
||||||
flag.Usage = usage
|
|
||||||
flag.Parse()
|
|
||||||
}
|
|
||||||
|
|
||||||
func usage() {
|
|
||||||
log.Printf("Usage: %s toml-file [ toml-file ... ]\n",
|
|
||||||
path.Base(os.Args[0]))
|
|
||||||
flag.PrintDefaults()
|
|
||||||
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
if flag.NArg() < 1 {
|
|
||||||
flag.Usage()
|
|
||||||
}
|
|
||||||
for _, f := range flag.Args() {
|
|
||||||
var tmp interface{}
|
|
||||||
md, err := toml.DecodeFile(f, &tmp)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Error in '%s': %s", f, err)
|
|
||||||
}
|
|
||||||
if flagTypes {
|
|
||||||
printTypes(md)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func printTypes(md toml.MetaData) {
|
|
||||||
tabw := tabwriter.NewWriter(os.Stdout, 0, 0, 2, ' ', 0)
|
|
||||||
for _, key := range md.Keys() {
|
|
||||||
fmt.Fprintf(tabw, "%s%s\t%s\n",
|
|
||||||
strings.Repeat(" ", len(key)-1), key, md.Type(key...))
|
|
||||||
}
|
|
||||||
tabw.Flush()
|
|
||||||
}
|
|
509
vendor/github.com/BurntSushi/toml/decode.go
generated
vendored
@ -1,509 +0,0 @@
|
|||||||
package toml
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"io/ioutil"
|
|
||||||
"math"
|
|
||||||
"reflect"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
func e(format string, args ...interface{}) error {
|
|
||||||
return fmt.Errorf("toml: "+format, args...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unmarshaler is the interface implemented by objects that can unmarshal a
|
|
||||||
// TOML description of themselves.
|
|
||||||
type Unmarshaler interface {
|
|
||||||
UnmarshalTOML(interface{}) error
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unmarshal decodes the contents of `p` in TOML format into a pointer `v`.
|
|
||||||
func Unmarshal(p []byte, v interface{}) error {
|
|
||||||
_, err := Decode(string(p), v)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Primitive is a TOML value that hasn't been decoded into a Go value.
|
|
||||||
// When using the various `Decode*` functions, the type `Primitive` may
|
|
||||||
// be given to any value, and its decoding will be delayed.
|
|
||||||
//
|
|
||||||
// A `Primitive` value can be decoded using the `PrimitiveDecode` function.
|
|
||||||
//
|
|
||||||
// The underlying representation of a `Primitive` value is subject to change.
|
|
||||||
// Do not rely on it.
|
|
||||||
//
|
|
||||||
// N.B. Primitive values are still parsed, so using them will only avoid
|
|
||||||
// the overhead of reflection. They can be useful when you don't know the
|
|
||||||
// exact type of TOML data until run time.
|
|
||||||
type Primitive struct {
|
|
||||||
undecoded interface{}
|
|
||||||
context Key
|
|
||||||
}
|
|
||||||
|
|
||||||
// DEPRECATED!
|
|
||||||
//
|
|
||||||
// Use MetaData.PrimitiveDecode instead.
|
|
||||||
func PrimitiveDecode(primValue Primitive, v interface{}) error {
|
|
||||||
md := MetaData{decoded: make(map[string]bool)}
|
|
||||||
return md.unify(primValue.undecoded, rvalue(v))
|
|
||||||
}
|
|
||||||
|
|
||||||
// PrimitiveDecode is just like the other `Decode*` functions, except it
|
|
||||||
// decodes a TOML value that has already been parsed. Valid primitive values
|
|
||||||
// can *only* be obtained from values filled by the decoder functions,
|
|
||||||
// including this method. (i.e., `v` may contain more `Primitive`
|
|
||||||
// values.)
|
|
||||||
//
|
|
||||||
// Meta data for primitive values is included in the meta data returned by
|
|
||||||
// the `Decode*` functions with one exception: keys returned by the Undecoded
|
|
||||||
// method will only reflect keys that were decoded. Namely, any keys hidden
|
|
||||||
// behind a Primitive will be considered undecoded. Executing this method will
|
|
||||||
// update the undecoded keys in the meta data. (See the example.)
|
|
||||||
func (md *MetaData) PrimitiveDecode(primValue Primitive, v interface{}) error {
|
|
||||||
md.context = primValue.context
|
|
||||||
defer func() { md.context = nil }()
|
|
||||||
return md.unify(primValue.undecoded, rvalue(v))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Decode will decode the contents of `data` in TOML format into a pointer
|
|
||||||
// `v`.
|
|
||||||
//
|
|
||||||
// TOML hashes correspond to Go structs or maps. (Dealer's choice. They can be
|
|
||||||
// used interchangeably.)
|
|
||||||
//
|
|
||||||
// TOML arrays of tables correspond to either a slice of structs or a slice
|
|
||||||
// of maps.
|
|
||||||
//
|
|
||||||
// TOML datetimes correspond to Go `time.Time` values.
|
|
||||||
//
|
|
||||||
// All other TOML types (float, string, int, bool and array) correspond
|
|
||||||
// to the obvious Go types.
|
|
||||||
//
|
|
||||||
// An exception to the above rules is if a type implements the
|
|
||||||
// encoding.TextUnmarshaler interface. In this case, any primitive TOML value
|
|
||||||
// (floats, strings, integers, booleans and datetimes) will be converted to
|
|
||||||
// a byte string and given to the value's UnmarshalText method. See the
|
|
||||||
// Unmarshaler example for a demonstration with time duration strings.
|
|
||||||
//
|
|
||||||
// Key mapping
|
|
||||||
//
|
|
||||||
// TOML keys can map to either keys in a Go map or field names in a Go
|
|
||||||
// struct. The special `toml` struct tag may be used to map TOML keys to
|
|
||||||
// struct fields that don't match the key name exactly. (See the example.)
|
|
||||||
// A case insensitive match to struct names will be tried if an exact match
|
|
||||||
// can't be found.
|
|
||||||
//
|
|
||||||
// The mapping between TOML values and Go values is loose. That is, there
|
|
||||||
// may exist TOML values that cannot be placed into your representation, and
|
|
||||||
// there may be parts of your representation that do not correspond to
|
|
||||||
// TOML values. This loose mapping can be made stricter by using the IsDefined
|
|
||||||
// and/or Undecoded methods on the MetaData returned.
|
|
||||||
//
|
|
||||||
// This decoder will not handle cyclic types. If a cyclic type is passed,
|
|
||||||
// `Decode` will not terminate.
|
|
||||||
func Decode(data string, v interface{}) (MetaData, error) {
|
|
||||||
rv := reflect.ValueOf(v)
|
|
||||||
if rv.Kind() != reflect.Ptr {
|
|
||||||
return MetaData{}, e("Decode of non-pointer %s", reflect.TypeOf(v))
|
|
||||||
}
|
|
||||||
if rv.IsNil() {
|
|
||||||
return MetaData{}, e("Decode of nil %s", reflect.TypeOf(v))
|
|
||||||
}
|
|
||||||
p, err := parse(data)
|
|
||||||
if err != nil {
|
|
||||||
return MetaData{}, err
|
|
||||||
}
|
|
||||||
md := MetaData{
|
|
||||||
p.mapping, p.types, p.ordered,
|
|
||||||
make(map[string]bool, len(p.ordered)), nil,
|
|
||||||
}
|
|
||||||
return md, md.unify(p.mapping, indirect(rv))
|
|
||||||
}
|
|
||||||
|
|
||||||
// DecodeFile is just like Decode, except it will automatically read the
|
|
||||||
// contents of the file at `fpath` and decode it for you.
|
|
||||||
func DecodeFile(fpath string, v interface{}) (MetaData, error) {
|
|
||||||
bs, err := ioutil.ReadFile(fpath)
|
|
||||||
if err != nil {
|
|
||||||
return MetaData{}, err
|
|
||||||
}
|
|
||||||
return Decode(string(bs), v)
|
|
||||||
}
|
|
||||||
|
|
||||||
// DecodeReader is just like Decode, except it will consume all bytes
|
|
||||||
// from the reader and decode it for you.
|
|
||||||
func DecodeReader(r io.Reader, v interface{}) (MetaData, error) {
|
|
||||||
bs, err := ioutil.ReadAll(r)
|
|
||||||
if err != nil {
|
|
||||||
return MetaData{}, err
|
|
||||||
}
|
|
||||||
return Decode(string(bs), v)
|
|
||||||
}
|
|
||||||
|
|
||||||
// unify performs a sort of type unification based on the structure of `rv`,
|
|
||||||
// which is the client representation.
|
|
||||||
//
|
|
||||||
// Any type mismatch produces an error. Finding a type that we don't know
|
|
||||||
// how to handle produces an unsupported type error.
|
|
||||||
func (md *MetaData) unify(data interface{}, rv reflect.Value) error {
|
|
||||||
|
|
||||||
// Special case. Look for a `Primitive` value.
|
|
||||||
if rv.Type() == reflect.TypeOf((*Primitive)(nil)).Elem() {
|
|
||||||
// Save the undecoded data and the key context into the primitive
|
|
||||||
// value.
|
|
||||||
context := make(Key, len(md.context))
|
|
||||||
copy(context, md.context)
|
|
||||||
rv.Set(reflect.ValueOf(Primitive{
|
|
||||||
undecoded: data,
|
|
||||||
context: context,
|
|
||||||
}))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Special case. Unmarshaler Interface support.
|
|
||||||
if rv.CanAddr() {
|
|
||||||
if v, ok := rv.Addr().Interface().(Unmarshaler); ok {
|
|
||||||
return v.UnmarshalTOML(data)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Special case. Handle time.Time values specifically.
|
|
||||||
// TODO: Remove this code when we decide to drop support for Go 1.1.
|
|
||||||
// This isn't necessary in Go 1.2 because time.Time satisfies the encoding
|
|
||||||
// interfaces.
|
|
||||||
if rv.Type().AssignableTo(rvalue(time.Time{}).Type()) {
|
|
||||||
return md.unifyDatetime(data, rv)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Special case. Look for a value satisfying the TextUnmarshaler interface.
|
|
||||||
if v, ok := rv.Interface().(TextUnmarshaler); ok {
|
|
||||||
return md.unifyText(data, v)
|
|
||||||
}
|
|
||||||
// BUG(burntsushi)
|
|
||||||
// The behavior here is incorrect whenever a Go type satisfies the
|
|
||||||
// encoding.TextUnmarshaler interface but also corresponds to a TOML
|
|
||||||
// hash or array. In particular, the unmarshaler should only be applied
|
|
||||||
// to primitive TOML values. But at this point, it will be applied to
|
|
||||||
// all kinds of values and produce an incorrect error whenever those values
|
|
||||||
// are hashes or arrays (including arrays of tables).
|
|
||||||
|
|
||||||
k := rv.Kind()
|
|
||||||
|
|
||||||
// laziness
|
|
||||||
if k >= reflect.Int && k <= reflect.Uint64 {
|
|
||||||
return md.unifyInt(data, rv)
|
|
||||||
}
|
|
||||||
switch k {
|
|
||||||
case reflect.Ptr:
|
|
||||||
elem := reflect.New(rv.Type().Elem())
|
|
||||||
err := md.unify(data, reflect.Indirect(elem))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
rv.Set(elem)
|
|
||||||
return nil
|
|
||||||
case reflect.Struct:
|
|
||||||
return md.unifyStruct(data, rv)
|
|
||||||
case reflect.Map:
|
|
||||||
return md.unifyMap(data, rv)
|
|
||||||
case reflect.Array:
|
|
||||||
return md.unifyArray(data, rv)
|
|
||||||
case reflect.Slice:
|
|
||||||
return md.unifySlice(data, rv)
|
|
||||||
case reflect.String:
|
|
||||||
return md.unifyString(data, rv)
|
|
||||||
case reflect.Bool:
|
|
||||||
return md.unifyBool(data, rv)
|
|
||||||
case reflect.Interface:
|
|
||||||
// we only support empty interfaces.
|
|
||||||
if rv.NumMethod() > 0 {
|
|
||||||
return e("unsupported type %s", rv.Type())
|
|
||||||
}
|
|
||||||
return md.unifyAnything(data, rv)
|
|
||||||
case reflect.Float32:
|
|
||||||
fallthrough
|
|
||||||
case reflect.Float64:
|
|
||||||
return md.unifyFloat64(data, rv)
|
|
||||||
}
|
|
||||||
return e("unsupported type %s", rv.Kind())
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyStruct(mapping interface{}, rv reflect.Value) error {
|
|
||||||
tmap, ok := mapping.(map[string]interface{})
|
|
||||||
if !ok {
|
|
||||||
if mapping == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return e("type mismatch for %s: expected table but found %T",
|
|
||||||
rv.Type().String(), mapping)
|
|
||||||
}
|
|
||||||
|
|
||||||
for key, datum := range tmap {
|
|
||||||
var f *field
|
|
||||||
fields := cachedTypeFields(rv.Type())
|
|
||||||
for i := range fields {
|
|
||||||
ff := &fields[i]
|
|
||||||
if ff.name == key {
|
|
||||||
f = ff
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if f == nil && strings.EqualFold(ff.name, key) {
|
|
||||||
f = ff
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if f != nil {
|
|
||||||
subv := rv
|
|
||||||
for _, i := range f.index {
|
|
||||||
subv = indirect(subv.Field(i))
|
|
||||||
}
|
|
||||||
if isUnifiable(subv) {
|
|
||||||
md.decoded[md.context.add(key).String()] = true
|
|
||||||
md.context = append(md.context, key)
|
|
||||||
if err := md.unify(datum, subv); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
md.context = md.context[0 : len(md.context)-1]
|
|
||||||
} else if f.name != "" {
|
|
||||||
// Bad user! No soup for you!
|
|
||||||
return e("cannot write unexported field %s.%s",
|
|
||||||
rv.Type().String(), f.name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyMap(mapping interface{}, rv reflect.Value) error {
|
|
||||||
tmap, ok := mapping.(map[string]interface{})
|
|
||||||
if !ok {
|
|
||||||
if tmap == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return badtype("map", mapping)
|
|
||||||
}
|
|
||||||
if rv.IsNil() {
|
|
||||||
rv.Set(reflect.MakeMap(rv.Type()))
|
|
||||||
}
|
|
||||||
for k, v := range tmap {
|
|
||||||
md.decoded[md.context.add(k).String()] = true
|
|
||||||
md.context = append(md.context, k)
|
|
||||||
|
|
||||||
rvkey := indirect(reflect.New(rv.Type().Key()))
|
|
||||||
rvval := reflect.Indirect(reflect.New(rv.Type().Elem()))
|
|
||||||
if err := md.unify(v, rvval); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
md.context = md.context[0 : len(md.context)-1]
|
|
||||||
|
|
||||||
rvkey.SetString(k)
|
|
||||||
rv.SetMapIndex(rvkey, rvval)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyArray(data interface{}, rv reflect.Value) error {
|
|
||||||
datav := reflect.ValueOf(data)
|
|
||||||
if datav.Kind() != reflect.Slice {
|
|
||||||
if !datav.IsValid() {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return badtype("slice", data)
|
|
||||||
}
|
|
||||||
sliceLen := datav.Len()
|
|
||||||
if sliceLen != rv.Len() {
|
|
||||||
return e("expected array length %d; got TOML array of length %d",
|
|
||||||
rv.Len(), sliceLen)
|
|
||||||
}
|
|
||||||
return md.unifySliceArray(datav, rv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifySlice(data interface{}, rv reflect.Value) error {
|
|
||||||
datav := reflect.ValueOf(data)
|
|
||||||
if datav.Kind() != reflect.Slice {
|
|
||||||
if !datav.IsValid() {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return badtype("slice", data)
|
|
||||||
}
|
|
||||||
n := datav.Len()
|
|
||||||
if rv.IsNil() || rv.Cap() < n {
|
|
||||||
rv.Set(reflect.MakeSlice(rv.Type(), n, n))
|
|
||||||
}
|
|
||||||
rv.SetLen(n)
|
|
||||||
return md.unifySliceArray(datav, rv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifySliceArray(data, rv reflect.Value) error {
|
|
||||||
sliceLen := data.Len()
|
|
||||||
for i := 0; i < sliceLen; i++ {
|
|
||||||
v := data.Index(i).Interface()
|
|
||||||
sliceval := indirect(rv.Index(i))
|
|
||||||
if err := md.unify(v, sliceval); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyDatetime(data interface{}, rv reflect.Value) error {
|
|
||||||
if _, ok := data.(time.Time); ok {
|
|
||||||
rv.Set(reflect.ValueOf(data))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return badtype("time.Time", data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyString(data interface{}, rv reflect.Value) error {
|
|
||||||
if s, ok := data.(string); ok {
|
|
||||||
rv.SetString(s)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return badtype("string", data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyFloat64(data interface{}, rv reflect.Value) error {
|
|
||||||
if num, ok := data.(float64); ok {
|
|
||||||
switch rv.Kind() {
|
|
||||||
case reflect.Float32:
|
|
||||||
fallthrough
|
|
||||||
case reflect.Float64:
|
|
||||||
rv.SetFloat(num)
|
|
||||||
default:
|
|
||||||
panic("bug")
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return badtype("float", data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyInt(data interface{}, rv reflect.Value) error {
|
|
||||||
if num, ok := data.(int64); ok {
|
|
||||||
if rv.Kind() >= reflect.Int && rv.Kind() <= reflect.Int64 {
|
|
||||||
switch rv.Kind() {
|
|
||||||
case reflect.Int, reflect.Int64:
|
|
||||||
// No bounds checking necessary.
|
|
||||||
case reflect.Int8:
|
|
||||||
if num < math.MinInt8 || num > math.MaxInt8 {
|
|
||||||
return e("value %d is out of range for int8", num)
|
|
||||||
}
|
|
||||||
case reflect.Int16:
|
|
||||||
if num < math.MinInt16 || num > math.MaxInt16 {
|
|
||||||
return e("value %d is out of range for int16", num)
|
|
||||||
}
|
|
||||||
case reflect.Int32:
|
|
||||||
if num < math.MinInt32 || num > math.MaxInt32 {
|
|
||||||
return e("value %d is out of range for int32", num)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
rv.SetInt(num)
|
|
||||||
} else if rv.Kind() >= reflect.Uint && rv.Kind() <= reflect.Uint64 {
|
|
||||||
unum := uint64(num)
|
|
||||||
switch rv.Kind() {
|
|
||||||
case reflect.Uint, reflect.Uint64:
|
|
||||||
// No bounds checking necessary.
|
|
||||||
case reflect.Uint8:
|
|
||||||
if num < 0 || unum > math.MaxUint8 {
|
|
||||||
return e("value %d is out of range for uint8", num)
|
|
||||||
}
|
|
||||||
case reflect.Uint16:
|
|
||||||
if num < 0 || unum > math.MaxUint16 {
|
|
||||||
return e("value %d is out of range for uint16", num)
|
|
||||||
}
|
|
||||||
case reflect.Uint32:
|
|
||||||
if num < 0 || unum > math.MaxUint32 {
|
|
||||||
return e("value %d is out of range for uint32", num)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
rv.SetUint(unum)
|
|
||||||
} else {
|
|
||||||
panic("unreachable")
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return badtype("integer", data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyBool(data interface{}, rv reflect.Value) error {
|
|
||||||
if b, ok := data.(bool); ok {
|
|
||||||
rv.SetBool(b)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return badtype("boolean", data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyAnything(data interface{}, rv reflect.Value) error {
|
|
||||||
rv.Set(reflect.ValueOf(data))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (md *MetaData) unifyText(data interface{}, v TextUnmarshaler) error {
|
|
||||||
var s string
|
|
||||||
switch sdata := data.(type) {
|
|
||||||
case TextMarshaler:
|
|
||||||
text, err := sdata.MarshalText()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
s = string(text)
|
|
||||||
case fmt.Stringer:
|
|
||||||
s = sdata.String()
|
|
||||||
case string:
|
|
||||||
s = sdata
|
|
||||||
case bool:
|
|
||||||
s = fmt.Sprintf("%v", sdata)
|
|
||||||
case int64:
|
|
||||||
s = fmt.Sprintf("%d", sdata)
|
|
||||||
case float64:
|
|
||||||
s = fmt.Sprintf("%f", sdata)
|
|
||||||
default:
|
|
||||||
return badtype("primitive (string-like)", data)
|
|
||||||
}
|
|
||||||
if err := v.UnmarshalText([]byte(s)); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// rvalue returns a reflect.Value of `v`. All pointers are resolved.
|
|
||||||
func rvalue(v interface{}) reflect.Value {
|
|
||||||
return indirect(reflect.ValueOf(v))
|
|
||||||
}
|
|
||||||
|
|
||||||
// indirect returns the value pointed to by a pointer.
|
|
||||||
// Pointers are followed until the value is not a pointer.
|
|
||||||
// New values are allocated for each nil pointer.
|
|
||||||
//
|
|
||||||
// An exception to this rule is if the value satisfies an interface of
|
|
||||||
// interest to us (like encoding.TextUnmarshaler).
|
|
||||||
func indirect(v reflect.Value) reflect.Value {
|
|
||||||
if v.Kind() != reflect.Ptr {
|
|
||||||
if v.CanSet() {
|
|
||||||
pv := v.Addr()
|
|
||||||
if _, ok := pv.Interface().(TextUnmarshaler); ok {
|
|
||||||
return pv
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return v
|
|
||||||
}
|
|
||||||
if v.IsNil() {
|
|
||||||
v.Set(reflect.New(v.Type().Elem()))
|
|
||||||
}
|
|
||||||
return indirect(reflect.Indirect(v))
|
|
||||||
}
|
|
||||||
|
|
||||||
func isUnifiable(rv reflect.Value) bool {
|
|
||||||
if rv.CanSet() {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
if _, ok := rv.Interface().(TextUnmarshaler); ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func badtype(expected string, data interface{}) error {
|
|
||||||
return e("cannot load TOML value of type %T into a Go %s", data, expected)
|
|
||||||
}
|
|