123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- FactoryBot.define do
- factory :'ticket/article', aliases: %i[ticket_article] do
- inbound_email
- ticket factory: :ticket, strategy: :create # or else build(:ticket_article).save fails
- from { 'factory-customer-1@example.com' }
- to { 'factory-customer-1@example.com' }
- subject { 'factory article' }
- message_id { 'factory@id_com_1' }
- body { 'some message 123' }
- internal { false }
- sender { Ticket::Article::Sender.find_by(name: sender_name) }
- type { Ticket::Article::Type.find_by(name: type_name) }
- updated_by_id { 1 }
- created_by_id { 1 }
- trait :inbound_email do
- transient do
- type_name { 'email' }
- sender_name { 'Customer' }
- end
- end
- trait :outbound_email do
- transient do
- type_name { 'email' }
- sender_name { 'Agent' }
- end
- from { ticket.group.name }
- to { "#{ticket.customer.fullname} <#{ticket.customer.email}>" }
- end
- trait :inbound_phone do
- transient do
- type_name { 'phone' }
- sender_name { 'Customer' }
- end
- end
- trait :outbound_phone do
- transient do
- type_name { 'phone' }
- sender_name { 'Agent' }
- end
- from { nil }
- to { ticket.customer.fullname }
- end
- trait :outbound_note do
- transient do
- type_name { 'note' }
- sender_name { 'Agent' }
- end
- from { ticket.group.name }
- end
- trait :internal_note do
- transient do
- type_name { 'note' }
- sender_name { 'Agent' }
- end
- from { ticket.group.name }
- internal { true }
- end
- trait :inbound_web do
- transient do
- type_name { 'web' }
- sender_name { 'Customer' }
- end
- end
- trait :outbound_web do
- transient do
- type_name { 'web' }
- sender_name { 'Agent' }
- end
- end
- factory :twitter_article do
- transient do
- type_name { 'twitter status' }
- sender_name { 'Agent' }
- end
- ticket factory: %i[twitter_ticket]
- subject { nil }
- body { Faker::Lorem.sentence }
- content_type { 'text/plain' }
- message_id { Faker::Number.unique.number(digits: 18) }
- after(:create) do |article, context|
- next if context.sender_name == 'Agent'
- context.ticket.title = article.body
- context.ticket.save!
- end
- trait :inbound do
- transient do
- sender_name { 'Customer' }
- username { Faker::Twitter.screen_name }
- sender_id { Faker::Number.unique.number(digits: 18) }
- recipient_id { Faker::Number.unique.number(digits: 19) }
- end
- from { "@#{username}" }
- to { "@#{ticket.preferences['channel_screen_name']}" }
- body { "#{to} #{Faker::Lorem.question}" }
- preferences do
- {
- twitter: {
- mention_ids: [recipient_id],
- geo: {},
- retweeted: false,
- possibly_sensitive: false,
- in_reply_to_user_id: recipient_id,
- place: {},
- retweet_count: 0,
- source: '<a href="https://mobile.twitter.com" rel="nofollow">Twitter Web App</a>',
- favorited: false,
- truncated: false
- },
- links: [
- {
- url: "https://twitter.com/_/status/#{sender_id}",
- target: '_blank',
- name: 'on Twitter',
- },
- ],
- }
- end
- end
- trait :outbound do
- transient do
- username { Faker::Twitter.screen_name }
- sender_id { Faker::Number.unique.number(digits: 18) }
- recipient_id { Faker::Number.unique.number(digits: 19) }
- end
- from { "@#{ticket.preferences['channel_screen_name']}" }
- to { "@#{username}" }
- body { "#{to} #{Faker::Lorem.sentence}" }
- in_reply_to { Faker::Number.unique.number(digits: 19) }
- preferences do
- {
- twitter: {
- mention_ids: [recipient_id],
- geo: {},
- retweeted: false,
- possibly_sensitive: false,
- in_reply_to_user_id: recipient_id,
- place: {},
- retweet_count: 0,
- source: '<a href="https://www.canva.com" rel="nofollow">Canva</a>',
- favorited: false,
- truncated: false
- },
- links: [
- {
- url: "https://twitter.com/_/status/#{sender_id}",
- target: '_blank',
- name: 'on Twitter',
- },
- ],
- }
- end
- end
- trait :reply do
- in_reply_to { Faker::Number.unique.number(digits: 19) }
- end
- end
- factory :twitter_dm_article do
- transient do
- type_name { 'twitter direct-message' }
- end
- ticket factory: %i[twitter_ticket]
- body { Faker::Lorem.sentence }
- trait :pending_delivery do
- transient do
- recipient { association :twitter_authorization }
- sender_id { Faker::Number.unique.number(digits: 10) }
- end
- from { ticket.owner.fullname }
- to { recipient.username }
- in_reply_to { Faker::Number.unique.number(digits: 19) }
- content_type { 'text/plain' }
- end
- trait :delivered do
- pending_delivery
- message_id { Faker::Number.unique.number(digits: 19) }
- preferences do
- {
- delivery_retry: 1,
- twitter: {
- recipient_id: recipient.uid,
- sender_id: sender_id
- },
- links: [
- {
- url: "https://twitter.com/messages/#{recipient.uid}-#{sender_id}",
- target: '_blank',
- name: 'on Twitter'
- }
- ],
- delivery_status_message: nil,
- delivery_status: 'success',
- delivery_status_date: Time.current
- }
- end
- end
- end
- factory :sms_article do
- inbound
- transient do
- type_name { 'sms' }
- end
- ticket factory: %i[sms_ticket]
- from { Faker::PhoneNumber.cell_phone_in_e164 }
- to { Faker::PhoneNumber.cell_phone_in_e164 }
- subject { nil }
- body { Faker::Lorem.sentence }
- message_id { Faker::Number.unique.number(digits: 19) }
- content_type { 'text/plain' }
- after(:create) do |article, context|
- next if context.sender_name == 'Agent'
- context.ticket.title = article.body
- context.ticket.preferences.tap do |p|
- p['sms'] = {
- originator: article.from,
- recipient: article.to,
- }
- end
- context.ticket.save!
- end
- trait :inbound do
- transient do
- sender_name { 'Customer' }
- end
- preferences do
- {
- channel_id: ticket.preferences['channel_id'],
- sms: {
- reference: message_id,
- },
- }
- end
- end
- trait :outbound do
- transient do
- sender_name { 'Agent' }
- end
- in_reply_to { Faker::Number.unique.number(digits: 19) }
- preferences do
- {
- delivery_retry: 1,
- delivery_status_message: nil,
- delivery_status: 'success',
- delivery_status_date: Time.current,
- }
- end
- end
- end
- factory :whatsapp_article do
- inbound
- transient do
- type_name { 'whatsapp message' }
- channel { Channel.find(ticket.preferences[:channel_id]) }
- from_phone_number { Faker::PhoneNumber.cell_phone_in_e164 }
- from_name { Faker::Name.unique.name }
- timestamp_incoming { Time.zone.now.to_i.to_s }
- end
- ticket factory: %i[whatsapp_ticket]
- to { "#{channel.options[:name]} (#{channel.options[:phone_number]})" }
- subject { nil }
- body { Faker::Lorem.sentence }
- content_type { 'text/plain' }
- before(:create) do |_article, context|
- next if context.sender_name == 'Agent' && context.ticket.preferences[:whatsapp].present?
- context.ticket.preferences.tap do |p|
- p['whatsapp'] = {
- from: {
- phone_number: context.from_phone_number.delete('+'),
- display_name: context.from_name,
- },
- timestamp_incoming: context.timestamp_incoming,
- }
- end
- context.ticket.title = "New WhatsApp message from #{context.from_name} (#{context.from_phone_number})"
- context.ticket.save!
- end
- trait :inbound do
- transient do
- sender_name { 'Customer' }
- end
- message_id { "wamid.#{Faker::Number.unique.number}" }
- from { "#{from_name} (#{from_phone_number})" }
- created_by_id { ticket.customer_id } # NB: influences the value for the from field!
- preferences do
- {
- whatsapp: {
- entry_id: channel[:options][:phone_number_id],
- message_id: message_id,
- }
- }
- end
- end
- trait :pending_delivery do
- transient do
- sender_name { 'Agent' }
- end
- preferences { {} }
- created_by_id { create(:agent).id } # NB: influences the value for the from field!
- in_reply_to { "wamid.#{Faker::Number.unique.number}" }
- end
- trait :outbound do
- pending_delivery
- message_id { "wamid.#{Faker::Number.unique.number}" }
- preferences do
- {
- delivery_retry: 1,
- whatsapp: {
- message_id:,
- },
- delivery_status_message: nil,
- delivery_status: 'success',
- delivery_status_date: Time.current,
- }
- end
- end
- trait :with_attachment_media_document do
- after(:create) do |article, _context|
- create(:store,
- object: article.class.name,
- o_id: article.id,
- data: Faker::Lorem.unique.sentence,
- filename: 'test.txt',
- preferences: { 'Content-Type' => 'text/plain' })
- article.preferences.tap do |prefs|
- prefs['whatsapp'] = {
- entry_id: Faker::Number.unique.number.to_s,
- message_id: "wamid.#{Faker::Number.unique.number}",
- type: 'document',
- media_id: Faker::Number.unique.number.to_s
- }
- end
- article.save!
- end
- end
- trait :with_media_error do
- after(:create) do |article, _context|
- article.preferences.tap do |prefs|
- prefs['whatsapp'] = {
- entry_id: Faker::Number.unique.number.to_s,
- message_id: "wamid.#{Faker::Number.unique.number}",
- type: 'document',
- media_id: Faker::Number.unique.number.to_s,
- media_error: true,
- }
- end
- article.save!
- end
- end
- end
- factory :telegram_article do
- inbound
- transient do
- type_name { 'telegram personal-message' }
- channel { Channel.find(ticket.preferences[:channel_id]) }
- username { Faker::Internet.username }
- end
- ticket factory: %i[telegram_ticket]
- to { "@#{channel[:options][:bot][:username]}" }
- subject { nil }
- body { Faker::Lorem.sentence }
- message_id { "#{Faker::Number.unique.decimal(l_digits: 1, r_digits: 10)}@telegram" }
- content_type { 'text/plain' }
- after(:create) do |article, context|
- next if context.sender_name == 'Agent'
- context.ticket.title = article.body
- context.ticket.preferences.tap do |p|
- p['telegram'] = {
- bid: context.channel[:options][:bot][:id],
- chat_id: (article.preferences[:telegram] && article.preferences[:telegram][:chat_id]) || Faker::Number.unique.number(digits: 10),
- }
- end
- context.ticket.save!
- end
- trait :inbound do
- transient do
- sender_name { 'Customer' }
- end
- created_by_id { ticket.customer_id } # NB: influences the value for the from field!
- preferences do
- {
- message: {
- created_at: Time.current.to_i,
- message_id: message_id,
- from: ActionController::Parameters.new(
- 'id' => Faker::Number.unique.number,
- 'is_bot' => false,
- 'first_name' => Faker::Name.unique.first_name,
- 'last_name' => Faker::Name.unique.last_name,
- 'username' => username,
- 'language_code' => 'en',
- ),
- },
- update_id: Faker::Number.unique.number(digits: 8),
- }
- end
- end
- trait :outbound do
- transient do
- sender_name { 'Agent' }
- end
- to { "@#{username}" }
- created_by_id { create(:agent).id } # NB: influences the value for the from field!
- in_reply_to { "#{Faker::Number.unique.decimal(l_digits: 1, r_digits: 10)}@telegram" }
- preferences do
- {
- delivery_retry: 1,
- telegram: {
- date: Time.current.to_i,
- from_id: Faker::Number.unique.number(digits: 10),
- chat_id: Faker::Number.unique.number(digits: 10),
- message_id: Faker::Number.unique.number,
- },
- delivery_status_message: nil,
- delivery_status: 'success',
- delivery_status_date: Time.current,
- }
- end
- end
- end
- factory :facebook_article do
- inbound
- transient do
- channel { Channel.find(ticket.preferences[:channel_id]) }
- post_id { Faker::Number.unique.number(digits: 15) }
- permalink_url { "https://www.facebook.com/#{channel[:options][:pages][0][:id]}/posts/#{post_id}/?comment_id=#{post_id}" }
- end
- ticket factory: %i[facebook_ticket]
- subject { nil }
- body { Faker::Lorem.sentence }
- message_id { "#{Faker::Number.unique.number(digits: 16)}_#{Faker::Number.unique.number(digits: 15)}" }
- content_type { 'text/plain' }
- after(:create) do |article, context|
- next if context.sender_name == 'Agent'
- context.ticket.title = article.body
- context.ticket.preferences.tap do |p|
- p['channel_fb_object_id'] = context.post_id,
- p['facebook'] = {
- permalink_url: context.permalink_url,
- }
- end
- context.ticket.save!
- end
- trait :inbound do
- transient do
- type_name { 'facebook feed post' }
- sender_name { 'Customer' }
- end
- from { ticket.customer.fullname }
- to { channel[:options][:pages][0][:name] }
- preferences do
- {
- links: [
- {
- url: permalink_url,
- target: '_blank',
- name: 'on Facebook',
- },
- ],
- }
- end
- end
- trait :outbound do
- transient do
- type_name { 'facebook feed comment' }
- sender_name { 'Agent' }
- end
- from { channel[:options][:pages][0][:name] }
- to { ticket.customer.fullname }
- in_reply_to { "#{Faker::Number.unique.number(digits: 16)}_#{Faker::Number.unique.number(digits: 15)}" }
- preferences do
- {
- delivery_retry: 1,
- delivery_status_message: nil,
- delivery_status: 'success',
- delivery_status_date: Time.current,
- }
- end
- end
- end
- trait :with_attachment do
- transient do
- attachment { File.open('spec/fixtures/files/upload/hello_world.txt') }
- end
- after(:create) do |article, context|
- create(:store,
- object: article.class.name,
- o_id: article.id,
- data: context.attachment.read,
- filename: File.basename(context.attachment.path),
- preferences: {})
- end
- end
- trait :with_prepended_attachment do
- transient do
- attachment { File.open('spec/fixtures/files/upload/hello_world.txt') }
- override_content_type { nil }
- attachments_count { 1 }
- end
- after(:build) do |article, context|
- filename = File.basename(context.attachment.path)
- content_type = context.override_content_type || MIME::Types.type_for(filename).first&.content_type
- attachments = []
- context.attachments_count.times do
- attachments << create(:store,
- data: context.attachment.read,
- filename: filename,
- preferences: { 'Content-Type' => content_type })
- end
- article.attachments = attachments
- end
- end
- end
- end
|