مراية لـ
https://github.com/postalserver/postal.git
تم المزامنة 2025-11-30 21:32:30 +00:00
refactor: remove moonrope but maintain legacy API actions (#2889)
هذا الالتزام موجود في:
133
app/controllers/legacy_api/base_controller.rb
Normal file
133
app/controllers/legacy_api/base_controller.rb
Normal file
@@ -0,0 +1,133 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
module LegacyAPI
|
||||
# The Legacy API is the Postal v1 API which existed from the start with main
|
||||
# aim of allowing e-mails to sent over HTTP rather than SMTP. The API itself
|
||||
# did not feature much functionality. This API was implemented using Moonrope
|
||||
# which was a self documenting API tool, however, is now no longer maintained.
|
||||
# In light of that, these controllers now implement the same functionality as
|
||||
# the original Moonrope API without the actual requirement to use any of the
|
||||
# Moonrope components.
|
||||
#
|
||||
# Important things to note about the API:
|
||||
#
|
||||
# * Moonrope allow params to be provided as JSON in the body of the request
|
||||
# along with the application/json content type. It also allowed for params
|
||||
# to be sent in the 'params' parameter when using the
|
||||
# application/x-www-form-urlencoded content type. Both methods are supported.
|
||||
#
|
||||
# * Authentication is performed using a X-Server-API-Key variable.
|
||||
#
|
||||
# * The method used to make the request is not important. Most clients use POST
|
||||
# but other methods should be supported. The routing for this legacvy
|
||||
# API supports GET, POST, PUT and PATCH.
|
||||
#
|
||||
# * The status code for responses will always be 200 OK. The actual status of
|
||||
# a request is determined by the value of the 'status' attribute in the
|
||||
# returned JSON.
|
||||
class BaseController < ActionController::Base
|
||||
|
||||
skip_before_action :set_browser_id
|
||||
skip_before_action :verify_authenticity_token
|
||||
|
||||
before_action :start_timer
|
||||
before_action :authenticate_as_server
|
||||
|
||||
private
|
||||
|
||||
# The Moonrope API spec allows for parameters to be provided in the body
|
||||
# along with the application/json content type or they can be provided,
|
||||
# as JSON, in the 'params' parameter when used with the
|
||||
# application/x-www-form-urlencoded content type. This legacy API needs
|
||||
# support both options for maximum compatibility.
|
||||
#
|
||||
# @return [Hash]
|
||||
def api_params
|
||||
if request.headers["content-type"] =~ /\Aapplication\/json/
|
||||
return params.to_unsafe_hash
|
||||
end
|
||||
|
||||
if params["params"].present?
|
||||
return JSON.parse(params["params"])
|
||||
end
|
||||
|
||||
{}
|
||||
end
|
||||
|
||||
# The API returns a length of time to complete a request. We'll start
|
||||
# a timer when the request starts and then use this method to calculate
|
||||
# the time taken to complete the request.
|
||||
#
|
||||
# @return [void]
|
||||
def start_timer
|
||||
@start_time = Time.now.to_f
|
||||
end
|
||||
|
||||
# The only method available to authenticate to the legacy API is using a
|
||||
# credential from the server itself. This method will attempt to find
|
||||
# that credential from the X-Server-API-Key header and will set the
|
||||
# current_credential instance variable if a token is valid. Otherwise it
|
||||
# will render an error to halt execution.
|
||||
#
|
||||
# @return [void]
|
||||
def authenticate_as_server
|
||||
key = request.headers["X-Server-API-Key"]
|
||||
if key.blank?
|
||||
render_error "AccessDenied",
|
||||
message: "Must be authenticated as a server."
|
||||
return
|
||||
end
|
||||
|
||||
credential = Credential.where(type: "API", key: key).first
|
||||
if credential.nil?
|
||||
render_error "InvalidServerAPIKey",
|
||||
message: "The API token provided in X-Server-API-Key was not valid.",
|
||||
token: key
|
||||
return
|
||||
end
|
||||
|
||||
if credential.server.suspended?
|
||||
render_error "ServerSuspended"
|
||||
return
|
||||
end
|
||||
|
||||
credential.use
|
||||
@current_credential = credential
|
||||
end
|
||||
|
||||
# Render a successful response to the client
|
||||
#
|
||||
# @param [Hash] data
|
||||
# @return [void]
|
||||
def render_success(data)
|
||||
render json: { status: "success",
|
||||
time: (Time.now.to_f - @start_time).round(3),
|
||||
flags: {},
|
||||
data: data }
|
||||
end
|
||||
|
||||
# Render an error response to the client
|
||||
#
|
||||
# @param [String] code
|
||||
# @param [Hash] data
|
||||
# @return [void]
|
||||
def render_error(code, data = {})
|
||||
render json: { status: "error",
|
||||
time: (Time.now.to_f - @start_time).round(3),
|
||||
flags: {},
|
||||
data: data.merge(code: code) }
|
||||
end
|
||||
|
||||
# Render a parameter error response to the client
|
||||
#
|
||||
# @param [String] message
|
||||
# @return [void]
|
||||
def render_parameter_error(message)
|
||||
render json: { status: "parameter-error",
|
||||
time: (Time.now.to_f - @start_time).round(3),
|
||||
flags: {},
|
||||
data: { message: message } }
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
140
app/controllers/legacy_api/messages_controller.rb
Normal file
140
app/controllers/legacy_api/messages_controller.rb
Normal file
@@ -0,0 +1,140 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
module LegacyAPI
|
||||
class MessagesController < BaseController
|
||||
|
||||
# Returns details about a given message
|
||||
#
|
||||
# URL: /api/v1/messages/message
|
||||
#
|
||||
# Parameters: id => REQ: The ID of the message
|
||||
# _expansions => An array of types of details t
|
||||
# to return
|
||||
#
|
||||
# Response: A hash containing message information
|
||||
# OR an error if the message does not exist.
|
||||
#
|
||||
def message
|
||||
if api_params["id"].blank?
|
||||
render_parameter_error "`id` parameter is required but is missing"
|
||||
return
|
||||
end
|
||||
|
||||
message = @current_credential.server.message(api_params["id"])
|
||||
message_hash = { id: message.id, token: message.token }
|
||||
expansions = api_params["_expansions"]
|
||||
|
||||
if expansions.include?("status")
|
||||
message_hash[:status] = {
|
||||
status: message.status,
|
||||
last_delivery_attempt: message.last_delivery_attempt&.to_f,
|
||||
held: message.held,
|
||||
hold_expiry: message.hold_expiry&.to_f
|
||||
}
|
||||
end
|
||||
|
||||
if expansions.include?("details")
|
||||
message_hash[:details] = {
|
||||
rcpt_to: message.rcpt_to,
|
||||
mail_from: message.mail_from,
|
||||
subject: message.subject,
|
||||
message_id: message.message_id,
|
||||
timestamp: message.timestamp.to_f,
|
||||
direction: message.scope,
|
||||
size: message.size,
|
||||
bounce: message.bounce,
|
||||
bounce_for_id: message.bounce_for_id,
|
||||
tag: message.tag,
|
||||
received_with_ssl: message.received_with_ssl
|
||||
}
|
||||
end
|
||||
|
||||
if expansions.include?("inspection")
|
||||
message_hash[:inspection] = {
|
||||
inspected: message.inspected,
|
||||
spam: message.spam,
|
||||
spam_score: message.spam_score.to_f,
|
||||
threat: message.threat,
|
||||
threat_details: message.threat_details
|
||||
}
|
||||
end
|
||||
|
||||
if expansions.include?("plain_body")
|
||||
message_hash[:plain_body] = message.plain_body
|
||||
end
|
||||
|
||||
if expansions.include?("html_body")
|
||||
message_hash[:html_body] = message.html_body
|
||||
end
|
||||
|
||||
if expansions.include?("attachments")
|
||||
message_hash[:attachments] = message.attachments.map do |attachment|
|
||||
{
|
||||
filename: attachment.filename.to_s,
|
||||
content_type: attachment.mime_type,
|
||||
data: Base64.encode64(attachment.body.to_s),
|
||||
size: attachment.body.to_s.bytesize,
|
||||
hash: Digest::SHA1.hexdigest(attachment.body.to_s)
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
if expansions.include?("headers")
|
||||
message_hash[:headers] = message.headers
|
||||
end
|
||||
|
||||
if expansions.include?("raw_message")
|
||||
message_hash[:raw_message] = Base64.encode64(message.raw_message)
|
||||
end
|
||||
|
||||
if expansions.include?("activity_entries")
|
||||
message_hash[:activity_entries] = {
|
||||
loads: message.loads,
|
||||
clicks: message.clicks
|
||||
}
|
||||
end
|
||||
|
||||
render_success message_hash
|
||||
rescue Postal::MessageDB::Message::NotFound
|
||||
render_error "MessageNotFound",
|
||||
message: "No message found matching provided ID",
|
||||
id: api_params["id"]
|
||||
end
|
||||
|
||||
# Returns all the deliveries for a given message
|
||||
#
|
||||
# URL: /api/v1/messages/deliveries
|
||||
#
|
||||
# Parameters: id => REQ: The ID of the message
|
||||
#
|
||||
# Response: A array of hashes containing delivery information
|
||||
# OR an error if the message does not exist.
|
||||
#
|
||||
def deliveries
|
||||
if api_params["id"].blank?
|
||||
render_parameter_error "`id` parameter is required but is missing"
|
||||
return
|
||||
end
|
||||
|
||||
message = @current_credential.server.message(api_params["id"])
|
||||
deliveries = message.deliveries.map do |d|
|
||||
{
|
||||
id: d.id,
|
||||
status: d.status,
|
||||
details: d.details,
|
||||
output: d.output&.strip,
|
||||
sent_with_ssl: d.sent_with_ssl,
|
||||
log_id: d.log_id,
|
||||
time: d.time&.to_f,
|
||||
timestamp: d.timestamp.to_f
|
||||
}
|
||||
end
|
||||
render_success deliveries
|
||||
rescue Postal::MessageDB::Message::NotFound
|
||||
render_error "MessageNotFound",
|
||||
message: "No message found matching provided ID",
|
||||
id: api_params["id"]
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
135
app/controllers/legacy_api/send_controller.rb
Normal file
135
app/controllers/legacy_api/send_controller.rb
Normal file
@@ -0,0 +1,135 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
module LegacyAPI
|
||||
class SendController < BaseController
|
||||
|
||||
ERROR_MESSAGES = {
|
||||
"NoRecipients" => "There are no recipients defined to receive this message",
|
||||
"NoContent" => "There is no content defined for this e-mail",
|
||||
"TooManyToAddresses" => "The maximum number of To addresses has been reached (maximum 50)",
|
||||
"TooManyCCAddresses" => "The maximum number of CC addresses has been reached (maximum 50)",
|
||||
"TooManyBCCAddresses" => "The maximum number of BCC addresses has been reached (maximum 50)",
|
||||
"FromAddressMissing" => "The From address is missing and is required",
|
||||
"UnauthenticatedFromAddress" => "The From address is not authorised to send mail from this server",
|
||||
"AttachmentMissingName" => "An attachment is missing a name",
|
||||
"AttachmentMissingData" => "An attachment is missing data"
|
||||
}.freeze
|
||||
|
||||
# Send a message with the given options
|
||||
#
|
||||
# URL: /api/v1/send/message
|
||||
#
|
||||
# Parameters: to => REQ: An array of emails addresses
|
||||
# cc => An array of email addresses to CC
|
||||
# bcc => An array of email addresses to BCC
|
||||
# from => The name/email to send the email from
|
||||
# sender => The name/email of the 'Sender'
|
||||
# reply_to => The name/email of the 'Reply-to'
|
||||
# plain_body => The plain body
|
||||
# html_body => The HTML body
|
||||
# bounce => Is this message a bounce?
|
||||
# tag => A custom tag to add to the message
|
||||
# custom_headers => A hash of custom headers
|
||||
# attachments => An array of attachments
|
||||
# (name, content_type and data (base64))
|
||||
#
|
||||
# Response: A array of hashes containing message information
|
||||
# OR an error if there is an issue sending the message
|
||||
#
|
||||
def message
|
||||
attributes = {}
|
||||
attributes[:to] = api_params["to"]
|
||||
attributes[:cc] = api_params["cc"]
|
||||
attributes[:bcc] = api_params["bcc"]
|
||||
attributes[:from] = api_params["from"]
|
||||
attributes[:sender] = api_params["sender"]
|
||||
attributes[:subject] = api_params["subject"]
|
||||
attributes[:reply_to] = api_params["reply_to"]
|
||||
attributes[:plain_body] = api_params["plain_body"]
|
||||
attributes[:html_body] = api_params["html_body"]
|
||||
attributes[:bounce] = api_params["bounce"] ? true : false
|
||||
attributes[:tag] = api_params["tag"]
|
||||
attributes[:custom_headers] = api_params["headers"] if api_params["headers"]
|
||||
attributes[:attachments] = []
|
||||
|
||||
(api_params["attachments"] || []).each do |attachment|
|
||||
next unless attachment.is_a?(Hash)
|
||||
|
||||
attributes[:attachments] << { name: attachment["name"], content_type: attachment["content_type"], data: attachment["data"], base64: true }
|
||||
end
|
||||
|
||||
message = OutgoingMessagePrototype.new(@current_credential.server, request.ip, "api", attributes)
|
||||
message.credential = @current_credential
|
||||
if message.valid?
|
||||
result = message.create_messages
|
||||
render_success message_id: message.message_id, messages: result
|
||||
else
|
||||
render_error message.errors.first, message: ERROR_MESSAGES[message.errors.first]
|
||||
end
|
||||
end
|
||||
|
||||
# Send a message by providing a raw message
|
||||
#
|
||||
# URL: /api/v1/send/raw
|
||||
#
|
||||
# Parameters: rcpt_to => REQ: An array of email addresses to send
|
||||
# the message to
|
||||
# mail_from => REQ: the address to send the email from
|
||||
# data => REQ: base64-encoded mail data
|
||||
#
|
||||
# Response: A array of hashes containing message information
|
||||
# OR an error if there is an issue sending the message
|
||||
#
|
||||
def raw
|
||||
unless api_params["rcpt_to"].is_a?(Array)
|
||||
render_parameter_error "`rcpt_to` parameter is required but is missing"
|
||||
return
|
||||
end
|
||||
|
||||
if api_params["mail_from"].blank?
|
||||
render_parameter_error "`mail_from` parameter is required but is missing"
|
||||
return
|
||||
end
|
||||
|
||||
if api_params["data"].blank?
|
||||
render_parameter_error "`data` parameter is required but is missing"
|
||||
return
|
||||
end
|
||||
|
||||
# Decode the raw message
|
||||
raw_message = Base64.decode64(api_params["data"])
|
||||
|
||||
# Parse through mail to get the from/sender headers
|
||||
mail = Mail.new(raw_message.split("\r\n\r\n", 2).first)
|
||||
from_headers = { "from" => mail.from, "sender" => mail.sender }
|
||||
authenticated_domain = @current_credential.server.find_authenticated_domain_from_headers(from_headers)
|
||||
|
||||
# If we're not authenticated, don't continue
|
||||
if authenticated_domain.nil?
|
||||
render_error "UnauthenticatedFromAddress"
|
||||
return
|
||||
end
|
||||
|
||||
# Store the result ready to return
|
||||
result = { message_id: nil, messages: {} }
|
||||
if api_params["rcpt_to"].is_a?(Array)
|
||||
api_params["rcpt_to"].uniq.each do |rcpt_to|
|
||||
message = @current_credential.server.message_db.new_message
|
||||
message.rcpt_to = rcpt_to
|
||||
message.mail_from = api_params["mail_from"]
|
||||
message.raw_message = raw_message
|
||||
message.received_with_ssl = true
|
||||
message.scope = "outgoing"
|
||||
message.domain_id = authenticated_domain.id
|
||||
message.credential_id = @current_credential.id
|
||||
message.bounce = api_params["bounce"] ? true : false
|
||||
message.save
|
||||
result[:message_id] = message.message_id if result[:message_id].nil?
|
||||
result[:messages][rcpt_to] = { id: message.id, token: message.token }
|
||||
end
|
||||
end
|
||||
render_success result
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
المرجع في مشكلة جديدة
حظر مستخدم