custom_payload_spec.rb 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519
  1. # Copyright (C) 2012-2024 Zammad Foundation, https://zammad-foundation.org/
  2. require 'rails_helper'
  3. RSpec.describe TriggerWebhookJob::CustomPayload do
  4. # rubocop:disable Lint/InterpolationCheck
  5. describe '.generate' do
  6. subject(:generate) { described_class.generate(record, { ticket:, article:, notification: }) }
  7. let(:ticket) { create(:ticket) }
  8. let(:article) { create(:ticket_article, body: "Text with\nnew line.") }
  9. let(:event) do
  10. {
  11. type: 'info',
  12. execution: 'trigger',
  13. changes: { 'state' => %w[open closed] },
  14. user_id: 1,
  15. }
  16. end
  17. let(:notification) { TriggerWebhookJob::CustomPayload::Track::Notification.generate({ ticket:, article: }, { event: }) }
  18. context 'when the payload is empty' do
  19. let(:record) { {}.to_json }
  20. let(:json_data) { {} }
  21. it 'returns an empty JSON object' do
  22. expect(generate).to eq(json_data)
  23. end
  24. end
  25. context 'when the placeholder is empty' do
  26. let(:record) { { 'ticket' => '#{}' }.to_json }
  27. let(:json_data) { { 'ticket' => '#{}' } }
  28. it 'returns the placeholder' do
  29. expect(generate).to eq(json_data)
  30. end
  31. end
  32. context 'when the placeholder is invalid' do
  33. let(:record) { { 'ticket' => '#{ticket.title', 'article' => '#{article.id article.note}' }.to_json }
  34. let(:json_data) { { 'ticket' => '#{ticket.title', 'article' => '#{article.id article.note}' } }
  35. it 'returns the placeholder' do
  36. expect(generate).to eq(json_data)
  37. end
  38. end
  39. context 'when the placeholder base object ticket or article is missing' do
  40. let(:record) { { 'ticket' => '#{.title}' }.to_json }
  41. let(:json_data) { { 'ticket' => '#{no object provided}' } }
  42. it 'returns the placeholder reporting "no object provided"' do
  43. expect(generate).to eq(json_data)
  44. end
  45. end
  46. context 'when the placeholder base object is other than ticket or article' do
  47. let(:record) { { 'user' => '#{user}' }.to_json }
  48. let(:json_data) { { 'user' => '#{user / no such object}' } }
  49. it 'returns the placehodler reporting "no such object"' do
  50. expect(generate).to eq(json_data)
  51. end
  52. end
  53. context 'when the placeholder contains only base object ticket or article' do
  54. let(:record) { { 'ticket' => '#{ticket}', 'Article' => '#{article}' }.to_json }
  55. let(:json_data) { { 'ticket' => '#{ticket / missing method}', 'Article' => '#{article / missing method}' } }
  56. it 'returns the placeholder reporting "missing method"' do
  57. expect(generate).to eq(json_data)
  58. end
  59. end
  60. context 'when the placeholder contains denied method' do
  61. let(:record) { { 'ticket' => '#{ticket.articles}' }.to_json }
  62. let(:json_data) { { 'ticket' => '#{ticket.articles / no such method}' } }
  63. it 'returns the placeholder reporting "no such method"' do
  64. expect(generate).to eq(json_data)
  65. end
  66. end
  67. context 'when the placeholder contains denied attribute' do
  68. let(:record) { { 'ticket.owner' => '#{ticket.owner.password}' }.to_json }
  69. let(:json_data) { { 'ticket.owner' => '#{ticket.owner.password / no such method}' } }
  70. it 'returns the placeholder reporting "no such method"' do
  71. expect(generate).to eq(json_data)
  72. end
  73. end
  74. context 'when the placeholder contains danger method' do
  75. let(:record) { { 'ticket.owner' => '#{ticket.destroy!}' }.to_json }
  76. let(:json_data) { { 'ticket.owner' => '#{ticket.destroy! / no such method}' } }
  77. it 'returns the placeholder reporting "no such method"' do
  78. expect(generate).to eq(json_data)
  79. end
  80. end
  81. context 'when the placeholder ends with complex object' do
  82. let(:record) { { 'ticket' => '#{ticket.group}' }.to_json }
  83. let(:json_data) { { 'ticket' => '#{ticket.group / no such method}' } }
  84. it 'returns the placeholder reporting "no such method"' do
  85. expect(generate).to eq(json_data)
  86. end
  87. end
  88. context 'when the placeholder contains valid object and method' do
  89. let(:record) { { 'ticket.id' => '#{ticket.id}' }.to_json }
  90. let(:json_data) { { 'ticket.id' => ticket.id } }
  91. it 'returns the determined value' do
  92. expect(generate).to eq(json_data)
  93. end
  94. end
  95. context 'when the placeholder contains valid object and method, but the value is nil' do
  96. let(:record) do
  97. {
  98. 'ticket.organization.name' => '#{ticket.organization.name}',
  99. 'ticket.title' => '#{ticket.title}'
  100. }.to_json
  101. end
  102. let(:json_data) do
  103. {
  104. 'ticket.organization.name' => '',
  105. 'ticket.title' => ticket.title
  106. }
  107. end
  108. it 'returns an empty string' do
  109. expect(generate).to eq(json_data)
  110. end
  111. end
  112. context 'when the placeholder contains multiple valid object and method' do
  113. let(:record) do
  114. {
  115. 'ticket' => { 'owner' => '#{ticket.owner.fullname}' },
  116. 'article' => { 'created_at' => '#{article.created_at}' }
  117. }.to_json
  118. end
  119. let(:json_data) do
  120. {
  121. 'ticket' => { 'owner' => ticket.owner.fullname.to_s },
  122. 'article' => { 'created_at' => article.created_at.to_s }
  123. }
  124. end
  125. it 'returns the determined value' do
  126. expect(generate).to eq(json_data)
  127. end
  128. end
  129. context 'when the placeholder contains multiple attributes' do
  130. let(:record) do
  131. {
  132. 'my_field' => 'Test #{ticket.id} // #{ticket.group.name} Test',
  133. 'my_field2' => '#{ticket.id} // #{ticket.group.name} Test',
  134. 'my_field3' => '#{ticket.id}',
  135. 'my_field4' => '#{ticket.group.name}',
  136. }.to_json
  137. end
  138. let(:json_data) do
  139. {
  140. 'my_field' => "Test #{ticket.id} // #{ticket.group.name} Test",
  141. 'my_field2' => "#{ticket.id} // #{ticket.group.name} Test",
  142. 'my_field3' => ticket.id,
  143. 'my_field4' => ticket.group.name.to_s,
  144. }
  145. end
  146. it 'returns the placeholder reporting "no such method"' do
  147. expect(generate).to eq(json_data)
  148. end
  149. end
  150. context 'when the payload contains a complex structure' do
  151. let(:record) do
  152. {
  153. 'current_user' => '#{current_user.fullname}',
  154. 'ticket' => {
  155. 'id' => '#{ticket.id}',
  156. 'owner' => '#{ticket.owner.fullname}',
  157. 'group' => '#{ticket.group.name}',
  158. 'article' => {
  159. 'id' => '#{article.id}',
  160. 'created_at' => '#{article.created_at}',
  161. 'subject' => '#{article.subject}',
  162. 'body' => '#{article.body}',
  163. 'attachments' => '#{article.attachments}',
  164. 'internal' => '#{article.internal}',
  165. }
  166. }
  167. }.to_json
  168. end
  169. let(:json_data) do
  170. {
  171. 'current_user' => '#{current_user / no such object}',
  172. 'ticket' => {
  173. 'id' => ticket.id,
  174. 'owner' => ticket.owner.fullname.to_s,
  175. 'group' => ticket.group.name.to_s,
  176. 'article' => {
  177. 'id' => article.id,
  178. 'created_at' => article.created_at.to_s,
  179. 'subject' => article.subject.to_s,
  180. 'body' => article.body.to_s,
  181. 'attachments' => '#{article.attachments / no such method}',
  182. 'internal' => article.internal
  183. }
  184. }
  185. }
  186. end
  187. it 'returns a valid JSON payload' do
  188. expect(generate).to eq(json_data)
  189. end
  190. end
  191. context 'when the replacement value contains double quotes' do
  192. let(:ticket) { create(:ticket, title: 'Test "Title"') }
  193. let(:record) { { 'ticket.title' => '#{ticket.title}' }.to_json }
  194. let(:json_data) { { 'ticket.title' => 'Test "Title"' } }
  195. it 'returns the determined value' do
  196. expect(generate).to eq(json_data)
  197. end
  198. end
  199. context 'when object attributes are used in the placeholder', db_strategy: :reset do
  200. let(:ticket) { create(:ticket, object_manager_attribute_name => object_manager_attribute_value) }
  201. before do
  202. create_object_manager_attribute
  203. ObjectManager::Attribute.migration_execute
  204. end
  205. shared_examples 'check different usage' do
  206. context 'when used in string context' do
  207. let(:record) do
  208. {
  209. "ticket.#{object_manager_attribute_name}" => "Test \#{ticket.#{object_manager_attribute_name}}"
  210. }.to_json
  211. end
  212. let(:json_data) do
  213. {
  214. "ticket.#{object_manager_attribute_name}" => "Test #{object_manager_attribute_value}"
  215. }
  216. end
  217. it 'returns the determined value' do
  218. expect(generate).to eq(json_data)
  219. end
  220. end
  221. context 'when used in direct context' do
  222. let(:record) do
  223. {
  224. "ticket.#{object_manager_attribute_name}" => "\#{ticket.#{object_manager_attribute_name}}"
  225. }.to_json
  226. end
  227. let(:json_data) do
  228. {
  229. "ticket.#{object_manager_attribute_name}" => object_manager_attribute_value
  230. }
  231. end
  232. it 'returns the determined value' do
  233. expect(generate).to eq(json_data)
  234. end
  235. end
  236. end
  237. context 'when multiselect is used inside the ticket' do
  238. let(:object_manager_attribute_name) { 'multiselect' }
  239. let(:object_manager_attribute_value) { %w[key_1 key_3] }
  240. let(:create_object_manager_attribute) do
  241. create(:object_manager_attribute_multiselect, name: object_manager_attribute_name)
  242. end
  243. include_examples 'check different usage'
  244. end
  245. context 'when external data source is used inside the ticket', db_adapter: :postgresql do
  246. let(:object_manager_attribute_name) { 'autocompletion_ajax_external_data_source' }
  247. let(:object_manager_attribute_value) do
  248. {
  249. 'value' => 123,
  250. 'label' => 'Example',
  251. }
  252. end
  253. let(:create_object_manager_attribute) do
  254. create(:object_manager_attribute_autocompletion_ajax_external_data_source, name: object_manager_attribute_name)
  255. end
  256. include_examples 'check different usage'
  257. end
  258. end
  259. describe "when the placeholder contains object 'notification'" do
  260. let(:record) do
  261. {
  262. 'subject' => '#{notification.subject}',
  263. 'message' => '#{notification.message}',
  264. 'changes' => '#{notification.changes}',
  265. 'body' => '#{notification.body}',
  266. 'link' => '#{notification.link}',
  267. }.to_json
  268. end
  269. context "when the event is of the type 'create'" do
  270. let(:event) do
  271. {
  272. type: 'create',
  273. execution: 'trigger',
  274. user_id: 1,
  275. }
  276. end
  277. it 'returns a valid json with a notification factory generated information"', :aggregate_failures do
  278. expect(generate['subject']).to eq(ticket.title)
  279. expect(generate['body']).to eq(article.body_as_text)
  280. expect(generate['link']).to match(%r{http.*#ticket/zoom/#{ticket.id}$})
  281. expect(generate['message']).to include('Created by')
  282. expect(generate['changes']).to include('State: new')
  283. end
  284. end
  285. context "when the event is of the type 'update'" do
  286. let(:event) do
  287. {
  288. type: 'update',
  289. execution: 'trigger',
  290. changes: { 'state' => %w[open closed] },
  291. user_id: 1,
  292. }
  293. end
  294. it 'returns a valid json with a notification factory generated information"', :aggregate_failures do
  295. expect(generate['subject']).to eq(ticket.title)
  296. expect(generate['body']).to eq(article.body_as_text)
  297. expect(generate['link']).to match(%r{http.*#ticket/zoom/#{ticket.id}$})
  298. expect(generate['message']).to include('Updated by')
  299. expect(generate['changes']).to include('state: open -> closed')
  300. end
  301. context 'without changes' do
  302. let(:event) do
  303. {
  304. type: 'update',
  305. execution: 'trigger',
  306. user_id: 1,
  307. }
  308. end
  309. it 'returns a valid json with a notification factory generated information"', :aggregate_failures do
  310. expect(generate['subject']).to eq(ticket.title)
  311. expect(generate['body']).to eq(article.body_as_text)
  312. expect(generate['link']).to match(%r{http.*#ticket/zoom/#{ticket.id}$})
  313. expect(generate['message']).to include('Updated by')
  314. end
  315. end
  316. end
  317. context "when the event is of the type 'info'" do
  318. let(:event) do
  319. {
  320. type: 'info',
  321. execution: 'trigger',
  322. changes: { 'state' => %w[open closed] },
  323. user_id: 1,
  324. }
  325. end
  326. it 'returns a valid json with a notification factory generated information"', :aggregate_failures do
  327. expect(generate['subject']).to eq(ticket.title)
  328. expect(generate['body']).to eq(article.body_as_text)
  329. expect(generate['link']).to match(%r{http.*#ticket/zoom/#{ticket.id}$})
  330. expect(generate['message']).to include('Last updated at')
  331. end
  332. end
  333. context "when the event is of the type 'escalation'" do
  334. let(:event) do
  335. {
  336. type: 'escalation',
  337. execution: 'trigger',
  338. user_id: 1,
  339. }
  340. end
  341. it 'returns a valid json with a notification factory generated information"', :aggregate_failures do
  342. expect(generate['subject']).to eq(ticket.title)
  343. expect(generate['body']).to eq(article.body_as_text)
  344. expect(generate['link']).to match(%r{http.*#ticket/zoom/#{ticket.id}$})
  345. expect(generate['message']).to include('Escalated at')
  346. expect(generate['changes']).to include('has been escalated since')
  347. end
  348. end
  349. context "when the event is of the type 'escalation warning'" do
  350. let(:event) do
  351. {
  352. type: 'escalation_warning',
  353. execution: 'trigger',
  354. user_id: 1,
  355. }
  356. end
  357. it 'returns a valid json with a notification factory generated information"', :aggregate_failures do
  358. expect(generate['subject']).to eq(ticket.title)
  359. expect(generate['body']).to eq(article.body_as_text)
  360. expect(generate['link']).to match(%r{http.*#ticket/zoom/#{ticket.id}$})
  361. expect(generate['message']).to include('Will escalate at')
  362. expect(generate['changes']).to include('will escalate at')
  363. end
  364. end
  365. context "when the event is of the type 'reminder reached'" do
  366. let(:event) do
  367. {
  368. type: 'reminder_reached',
  369. execution: 'trigger',
  370. user_id: 1,
  371. }
  372. end
  373. it 'returns a valid json with a notification factory generated information"', :aggregate_failures do
  374. expect(generate['subject']).to eq(ticket.title)
  375. expect(generate['body']).to eq(article.body_as_text)
  376. expect(generate['link']).to match(%r{http.*#ticket/zoom/#{ticket.id}$})
  377. expect(generate['message']).to include('Reminder reached!')
  378. expect(generate['changes']).to include('reminder reached for')
  379. end
  380. end
  381. context "when the event is triggered by a 'job'" do
  382. let(:event) do
  383. {
  384. type: '',
  385. execution: 'job',
  386. changes: { 'state' => %w[open closed] },
  387. user_id: 1,
  388. }
  389. end
  390. let(:article) { nil }
  391. it 'returns a valid json with a notification factory generated information"', :aggregate_failures do
  392. expect(generate['subject']).to eq(ticket.title)
  393. expect(generate['body']).to be_empty
  394. expect(generate['link']).to match(%r{http.*#ticket/zoom/#{ticket.id}$})
  395. expect(generate['message']).to include('Last updated at')
  396. end
  397. end
  398. end
  399. describe 'when the payload is a pre-defined webhook' do
  400. subject(:generate) { described_class.generate(record, { ticket:, article:, notification:, webhook: struct_webhook }) }
  401. let(:webhook) { create(:mattermost_webhook) }
  402. let(:struct_webhook) { TriggerWebhookJob::CustomPayload::Track::PreDefinedWebhook.generate({ ticket:, article: }, { event:, webhook: }) }
  403. let(:record) { TriggerWebhookJob::CustomPayload::Track::PreDefinedWebhook.payload('Mattermost') }
  404. it 'returns a valid json with webhook information"', :aggregate_failures do
  405. info = webhook.preferences[:pre_defined_webhook]
  406. expect(generate[:channel]).to eq(info[:channel])
  407. expect(generate[:icon_url]).to eq(info[:icon_url])
  408. end
  409. context 'when event has no changes' do
  410. let(:event) do
  411. {
  412. type: 'info',
  413. execution: 'trigger',
  414. changes: { 'state' => %w[open closed] },
  415. user_id: 1,
  416. }
  417. end
  418. it "returns a valid json with webhook information without 'attachments'", :aggregate_failures do
  419. info = webhook.preferences[:pre_defined_webhook]
  420. expect(generate[:channel]).to eq(info[:channel])
  421. expect(generate[:icon_url]).to eq(info[:icon_url])
  422. expect(generate).to not_include(:attachments)
  423. end
  424. end
  425. context 'when pre-defined webhook has no additional values' do
  426. let(:webhook) { create(:slack_webhook) }
  427. let(:record) { TriggerWebhookJob::CustomPayload::Track::PreDefinedWebhook.payload('Slack') }
  428. it 'returns a valid json with webhook information"', :aggregate_failures do
  429. expect(generate['text']).to eq("# #{ticket.title}")
  430. end
  431. end
  432. end
  433. end
  434. # rubocop:enable Lint/InterpolationCheck
  435. describe '.replacements' do
  436. subject(:replacements) { described_class.replacements(pre_defined_webhook_type: 'Mattermost') }
  437. it 'returns a hash with the replacement variables', :aggregate_failures do
  438. expect(replacements).to be_a(Hash)
  439. expect(replacements.keys).to include(:article, :ticket, :notification, :webhook)
  440. end
  441. end
  442. end