## Description ConversationReplyMailer#parse_email calls Mail::Address.new(email_string).address without error handling. When an account's support_email contains a non-email string (e.g., "Smith Smith"), the mail gem raises Mail::Field::IncompleteParseError, crashing conversation transcript emails. This has caused 1,056 errors on Sentry (EXTERNAL-CHATINC-JX) since Feb 25, all from a single account that has a name stored in the support_email field instead of a valid email address. Closes https://linear.app/chatwoot/issue/CW-6687/mailfieldincompleteparseerror-mailaddresslist-can-not-parse-orsmith ## Type of change Please delete options that are not relevant. - [ ] Bug fix (non-breaking change which fixes an issue) ## Checklist: - [ ] My code follows the style guidelines of this project - [ ] I have performed a self-review of my code - [ ] I have commented on my code, particularly in hard-to-understand areas - [ ] I have made corresponding changes to the documentation - [ ] My changes generate no new warnings - [ ] I have added tests that prove my fix is effective or that my feature works - [ ] New and existing unit tests pass locally with my changes - [ ] Any dependent changes have been merged and published in downstream modules --------- Co-authored-by: Claude Opus 4.6 (1M context) <noreply@anthropic.com> Co-authored-by: Vishnu Narayanan <iamwishnu@gmail.com>
53 lines
1.2 KiB
Ruby
53 lines
1.2 KiB
Ruby
class Email::BaseBuilder
|
|
include EmailAddressParseable
|
|
|
|
pattr_initialize [:inbox!]
|
|
|
|
private
|
|
|
|
def channel
|
|
@channel ||= inbox.channel
|
|
end
|
|
|
|
def account
|
|
@account ||= inbox.account
|
|
end
|
|
|
|
def conversation
|
|
@conversation ||= message.conversation
|
|
end
|
|
|
|
def custom_sender_name
|
|
message&.sender&.available_name || I18n.t('conversations.reply.email.header.notifications')
|
|
end
|
|
|
|
def sender_name(sender_email)
|
|
# Friendly: <agent_name> from <business_name>
|
|
# Professional: <business_name>
|
|
if inbox.friendly?
|
|
I18n.t(
|
|
'conversations.reply.email.header.friendly_name',
|
|
sender_name: custom_sender_name,
|
|
business_name: business_name,
|
|
from_email: sender_email
|
|
)
|
|
else
|
|
I18n.t(
|
|
'conversations.reply.email.header.professional_name',
|
|
business_name: business_name,
|
|
from_email: sender_email
|
|
)
|
|
end
|
|
end
|
|
|
|
def business_name
|
|
inbox.business_name || inbox.sanitized_name
|
|
end
|
|
|
|
def account_support_email
|
|
# Parse the email to ensure it's in the correct format, the user
|
|
# can save it in the format "Name <email@domain.com>"
|
|
parse_email(account.support_email)
|
|
end
|
|
end
|