data_output_agent_spec.rb 21KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550
  1. # encoding: utf-8
  2. require 'rails_helper'
  3. describe Agents::DataOutputAgent do
  4. let(:agent) do
  5. _agent = Agents::DataOutputAgent.new(:name => 'My Data Output Agent')
  6. _agent.options = _agent.default_options.merge('secrets' => ['secret1', 'secret2'], 'events_to_show' => 3)
  7. _agent.options['template']['item']['pubDate'] = "{{date}}"
  8. _agent.user = users(:bob)
  9. _agent.sources << agents(:bob_website_agent)
  10. _agent.save!
  11. _agent
  12. end
  13. describe "#working?" do
  14. it "checks if events have been received within expected receive period" do
  15. expect(agent).not_to be_working
  16. Agents::DataOutputAgent.async_receive agent.id, [events(:bob_website_agent_event).id]
  17. expect(agent.reload).to be_working
  18. two_days_from_now = 2.days.from_now
  19. stub(Time).now { two_days_from_now }
  20. expect(agent.reload).not_to be_working
  21. end
  22. end
  23. describe "validation" do
  24. before do
  25. expect(agent).to be_valid
  26. end
  27. it "should validate presence and length of secrets" do
  28. agent.options[:secrets] = ""
  29. expect(agent).not_to be_valid
  30. agent.options[:secrets] = "foo"
  31. expect(agent).not_to be_valid
  32. agent.options[:secrets] = "foo/bar"
  33. expect(agent).not_to be_valid
  34. agent.options[:secrets] = "foo.xml"
  35. expect(agent).not_to be_valid
  36. agent.options[:secrets] = false
  37. expect(agent).not_to be_valid
  38. agent.options[:secrets] = []
  39. expect(agent).not_to be_valid
  40. agent.options[:secrets] = ["foo.xml"]
  41. expect(agent).not_to be_valid
  42. agent.options[:secrets] = ["hello", true]
  43. expect(agent).not_to be_valid
  44. agent.options[:secrets] = ["hello"]
  45. expect(agent).to be_valid
  46. agent.options[:secrets] = ["hello", "world"]
  47. expect(agent).to be_valid
  48. end
  49. it "should validate presence of expected_receive_period_in_days" do
  50. agent.options[:expected_receive_period_in_days] = ""
  51. expect(agent).not_to be_valid
  52. agent.options[:expected_receive_period_in_days] = 0
  53. expect(agent).not_to be_valid
  54. agent.options[:expected_receive_period_in_days] = -1
  55. expect(agent).not_to be_valid
  56. end
  57. it "should validate presence of template and template.item" do
  58. agent.options[:template] = ""
  59. expect(agent).not_to be_valid
  60. agent.options[:template] = {}
  61. expect(agent).not_to be_valid
  62. agent.options[:template] = { 'item' => 'foo' }
  63. expect(agent).not_to be_valid
  64. agent.options[:template] = { 'item' => { 'title' => 'hi' } }
  65. expect(agent).to be_valid
  66. end
  67. end
  68. describe "#receive" do
  69. it "should push to hubs when push_hubs is given" do
  70. agent.options[:push_hubs] = %w[http://push.example.com]
  71. agent.options[:template] = { 'link' => 'http://huginn.example.org' }
  72. alist = nil
  73. stub_request(:post, 'http://push.example.com/')
  74. .with(headers: { 'Content-Type' => %r{\Aapplication/x-www-form-urlencoded\s*(?:;|\z)} })
  75. .to_return { |request|
  76. alist = URI.decode_www_form(request.body).sort
  77. { status: 200, body: 'ok' }
  78. }
  79. agent.receive(events(:bob_website_agent_event))
  80. expect(alist).to eq [
  81. ["hub.mode", "publish"],
  82. ["hub.url", agent.feed_url(secret: agent.options[:secrets].first, format: :xml)]
  83. ]
  84. end
  85. end
  86. describe "#receive_web_request" do
  87. before do
  88. current_time = Time.now
  89. stub(Time).now { current_time }
  90. agents(:bob_website_agent).events.destroy_all
  91. end
  92. it "requires a valid secret" do
  93. content, status, content_type = agent.receive_web_request({ 'secret' => 'fake' }, 'get', 'text/xml')
  94. expect(status).to eq(401)
  95. expect(content).to eq("Not Authorized")
  96. content, status, content_type = agent.receive_web_request({ 'secret' => 'fake' }, 'get', 'application/json')
  97. expect(status).to eq(401)
  98. expect(content).to eq({ :error => "Not Authorized" })
  99. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'application/json')
  100. expect(status).to eq(200)
  101. end
  102. describe "outputting events as RSS and JSON" do
  103. let!(:event1) do
  104. agents(:bob_website_agent).create_event :payload => {
  105. "site_title" => "XKCD",
  106. "url" => "http://imgs.xkcd.com/comics/evolving.png",
  107. "title" => "Evolving",
  108. "hovertext" => "Biologists play reverse Pokemon, trying to avoid putting any one team member on the front lines long enough for the experience to cause evolution."
  109. }
  110. end
  111. let!(:event2) do
  112. agents(:bob_website_agent).create_event :payload => {
  113. "site_title" => "XKCD",
  114. "url" => "http://imgs.xkcd.com/comics/evolving2.png",
  115. "title" => "Evolving again",
  116. "date" => '',
  117. "hovertext" => "Something else"
  118. }
  119. end
  120. let!(:event3) do
  121. agents(:bob_website_agent).create_event :payload => {
  122. "site_title" => "XKCD",
  123. "url" => "http://imgs.xkcd.com/comics/evolving0.png",
  124. "title" => "Evolving yet again with a past date",
  125. "date" => '2014/05/05',
  126. "hovertext" => "A small text"
  127. }
  128. end
  129. it "can output RSS" do
  130. stub(agent).feed_link { "https://yoursite.com" }
  131. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  132. expect(status).to eq(200)
  133. expect(content_type).to eq('text/xml')
  134. expect(content.gsub(/\s+/, '')).to eq Utils.unindent(<<-XML).gsub(/\s+/, '')
  135. <?xml version="1.0" encoding="UTF-8" ?>
  136. <rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:media="http://search.yahoo.com/mrss/">
  137. <channel>
  138. <atom:link href="https://yoursite.com/users/#{agent.user.id}/web_requests/#{agent.id}/secret1.xml" rel="self" type="application/rss+xml"/>
  139. <atom:icon>https://yoursite.com/favicon.ico</atom:icon>
  140. <title>XKCD comics as a feed</title>
  141. <description>This is a feed of recent XKCD comics, generated by Huginn</description>
  142. <link>https://yoursite.com</link>
  143. <lastBuildDate>#{Time.now.rfc2822}</lastBuildDate>
  144. <pubDate>#{Time.now.rfc2822}</pubDate>
  145. <ttl>60</ttl>
  146. <item>
  147. <title>Evolving yet again with a past date</title>
  148. <description>Secret hovertext: A small text</description>
  149. <link>http://imgs.xkcd.com/comics/evolving0.png</link>
  150. <pubDate>#{Time.zone.parse(event3.payload['date']).rfc2822}</pubDate>
  151. <guid isPermaLink="false">#{event3.id}</guid>
  152. </item>
  153. <item>
  154. <title>Evolving again</title>
  155. <description>Secret hovertext: Something else</description>
  156. <link>http://imgs.xkcd.com/comics/evolving2.png</link>
  157. <pubDate>#{event2.created_at.rfc2822}</pubDate>
  158. <guid isPermaLink="false">#{event2.id}</guid>
  159. </item>
  160. <item>
  161. <title>Evolving</title>
  162. <description>Secret hovertext: Biologists play reverse Pokemon, trying to avoid putting any one team member on the front lines long enough for the experience to cause evolution.</description>
  163. <link>http://imgs.xkcd.com/comics/evolving.png</link>
  164. <pubDate>#{event1.created_at.rfc2822}</pubDate>
  165. <guid isPermaLink="false">#{event1.id}</guid>
  166. </item>
  167. </channel>
  168. </rss>
  169. XML
  170. end
  171. it "can output RSS with hub links when push_hubs is specified" do
  172. stub(agent).feed_link { "https://yoursite.com" }
  173. agent.options[:push_hubs] = %w[https://pubsubhubbub.superfeedr.com/ https://pubsubhubbub.appspot.com/]
  174. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  175. expect(status).to eq(200)
  176. expect(content_type).to eq('text/xml')
  177. xml = Nokogiri::XML(content)
  178. expect(xml.xpath('/rss/channel/atom:link[@rel="hub"]/@href').map(&:text).sort).to eq agent.options[:push_hubs].sort
  179. end
  180. it "can output JSON" do
  181. agent.options['template']['item']['foo'] = "hi"
  182. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret2' }, 'get', 'application/json')
  183. expect(status).to eq(200)
  184. expect(content).to eq({
  185. 'title' => 'XKCD comics as a feed',
  186. 'description' => 'This is a feed of recent XKCD comics, generated by Huginn',
  187. 'pubDate' => Time.now,
  188. 'items' => [
  189. {
  190. 'title' => 'Evolving yet again with a past date',
  191. 'description' => 'Secret hovertext: A small text',
  192. 'link' => 'http://imgs.xkcd.com/comics/evolving0.png',
  193. 'guid' => {"contents" => event3.id, "isPermaLink" => "false"},
  194. 'pubDate' => Time.zone.parse(event3.payload['date']).rfc2822,
  195. 'foo' => 'hi'
  196. },
  197. {
  198. 'title' => 'Evolving again',
  199. 'description' => 'Secret hovertext: Something else',
  200. 'link' => 'http://imgs.xkcd.com/comics/evolving2.png',
  201. 'guid' => {"contents" => event2.id, "isPermaLink" => "false"},
  202. 'pubDate' => event2.created_at.rfc2822,
  203. 'foo' => 'hi'
  204. },
  205. {
  206. 'title' => 'Evolving',
  207. 'description' => 'Secret hovertext: Biologists play reverse Pokemon, trying to avoid putting any one team member on the front lines long enough for the experience to cause evolution.',
  208. 'link' => 'http://imgs.xkcd.com/comics/evolving.png',
  209. 'guid' => {"contents" => event1.id, "isPermaLink" => "false"},
  210. 'pubDate' => event1.created_at.rfc2822,
  211. 'foo' => 'hi'
  212. }
  213. ]
  214. })
  215. end
  216. describe 'ordering' do
  217. before do
  218. agent.options['events_order'] = ['{{hovertext}}']
  219. agent.options['events_list_order'] = ['{{title}}']
  220. end
  221. it 'can reorder the events_to_show last events based on a Liquid expression' do
  222. agent.options['events_to_show'] = 2
  223. asc_content, _status, _content_type = agent.receive_web_request({ 'secret' => 'secret2' }, 'get', 'application/json')
  224. expect(asc_content['items'].map {|i| i["title"] }).to eq(["Evolving", "Evolving again"])
  225. agent.options['events_to_show'] = 40
  226. asc_content, _status, _content_type = agent.receive_web_request({ 'secret' => 'secret2' }, 'get', 'application/json')
  227. expect(asc_content['items'].map {|i| i["title"] }).to eq(["Evolving", "Evolving again", "Evolving yet again with a past date"])
  228. agent.options['events_list_order'] = [['{{title}}', 'string', true]]
  229. desc_content, _status, _content_type = agent.receive_web_request({ 'secret' => 'secret2' }, 'get', 'application/json')
  230. expect(desc_content['items']).to eq(asc_content['items'].reverse)
  231. end
  232. end
  233. describe "interpolating \"events\"" do
  234. before do
  235. agent.options['template']['title'] = "XKCD comics as a feed{% if events.first.site_title %} ({{events.first.site_title}}){% endif %}"
  236. agent.save!
  237. end
  238. it "can output RSS" do
  239. stub(agent).feed_link { "https://yoursite.com" }
  240. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  241. expect(status).to eq(200)
  242. expect(content_type).to eq('text/xml')
  243. expect(Nokogiri(content).at('/rss/channel/title/text()').text).to eq('XKCD comics as a feed (XKCD)')
  244. end
  245. it "can output JSON" do
  246. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret2' }, 'get', 'application/json')
  247. expect(status).to eq(200)
  248. expect(content['title']).to eq('XKCD comics as a feed (XKCD)')
  249. end
  250. end
  251. describe "with a specified icon" do
  252. before do
  253. agent.options['template']['icon'] = 'https://somesite.com/icon.png'
  254. agent.save!
  255. end
  256. it "can output RSS" do
  257. stub(agent).feed_link { "https://yoursite.com" }
  258. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  259. expect(status).to eq(200)
  260. expect(content_type).to eq('text/xml')
  261. expect(Nokogiri(content).at('/rss/channel/atom:icon/text()').text).to eq('https://somesite.com/icon.png')
  262. end
  263. end
  264. describe "with media namespace not set" do
  265. before do
  266. agent.options['ns_media'] = nil
  267. agent.save!
  268. end
  269. it "can output RSS" do
  270. stub(agent).feed_link { "https://yoursite.com" }
  271. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  272. expect(status).to eq(200)
  273. expect(content_type).to eq('text/xml')
  274. doc = Nokogiri(content)
  275. namespaces = doc.collect_namespaces
  276. expect(namespaces).not_to include("xmlns:media")
  277. end
  278. end
  279. describe "with media namespace set true" do
  280. before do
  281. agent.options['ns_media'] = 'true'
  282. agent.save!
  283. end
  284. it "can output RSS" do
  285. stub(agent).feed_link { "https://yoursite.com" }
  286. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  287. expect(status).to eq(200)
  288. expect(content_type).to eq('text/xml')
  289. doc = Nokogiri(content)
  290. namespaces = doc.collect_namespaces
  291. expect(namespaces).to include(
  292. "xmlns:media" => 'http://search.yahoo.com/mrss/'
  293. )
  294. end
  295. end
  296. describe "with media namespace set false" do
  297. before do
  298. agent.options['ns_media'] = 'false'
  299. agent.save!
  300. end
  301. it "can output RSS" do
  302. stub(agent).feed_link { "https://yoursite.com" }
  303. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  304. expect(status).to eq(200)
  305. expect(content_type).to eq('text/xml')
  306. doc = Nokogiri(content)
  307. namespaces = doc.collect_namespaces
  308. expect(namespaces).not_to include("xmlns:media")
  309. end
  310. end
  311. describe "with itunes namespace not set" do
  312. before do
  313. agent.options['ns_itunes'] = nil
  314. agent.save!
  315. end
  316. it "can output RSS" do
  317. stub(agent).feed_link { "https://yoursite.com" }
  318. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  319. expect(status).to eq(200)
  320. expect(content_type).to eq('text/xml')
  321. doc = Nokogiri(content)
  322. namespaces = doc.collect_namespaces
  323. expect(namespaces).not_to include("xmlns:itunes")
  324. end
  325. end
  326. describe "with itunes namespace set true" do
  327. before do
  328. agent.options['ns_itunes'] = 'true'
  329. agent.save!
  330. end
  331. it "can output RSS" do
  332. stub(agent).feed_link { "https://yoursite.com" }
  333. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  334. expect(status).to eq(200)
  335. expect(content_type).to eq('text/xml')
  336. doc = Nokogiri(content)
  337. namespaces = doc.collect_namespaces
  338. expect(namespaces).to include(
  339. "xmlns:itunes" => 'http://www.itunes.com/dtds/podcast-1.0.dtd'
  340. )
  341. end
  342. end
  343. describe "with itunes namespace set false" do
  344. before do
  345. agent.options['ns_itunes'] = 'false'
  346. agent.save!
  347. end
  348. it "can output RSS" do
  349. stub(agent).feed_link { "https://yoursite.com" }
  350. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  351. expect(status).to eq(200)
  352. expect(content_type).to eq('text/xml')
  353. doc = Nokogiri(content)
  354. namespaces = doc.collect_namespaces
  355. expect(namespaces).not_to include("xmlns:itunes")
  356. end
  357. end
  358. end
  359. describe "outputting nesting" do
  360. before do
  361. agent.options['template']['item']['enclosure'] = {
  362. "_attributes" => {
  363. "type" => "audio/mpeg",
  364. "url" => "{{media_url}}"
  365. }
  366. }
  367. agent.options['template']['item']['foo'] = {
  368. "_attributes" => {
  369. "attr" => "attr-value-{{foo}}"
  370. },
  371. "_contents" => "Foo: {{foo}}"
  372. }
  373. agent.options['template']['item']['nested'] = {
  374. "_attributes" => {
  375. "key" => "value"
  376. },
  377. "_contents" => {
  378. "title" => "some title"
  379. }
  380. }
  381. agent.options['template']['item']['simpleNested'] = {
  382. "title" => "some title",
  383. "complex" => {
  384. "_attributes" => {
  385. "key" => "value"
  386. },
  387. "_contents" => {
  388. "first" => {
  389. "_attributes" => {
  390. "a" => "b"
  391. },
  392. "_contents" => {
  393. "second" => "value"
  394. }
  395. }
  396. }
  397. }
  398. }
  399. agent.save!
  400. end
  401. let!(:event) do
  402. agents(:bob_website_agent).create_event :payload => {
  403. "url" => "http://imgs.xkcd.com/comics/evolving.png",
  404. "title" => "Evolving",
  405. "hovertext" => "Biologists play reverse Pokemon, trying to avoid putting any one team member on the front lines long enough for the experience to cause evolution.",
  406. "media_url" => "http://google.com/audio.mpeg",
  407. "foo" => 1
  408. }
  409. end
  410. it "can output JSON" do
  411. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret2' }, 'get', 'application/json')
  412. expect(status).to eq(200)
  413. expect(content['items'].first).to eq(
  414. {
  415. 'title' => 'Evolving',
  416. 'description' => 'Secret hovertext: Biologists play reverse Pokemon, trying to avoid putting any one team member on the front lines long enough for the experience to cause evolution.',
  417. 'link' => 'http://imgs.xkcd.com/comics/evolving.png',
  418. 'guid' => {"contents" => event.id, "isPermaLink" => "false"},
  419. 'pubDate' => event.created_at.rfc2822,
  420. 'enclosure' => {
  421. "type" => "audio/mpeg",
  422. "url" => "http://google.com/audio.mpeg"
  423. },
  424. 'foo' => {
  425. 'attr' => 'attr-value-1',
  426. 'contents' => 'Foo: 1'
  427. },
  428. 'nested' => {
  429. "key" => "value",
  430. "title" => "some title"
  431. },
  432. 'simpleNested' => {
  433. "title" => "some title",
  434. "complex" => {
  435. "key"=>"value",
  436. "first" => {
  437. "a" => "b",
  438. "second"=>"value"
  439. }
  440. }
  441. }
  442. }
  443. )
  444. end
  445. it "can output RSS" do
  446. stub(agent).feed_link { "https://yoursite.com" }
  447. content, status, content_type = agent.receive_web_request({ 'secret' => 'secret1' }, 'get', 'text/xml')
  448. expect(status).to eq(200)
  449. expect(content_type).to eq('text/xml')
  450. expect(content.gsub(/\s+/, '')).to eq Utils.unindent(<<-XML).gsub(/\s+/, '')
  451. <?xml version="1.0" encoding="UTF-8" ?>
  452. <rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:media="http://search.yahoo.com/mrss/" >
  453. <channel>
  454. <atom:link href="https://yoursite.com/users/#{agent.user.id}/web_requests/#{agent.id}/secret1.xml" rel="self" type="application/rss+xml"/>
  455. <atom:icon>https://yoursite.com/favicon.ico</atom:icon>
  456. <title>XKCD comics as a feed</title>
  457. <description>This is a feed of recent XKCD comics, generated by Huginn</description>
  458. <link>https://yoursite.com</link>
  459. <lastBuildDate>#{Time.now.rfc2822}</lastBuildDate>
  460. <pubDate>#{Time.now.rfc2822}</pubDate>
  461. <ttl>60</ttl>
  462. <item>
  463. <title>Evolving</title>
  464. <description>Secret hovertext: Biologists play reverse Pokemon, trying to avoid putting any one team member on the front lines long enough for the experience to cause evolution.</description>
  465. <link>http://imgs.xkcd.com/comics/evolving.png</link>
  466. <pubDate>#{event.created_at.rfc2822}</pubDate>
  467. <enclosure type="audio/mpeg" url="http://google.com/audio.mpeg" />
  468. <foo attr="attr-value-1">Foo: 1</foo>
  469. <nested key="value"><title>some title</title></nested>
  470. <simpleNested>
  471. <title>some title</title>
  472. <complex key="value">
  473. <first a="b">
  474. <second>value</second>
  475. </first>
  476. </complex>
  477. </simpleNested>
  478. <guid isPermaLink="false">#{event.id}</guid>
  479. </item>
  480. </channel>
  481. </rss>
  482. XML
  483. end
  484. end
  485. end
  486. end