You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

maillist.lua 7.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. -- Module for checking mail list headers
  2. local symbol = 'MAILLIST'
  3. -- EZMLM
  4. -- Mailing-List: .*run by ezmlm
  5. -- Precedence: bulk
  6. -- List-Post: <mailto:
  7. -- List-Help: <mailto:
  8. -- List-Unsubscribe: <mailto:[a-zA-Z\.-]+-unsubscribe@
  9. -- List-Subscribe: <mailto:[a-zA-Z\.-]+-subscribe@
  10. function check_ml_ezmlm(task)
  11. local message = task:get_message()
  12. -- Mailing-List
  13. local header = message:get_header('mailing-list')
  14. if not header or not string.find(header[1], 'ezmlm$') then
  15. return false
  16. end
  17. -- Precedence
  18. header = message:get_header('precedence')
  19. if not header or not string.match(header[1], '^bulk$') then
  20. return false
  21. end
  22. -- Other headers
  23. header = message:get_header('list-post')
  24. if not header or not string.find(header[1], '^<mailto:') then
  25. return false
  26. end
  27. header = message:get_header('list-help')
  28. if not header or not string.find(header[1], '^<mailto:') then
  29. return false
  30. end
  31. -- Subscribe and unsubscribe
  32. header = message:get_header('list-subscribe')
  33. if not header or not string.find(header[1], '<mailto:[a-zA-Z.-]+-subscribe@') then
  34. return false
  35. end
  36. header = message:get_header('list-unsubscribe')
  37. if not header or not string.find(header[1], '<mailto:[a-zA-Z.-]+-unsubscribe@') then
  38. return false
  39. end
  40. return true
  41. end
  42. -- MailMan (the gnu mailing list manager)
  43. -- Precedence: bulk [or list for v2]
  44. -- List-Help: <mailto:
  45. -- List-Post: <mailto:
  46. -- List-Subscribe: .*<mailto:.*=subscribe>
  47. -- List-Id:
  48. -- List-Unsubscribe: .*<mailto:.*=unsubscribe>
  49. -- List-Archive:
  50. -- X-Mailman-Version: \d
  51. function check_ml_mailman(task)
  52. local message = task:get_message()
  53. -- Mailing-List
  54. local header = message:get_header('x-mailman-version')
  55. if not header or not string.find(header[1], '^%d') then
  56. return false
  57. end
  58. -- Precedence
  59. header = message:get_header('precedence')
  60. if not header or (not string.match(header[1], '^bulk$') and not string.match(header[1], '^list$')) then
  61. return false
  62. end
  63. -- For reminders we have other headers than for normal messages
  64. header = message:get_header('x-list-administrivia')
  65. local subject = message:get_header('subject')
  66. if (header and string.find(header[1], 'yes')) or (subject and string.find(subject[1], 'mailing list memberships reminder$')) then
  67. if not message:get_header('errors-to') or not message:get_header('x-beenthere') then
  68. return false
  69. end
  70. header = message:get_header('x-no-archive')
  71. if not header or not string.find(header[1], 'yes') then
  72. return false
  73. end
  74. return true
  75. end
  76. -- Other headers
  77. header = message:get_header('list-id')
  78. if not header then
  79. return false
  80. end
  81. header = message:get_header('list-post')
  82. if not header or not string.find(header[1], '^<mailto:') then
  83. return false
  84. end
  85. header = message:get_header('list-help')
  86. if not header or not string.find(header[1], '^<mailto:') then
  87. return false
  88. end
  89. -- Subscribe and unsubscribe
  90. header = message:get_header('list-subscribe')
  91. if not header or not string.find(header[1], '<mailto:.*=subscribe>') then
  92. return false
  93. end
  94. header = message:get_header('list-unsubscribe')
  95. if not header or not string.find(header[1], '<mailto:.*=unsubscribe>') then
  96. return false
  97. end
  98. return true
  99. end
  100. -- Subscribe.ru
  101. -- Precedence: normal
  102. -- List-Id: <.*.subscribe.ru>
  103. -- List-Help: <http://subscribe.ru/catalog/.*>
  104. -- List-Subscribe: <mailto:.*-sub@subscribe.ru>
  105. -- List-Unsubscribe: <mailto:.*-unsub@subscribe.ru>
  106. -- List-Archive: <http://subscribe.ru/archive/.*>
  107. -- List-Owner: <mailto:.*-owner@subscribe.ru>
  108. -- List-Post: NO
  109. function check_ml_subscriberu(task)
  110. local message = task:get_message()
  111. -- List-Id
  112. local header = message:get_header('list-id')
  113. if not header or not string.find(header[1], '^<.*%.subscribe%.ru>$') then
  114. return false
  115. end
  116. -- Precedence
  117. header = message:get_header('precedence')
  118. if not header or not string.match(header[1], '^normal$') then
  119. return false
  120. end
  121. -- Other headers
  122. header = message:get_header('list-archive')
  123. if not header or not string.find(header[1], '^<http://subscribe.ru/archive/.*>$') then
  124. return false
  125. end
  126. header = message:get_header('list-owner')
  127. if not header or not string.find(header[1], '^<mailto:.*-owner@subscribe.ru>$') then
  128. return false
  129. end
  130. header = message:get_header('list-help')
  131. if not header or not string.find(header[1], '^<http://subscribe.ru/catalog/.*>$') then
  132. return false
  133. end
  134. -- Subscribe and unsubscribe
  135. header = message:get_header('list-subscribe')
  136. if not header or not string.find(header[1], '^<mailto:.*-sub@subscribe.ru>$') then
  137. return false
  138. end
  139. header = message:get_header('list-unsubscribe')
  140. if not header or not string.find(header[1], '^<mailto:.*-unsub@subscribe.ru>$') then
  141. return false
  142. end
  143. return true
  144. end
  145. -- RFC 2369 headers
  146. function check_rfc2369(task)
  147. local message = task:get_message()
  148. local header = message:get_header('List-Id')
  149. if not header then
  150. return false
  151. end
  152. header = message:get_header('List-Unsubscribe')
  153. if not header or not string.find(header[1], '^^<.+>$') then
  154. return false
  155. end
  156. header = message:get_header('List-Subscribe')
  157. if not header or not string.find(header[1], '^^<.+>$') then
  158. return false
  159. end
  160. return true
  161. end
  162. -- RFC 2919 headers
  163. function check_rfc2919(task)
  164. local message = task:get_message()
  165. local header = message:get_header('List-Id')
  166. if not header or not string.find(header[1], '^<.+>$') then
  167. return false
  168. end
  169. return check_rfc2369(task)
  170. end
  171. -- Google groups detector
  172. -- header exists X-Google-Loop
  173. -- RFC 2919 headers exist
  174. --
  175. function check_ml_googlegroup(task)
  176. local message = task:get_message()
  177. local header = message:get_header('X-Google-Loop')
  178. if not header then
  179. return false
  180. end
  181. return check_rfc2919(task)
  182. end
  183. -- Majordomo detector
  184. -- Check Sender for owner- or -owner
  185. -- Check Precendence for 'Bulk' or 'List'
  186. --
  187. -- And nothing more can be extracted :(
  188. function check_ml_majordomo(task)
  189. local message = task:get_message()
  190. local header = message:get_header('Sender')
  191. if not header or (not string.find(header[1], '^owner-.*$') and not string.find(header[1], '^.*-owner$')) then
  192. return false
  193. end
  194. local header = message:get_header('Precedence')
  195. if not header or (header[1] ~= 'list' and header[1] ~= 'bulk') then
  196. return false
  197. end
  198. return true
  199. end
  200. -- CGP detector
  201. -- X-Listserver = CommuniGate Pro LIST
  202. -- RFC 2919 headers exist
  203. --
  204. function check_ml_cgp(task)
  205. local message = task:get_message()
  206. local header = message:get_header('X-Listserver')
  207. if not header or header ~= 'CommuniGate Pro LIST' then
  208. return false
  209. end
  210. return check_rfc2919(task)
  211. end
  212. function check_maillist(task)
  213. if check_ml_ezmlm(task) then
  214. task:insert_result(symbol, 1, 'ezmlm')
  215. elseif check_ml_mailman(task) then
  216. task:insert_result(symbol, 1, 'mailman')
  217. elseif check_ml_subscriberu(task) then
  218. task:insert_result(symbol, 1, 'subscribe.ru')
  219. elseif check_ml_googlegroup(task) then
  220. task:insert_result(symbol, 1, 'googlegroups')
  221. elseif check_ml_majordomo(task) then
  222. task:insert_result(symbol, 1, 'majordomo')
  223. elseif check_ml_cgp(task) then
  224. task:insert_result(symbol, 1, 'cgp')
  225. end
  226. end
  227. -- Registration
  228. if type(rspamd_config.get_api_version) ~= 'nil' then
  229. if rspamd_config:get_api_version() >= 1 then
  230. rspamd_config:register_module_option('maillist', 'symbol', 'string')
  231. end
  232. end
  233. -- Configuration
  234. local opts = rspamd_config:get_all_opt('maillist')if opts then
  235. if opts['symbol'] then
  236. symbol = opts['symbol']
  237. rspamd_config:register_symbol(symbol, 1.0, 'check_maillist')
  238. end
  239. end