123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- module Import
- module OTRS
- class Article
- include Import::Helper
- include Import::OTRS::Helper
- MAPPING = {
- TicketID: :ticket_id,
- ArticleID: :id,
- Body: :body,
- From: :from,
- To: :to,
- Cc: :cc,
- Subject: :subject,
- InReplyTo: :in_reply_to,
- MessageID: :message_id,
- # ReplyTo: :reply_to,
- # References: :references,
- ContentType: :content_type,
- ChangeTime: :updated_at,
- CreateTime: :created_at,
- ChangeBy: :updated_by_id,
- CreateBy: :created_by_id,
- # Support also the legacy key names (lower then OTRS 6)
- Changed: :updated_at,
- Created: :created_at,
- ChangedBy: :updated_by_id,
- CreatedBy: :created_by_id,
- }.freeze
- def initialize(article)
- initialize_article_sender_types
- initialize_article_types
- utf8_encode(article)
- import(article)
- end
- private
- def import(article)
- create_or_update(map(article))
- return if article['Attachments'].blank?
- Import::OTRS::Article::AttachmentFactory.import(
- attachments: article['Attachments'],
- local_article: @local_article
- )
- end
- def create_or_update(article)
- return if updated?(article)
- create(article)
- end
- def updated?(article)
- @local_article = ::Ticket::Article.find_by(id: article[:id])
- return false if !@local_article
- log "update Ticket::Article.find_by(id: #{article[:id]})"
- @local_article.update!(article)
- true
- end
- def create(article)
- log "add Ticket::Article.find_by(id: #{article[:id]})"
- @local_article = ::Ticket::Article.new(article)
- @local_article.id = article[:id]
- @local_article.save
- reset_primary_key_sequence('ticket_articles')
- rescue ActiveRecord::RecordNotUnique
- log "Ticket #{article[:ticket_id]} (article #{article[:id]}) is handled by another thead, skipping."
- end
- def map(article)
- mapped = map_default(article)
- map_content_type(mapped)
- mapped[:body] ||= ''
- mapped
- end
- def map_default(article)
- {
- created_by_id: 1,
- updated_by_id: 1,
- }
- .merge(from_mapping(article))
- .merge(article_type(article))
- .merge(article_sender_type(article))
- .merge(article_created_and_updated_by_id(article))
- end
- def map_content_type(mapped)
- # if no content type is set make sure to remove it
- # so Zammad can set the default content type
- mapped.delete(:content_type) if mapped[:content_type].blank?
- return mapped if !mapped[:content_type]
- mapped[:content_type].sub!(%r{[;,]\s?.+?$}, '')
- mapped
- end
- def article_type(article)
- @article_types[article['ArticleType']] || @article_types['note-internal']
- end
- def article_sender_type(article)
- {
- sender_id: @sender_type_id[article['SenderType']] || @sender_type_id['note-internal']
- }
- end
- def article_created_and_updated_by_id(article)
- return {} if article['SenderType'] != 'customer'
- created_by_id = get_article_created_by_id(article)
- return {} if get_article_created_by_id(article) != 1
- return {} if article['From'].blank?
- user = Import::OTRS::ArticleCustomer.find(article)
- return {} if !user
- mapping = {
- created_by_id: user.id,
- }
- updated_by_id = get_article_updated_by_id(article)
- if !updated_by_id || updated_by_id == created_by_id
- mapping[:updated_by_id] = user.id
- end
- mapping
- end
- def get_article_created_by_id(article)
- return article['CreatedBy'].to_i if article['CreatedBy'].present?
- article['CreateBy'].to_i
- end
- def get_article_updated_by_id(article)
- return article['UpdatedBy'].to_i if article['UpdatedBy'].present?
- return article['UpdateBy'].to_i if article['UpdateBy'].present?
- nil
- end
- def initialize_article_sender_types
- @sender_type_id = {
- 'customer' => article_sender_type_id_lookup('Customer'),
- 'agent' => article_sender_type_id_lookup('Agent'),
- 'system' => article_sender_type_id_lookup('System'),
- }
- end
- def article_sender_type_id_lookup(name)
- ::Ticket::Article::Sender.find_by(name: name).id
- end
- def initialize_article_types
- @article_types = {
- 'email-external' => {
- type_id: article_type_id_lookup('email'),
- internal: false
- },
- 'email-internal' => {
- type_id: article_type_id_lookup('email'),
- internal: true
- },
- 'note-external' => {
- type_id: article_type_id_lookup('note'),
- internal: false
- },
- 'note-internal' => {
- type_id: article_type_id_lookup('note'),
- internal: true
- },
- 'phone' => {
- type_id: article_type_id_lookup('phone'),
- internal: false
- },
- 'webrequest' => {
- type_id: article_type_id_lookup('web'),
- internal: false
- },
- }
- end
- def article_type_id_lookup(name)
- ::Ticket::Article::Type.lookup(name: name).id
- end
- end
- end
- end
|