CtrlK
BlogDocsLog inGet started
Tessl Logo

postmark-inbound

Use when processing incoming emails with Postmark inbound webhooks — building reply-by-email, email-to-ticket, document extraction, or any workflow that receives and parses email.

91

1.71x
Quality

87%

Does it follow best practices?

Impact

96%

1.71x

Average score across 3 eval scenarios

SecuritybySnyk

Advisory

Suggest reviewing before use

SKILL.md
Quality
Evals
Security

Process Inbound Email with Postmark

Overview

Postmark's inbound processing parses incoming emails and delivers them as structured JSON to your webhook endpoint. This enables workflows like:

  • Reply-by-email — Threading replies back to conversations
  • Email-to-ticket — Converting emails into support tickets
  • Document extraction — Processing email attachments automatically
  • Command processing — Parsing structured data from emails
  • Forwarding/routing — Routing emails to different services based on content

How It Works

  1. Configure an inbound address or domain in your Postmark server
  2. Set webhook URL where Postmark will POST parsed email data
  3. Receive JSON — Postmark processes the raw email and delivers structured data
  4. Respond with 200 — Your endpoint must return HTTP 200 to acknowledge receipt
Sender → Email → Postmark → Parses email → POST JSON → Your webhook endpoint

Quick Start

  1. Set up inbound domain — Configure MX records or email forwarding for your domain
  2. Set webhook URL — In your Postmark server settings, set the Inbound webhook URL
  3. Build your endpoint — Create an HTTP POST handler that accepts the inbound JSON payload
  4. Return 200 — Always respond with HTTP 200 to confirm receipt

Error Handling and Retries

If your endpoint returns a non-200 status code, Postmark will automatically retry delivery up to 10 times over approximately 10.5 hours with escalating intervals:

RetryInterval After Previous Attempt
11 minute
25 minutes
310 minutes
410 minutes
510 minutes
615 minutes
730 minutes
81 hour
92 hours
106 hours

Important: A 403 response immediately stops all retries — Postmark interprets this as intentional rejection. After all retries are exhausted, the message is marked as "Failed" and appears as an "Inbound Error" in your activity page. You can manually retry failed messages via the API (PUT /messages/inbound/{messageid}/retry).

Inbound Configuration

Two setup options — MX record (recommended) or email forwarding. Constraints: one inbound stream per server, one domain per stream, one webhook URL per stream.

See references/inbound-setup.md for full DNS steps, forwarding caveats, retry schedule, and how to set your webhook URL.

Webhook Payload

Key fields in the JSON Postmark POSTs to your endpoint:

FieldDescription
FromSender email address
SubjectEmail subject line
MailboxHashThe + hash from the recipient address — primary routing mechanism
TextBodyFull plain text body (includes quoted replies)
StrippedTextReplyReply text only — quoted content stripped
HtmlBodyFull HTML body
AttachmentsArray of {Name, Content, ContentType, ContentLength, ContentID}
MessageIDUnique Postmark message identifier
HeadersAll email headers as [{Name, Value}]

See references/payload-structure.md for the full payload JSON, attachment fields, and header threading examples.

MailboxHash for Routing

Use + addressing to route emails to specific records or conversations:

support+ticket-456@yourdomain.com  →  MailboxHash: "ticket-456"
notifications+order-789@yourdomain.com → MailboxHash: "order-789"

This is the primary mechanism for threading replies back to conversations or routing to specific records.

Basic Endpoint

const express = require('express');
const app = express();
app.use(express.json({ limit: '50mb' }));

app.post('/webhooks/inbound', (req, res) => {
  const { From, Subject, MailboxHash, StrippedTextReply, TextBody } = req.body;

  if (MailboxHash) {
    // Threaded reply — parse the hash to find the related record
    const [type, id] = MailboxHash.split('-');
    console.log(`Reply for ${type} #${id} from ${From}`);
  } else {
    console.log(`New inbound email from ${From}: ${Subject}`);
  }

  // Always prefer StrippedTextReply for replies
  const replyText = StrippedTextReply || TextBody;

  res.sendStatus(200); // Must return 200
});

See references/handler-examples.md for Node.js, Python, attachment processing, reply-by-email, and async processing patterns.

Inbound Rules and Messages API

Block unwanted senders by address or domain, and query/retry processed messages via the API.

See references/inbound-api.md for inbound rules endpoints and the Messages API.

Common Mistakes

MistakeFix
Not returning HTTP 200Always respond 200 — even if you process asynchronously
Returning 403 accidentallyThis permanently stops retries for that message
Not parsing MailboxHashUse + addressing for routing — it's the primary threading mechanism
Using TextBody instead of StrippedTextReplyStrippedTextReply removes quoted content from replies
No size limit on body parserSet body parser limit to 50mb for messages with attachments
Slow webhook processingProcess async (queue the work) and respond 200 immediately
Ignoring ContentID on attachmentsAttachments with ContentID are inline images, not standalone files

Notes

  • One Inbound Stream per server — use separate servers for different inbound domains
  • Inbound webhook payloads can be large due to attachments — set appropriate body size limits
  • StrippedTextReply strips quoted content, giving you just the new reply text
  • The MailboxHash field is the portion after + in the recipient address — use it for routing
  • Headers array contains all original email headers for advanced processing
  • Same server can handle both inbound and outbound email
  • Inbound processing is separate from outbound — different streams, different configuration
Repository
ActiveCampaign/postmark-skills
Last updated
Created

Is this your skill?

If you maintain this skill, you can claim it as your own. Once claimed, you can manage eval scenarios, bundle related skills, attach documentation or rules, and ensure cross-agent compatibility.