123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- module CanCloneAttachments
- extend ActiveSupport::Concern
- =begin
- clone existing attachments of article to the target object
- article_parent = Ticket::Article.find(123)
- article_new = Ticket::Article.find(456)
- attached_attachments = article_parent.clone_attachments(article_new.class.name, article_new.id, only_attached_attachments: true)
- inline_attachments = article_parent.clone_attachments(article_new.class.name, article_new.id, only_inline_attachments: true)
- returns
- [attachment1, attachment2, ...]
- =end
- def clone_attachments(object_type, object_id, options = {})
- existing_attachments = Store.list(
- object: object_type,
- o_id: object_id,
- )
- is_html_content = content_type.present? && content_type =~ %r{text/html}i
- attachments
- .select do |elem|
- next if elem.preferences['content-alternative'] == true
- # only_attached_attachments mode is used by apply attached attachments to forwared article
- if options[:only_attached_attachments] == true && is_html_content
- content_id = elem.preferences['Content-ID'] || elem.preferences['content_id']
- next if content_id.present? && body.present? && body.match?(%r{#{Regexp.quote(content_id)}}i)
- end
- # only_inline_attachments mode is used when quoting HTML mail with #{article.body_as_html}
- if options[:only_inline_attachments] == true
- next if !is_html_content
- next if body.blank?
- content_disposition = elem.preferences['Content-Disposition'] || elem.preferences['content_disposition']
- next if content_disposition.present? && content_disposition.exclude?('inline')
- content_id = elem.preferences['Content-ID'] || elem.preferences['content_id']
- next if content_id.blank?
- next if !body.match?(%r{#{Regexp.quote(content_id)}}i)
- end
- next if existing_attachments.any? do |existing_attachment|
- existing_attachment.filename == elem.filename && existing_attachment.size == elem.size
- end
- true
- end
- .map do |elem|
- Store.create!(
- object: object_type,
- o_id: object_id,
- data: elem.content,
- filename: elem.filename,
- preferences: elem.preferences,
- )
- end
- end
- def attach_upload_cache(form_id, source_object_name: 'UploadCache')
- attachments
- .reject(&:inline?)
- .each { |attachment| Store.remove_item(attachment.id) }
- Store
- .list(object: source_object_name, o_id: form_id)
- .reject(&:inline?)
- .map do |old_attachment|
- Store.create!(
- object: self.class.name,
- o_id: id,
- data: old_attachment.content,
- filename: old_attachment.filename,
- preferences: old_attachment.preferences,
- )
- end
- end
- end
|