imap_folder_agent_spec.rb 8.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. require 'rails_helper'
  2. require 'time'
  3. describe Agents::ImapFolderAgent do
  4. describe 'checking IMAP' do
  5. before do
  6. @site = {
  7. 'expected_update_period_in_days' => 1,
  8. 'host' => 'mail.example.net',
  9. 'ssl' => true,
  10. 'username' => 'foo',
  11. 'password' => 'bar',
  12. 'folders' => ['INBOX'],
  13. 'conditions' => {
  14. }
  15. }
  16. @checker = Agents::ImapFolderAgent.new(:name => 'Example', :options => @site, :keep_events_for => 2.days)
  17. @checker.user = users(:bob)
  18. @checker.save!
  19. message_mixin = Module.new {
  20. def folder
  21. 'INBOX'
  22. end
  23. def uidvalidity
  24. 100
  25. end
  26. def has_attachment?
  27. false
  28. end
  29. def body_parts(mime_types = %[text/plain text/enriched text/html])
  30. mime_types.map { |type|
  31. all_parts.find { |part|
  32. part.mime_type == type
  33. }
  34. }.compact.map! { |part|
  35. part.extend(Agents::ImapFolderAgent::Message::Scrubbed)
  36. }
  37. end
  38. include Agents::ImapFolderAgent::Message::Scrubbed
  39. }
  40. @mails = [
  41. Mail.read(Rails.root.join('spec/data_fixtures/imap1.eml')).tap { |mail|
  42. mail.extend(message_mixin)
  43. stub(mail).uid.returns(1)
  44. },
  45. Mail.read(Rails.root.join('spec/data_fixtures/imap2.eml')).tap { |mail|
  46. mail.extend(message_mixin)
  47. stub(mail).uid.returns(2)
  48. stub(mail).has_attachment?.returns(true)
  49. },
  50. ]
  51. stub(@checker).each_unread_mail.returns { |yielder|
  52. seen = @checker.lastseen
  53. notified = @checker.notified
  54. @mails.each_with_object(notified) { |mail|
  55. yielder[mail, notified]
  56. seen[mail.uidvalidity] = mail.uid
  57. }
  58. @checker.lastseen = seen
  59. @checker.notified = notified
  60. nil
  61. }
  62. @payloads = [
  63. {
  64. 'folder' => 'INBOX',
  65. 'from' => 'nanashi.gombeh@example.jp',
  66. 'to' => ['jane.doe@example.com', 'john.doe@example.com'],
  67. 'cc' => [],
  68. 'date' => '2014-05-09T16:00:00+09:00',
  69. 'subject' => 'some subject',
  70. 'body' => "Some plain text\nSome second line\n",
  71. 'has_attachment' => false,
  72. 'matches' => {},
  73. 'mime_type' => 'text/plain',
  74. },
  75. {
  76. 'folder' => 'INBOX',
  77. 'from' => 'john.doe@example.com',
  78. 'to' => ['jane.doe@example.com', 'nanashi.gombeh@example.jp'],
  79. 'cc' => [],
  80. 'subject' => 'Re: some subject',
  81. 'body' => "Some reply\n",
  82. 'date' => '2014-05-09T17:00:00+09:00',
  83. 'has_attachment' => true,
  84. 'matches' => {},
  85. 'mime_type' => 'text/plain',
  86. }
  87. ]
  88. end
  89. describe 'validations' do
  90. before do
  91. expect(@checker).to be_valid
  92. end
  93. it 'should validate the integer fields' do
  94. @checker.options['expected_update_period_in_days'] = 'nonsense'
  95. expect(@checker).not_to be_valid
  96. @checker.options['expected_update_period_in_days'] = '2'
  97. expect(@checker).to be_valid
  98. @checker.options['port'] = -1
  99. expect(@checker).not_to be_valid
  100. @checker.options['port'] = 'imap'
  101. expect(@checker).not_to be_valid
  102. @checker.options['port'] = '143'
  103. expect(@checker).to be_valid
  104. @checker.options['port'] = 993
  105. expect(@checker).to be_valid
  106. end
  107. it 'should validate the boolean fields' do
  108. %w[ssl mark_as_read].each do |key|
  109. @checker.options[key] = 1
  110. expect(@checker).not_to be_valid
  111. @checker.options[key] = false
  112. expect(@checker).to be_valid
  113. @checker.options[key] = 'true'
  114. expect(@checker).to be_valid
  115. @checker.options[key] = ''
  116. expect(@checker).to be_valid
  117. end
  118. end
  119. it 'should validate regexp conditions' do
  120. @checker.options['conditions'] = {
  121. 'subject' => '(foo'
  122. }
  123. expect(@checker).not_to be_valid
  124. @checker.options['conditions'] = {
  125. 'body' => '***'
  126. }
  127. expect(@checker).not_to be_valid
  128. @checker.options['conditions'] = {
  129. 'subject' => '\ARe:',
  130. 'body' => '(?<foo>http://\S+)'
  131. }
  132. expect(@checker).to be_valid
  133. end
  134. end
  135. describe '#check' do
  136. it 'should check for mails and save memory' do
  137. expect { @checker.check }.to change { Event.count }.by(2)
  138. expect(@checker.notified.sort).to eq(@mails.map(&:message_id).sort)
  139. expect(@checker.lastseen).to eq(@mails.each_with_object(@checker.make_seen) { |mail, seen|
  140. seen[mail.uidvalidity] = mail.uid
  141. })
  142. Event.last(2).map(&:payload) == @payloads
  143. expect { @checker.check }.not_to change { Event.count }
  144. end
  145. it 'should narrow mails by To' do
  146. @checker.options['conditions']['to'] = 'John.Doe@*'
  147. expect { @checker.check }.to change { Event.count }.by(1)
  148. expect(@checker.notified.sort).to eq([@mails.first.message_id])
  149. expect(@checker.lastseen).to eq(@mails.each_with_object(@checker.make_seen) { |mail, seen|
  150. seen[mail.uidvalidity] = mail.uid
  151. })
  152. expect(Event.last.payload).to eq(@payloads.first)
  153. expect { @checker.check }.not_to change { Event.count }
  154. end
  155. it 'should not fail when a condition on Cc is given and a mail does not have the field' do
  156. @checker.options['conditions']['cc'] = 'John.Doe@*'
  157. expect {
  158. expect { @checker.check }.not_to change { Event.count }
  159. }.not_to raise_exception
  160. end
  161. it 'should perform regexp matching and save named captures' do
  162. @checker.options['conditions'].update(
  163. 'subject' => '\ARe: (?<a>.+)',
  164. 'body' => 'Some (?<b>.+) reply',
  165. )
  166. expect { @checker.check }.to change { Event.count }.by(1)
  167. expect(@checker.notified.sort).to eq([@mails.last.message_id])
  168. expect(@checker.lastseen).to eq(@mails.each_with_object(@checker.make_seen) { |mail, seen|
  169. seen[mail.uidvalidity] = mail.uid
  170. })
  171. expect(Event.last.payload).to eq(@payloads.last.update(
  172. 'body' => "<div dir=\"ltr\">Some HTML reply<br></div>\n",
  173. 'matches' => { 'a' => 'some subject', 'b' => 'HTML' },
  174. 'mime_type' => 'text/html',
  175. ))
  176. expect { @checker.check }.not_to change { Event.count }
  177. end
  178. it 'should narrow mails by has_attachment (true)' do
  179. @checker.options['conditions']['has_attachment'] = true
  180. expect { @checker.check }.to change { Event.count }.by(1)
  181. expect(Event.last.payload['subject']).to eq('Re: some subject')
  182. end
  183. it 'should narrow mails by has_attachment (false)' do
  184. @checker.options['conditions']['has_attachment'] = false
  185. expect { @checker.check }.to change { Event.count }.by(1)
  186. expect(Event.last.payload['subject']).to eq('some subject')
  187. end
  188. it 'should narrow mail parts by MIME types' do
  189. @checker.options['mime_types'] = %w[text/plain]
  190. @checker.options['conditions'].update(
  191. 'subject' => '\ARe: (?<a>.+)',
  192. 'body' => 'Some (?<b>.+) reply',
  193. )
  194. expect { @checker.check }.not_to change { Event.count }
  195. expect(@checker.notified.sort).to eq([])
  196. expect(@checker.lastseen).to eq(@mails.each_with_object(@checker.make_seen) { |mail, seen|
  197. seen[mail.uidvalidity] = mail.uid
  198. })
  199. end
  200. it 'should never mark mails as read unless mark_as_read is true' do
  201. @mails.each { |mail|
  202. stub(mail).mark_as_read.never
  203. }
  204. expect { @checker.check }.to change { Event.count }.by(2)
  205. end
  206. it 'should mark mails as read if mark_as_read is true' do
  207. @checker.options['mark_as_read'] = true
  208. @mails.each { |mail|
  209. stub(mail).mark_as_read.once
  210. }
  211. expect { @checker.check }.to change { Event.count }.by(2)
  212. end
  213. it 'should create just one event for multiple mails with the same Message-Id' do
  214. @mails.first.message_id = @mails.last.message_id
  215. @checker.options['mark_as_read'] = true
  216. @mails.each { |mail|
  217. stub(mail).mark_as_read.once
  218. }
  219. expect { @checker.check }.to change { Event.count }.by(1)
  220. end
  221. end
  222. end
  223. describe 'Agents::ImapFolderAgent::Message::Scrubbed' do
  224. before do
  225. @class = Class.new do
  226. def subject
  227. "broken\xB7subject\xB6"
  228. end
  229. def body
  230. "broken\xB7body\xB6"
  231. end
  232. include Agents::ImapFolderAgent::Message::Scrubbed
  233. end
  234. @object = @class.new
  235. end
  236. describe '#scrubbed' do
  237. it 'should return a scrubbed string' do
  238. expect(@object.scrubbed(:subject)).to eq("broken<b7>subject<b6>")
  239. expect(@object.scrubbed(:body)).to eq("broken<b7>body<b6>")
  240. end
  241. end
  242. end
  243. end