# Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ class Service::Ticket::Update < Service::BaseWithCurrentUser include Service::Concerns::HandlesCoreWorkflow def execute(ticket:, ticket_data:, skip_validators: nil, macro: nil) Pundit.authorize current_user, ticket, :update? set_core_workflow_information(ticket_data, ::Ticket, 'edit') article_data = ticket_data.delete(:article) validate!(current_user, ticket, ticket_data, article_data, skip_validators) save_ticket!(ticket, ticket_data, article_data, macro) ticket.reload end private def save_ticket!(ticket, ticket_data, article_data, macro) ticket.with_lock do if macro save_ticket_attributes_and_apply_macro!(ticket, ticket_data, article_data, macro) else save_ticket_attributes!(ticket, ticket_data, article_data) end end end def save_ticket_attributes!(ticket, ticket_data, article_data) ticket.update!(ticket_data) create_article(ticket, article_data) end def save_ticket_attributes_and_apply_macro!(ticket, ticket_data, article_data, macro) ticket.assign_attributes(ticket_data) ticket.perform_changes(macro, 'macro', ticket, current_user.id) do |object, _save_needed| object.save! create_article(ticket, article_data) end end def create_article(ticket, article_data) return if article_data.blank? preprocess_article_data! article_data Service::Ticket::Article::Create .new(current_user: current_user) .execute(article_data: article_data, ticket: ticket) end # Desktop UI supplies this data from frontend # Mobile UI leaves this processing for GraphQL def preprocess_article_data!(article_input) article_input[:from] = current_user.fullname end def validate!(user, ticket, ticket_data, article_data, skip_validators) Service::Ticket::Update::Validator.new(user:, ticket:, ticket_data:, article_data:, skip_validators:).validate! end end