Skip to content

Commit 3e7d4f8

Browse files
committed
Feature: Add support for sending inline attachments via HTTP API (#399)
Optional settings for Attachment ContentID & ContentType
1 parent 22cae16 commit 3e7d4f8

File tree

3 files changed

+47
-11
lines changed

3 files changed

+47
-11
lines changed

server/apiv1/send.go

Lines changed: 23 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -80,23 +80,33 @@ type SendRequest struct {
8080
Subject string
8181

8282
// Message body (text)
83-
// example: This is the text body
83+
// example: Mailpit is awesome!
8484
Text string
8585

8686
// Message body (HTML)
87-
// example: <p style="font-family: arial">Mailpit is <b>awesome</b>!</p>
87+
// example: <div style="text-align:center"><p style="font-family: arial; font-size: 24px;">Mailpit is <b>awesome</b>!</p><p><img src="cid:mailpit-logo" /></p></div>
8888
HTML string
8989

9090
// Attachments
9191
Attachments []struct {
9292
// Base64-encoded string of the file content
9393
// required: true
94-
// example: VGhpcyBpcyBhIHBsYWluIHRleHQgYXR0YWNobWVudA==
94+
// example: iVBORw0KGgoAAAANSUhEUgAAAEEAAAA8CAMAAAAOlSdoAAAACXBIWXMAAAHrAAAB6wGM2bZBAAAAS1BMVEVHcEwRfnUkZ2gAt4UsSF8At4UtSV4At4YsSV4At4YsSV8At4YsSV4At4YsSV4sSV4At4YsSV4At4YtSV4At4YsSV4At4YtSV8At4YsUWYNAAAAGHRSTlMAAwoXGiktRE5dbnd7kpOlr7zJ0d3h8PD8PCSRAAACWUlEQVR42pXT4ZaqIBSG4W9rhqQYocG+/ys9Y0Z0Br+x3j8zaxUPewFh65K+7yrIMeIY4MT3wPfEJCidKXEMnLaVkxDiELiMz4WEOAZSFghxBIypCOlKiAMgXfIqTnBgSm8CIQ6BImxEUxEckClVQiHGj4Ba4AQHikAIClwTE9KtIghAhUJwoLkmLnCiAHJLRKgIMsEtVUKbBUIwoAg2C4QgQBE6l4VCnApBgSKYLLApCnCa0+96AEMW2BQcmC+Pr3nfp7o5Exy49gIADcIqUELGfeA+bp93LmAJp8QJoEcN3C7NY3sbVANixMyI0nku20/n5/ZRf3KI2k6JEDWQtxcbdGuAqu3TAXG+/799Oyyas1B1MnMiA+XyxHp9q0PUKGPiRAau1fZbLRZV09wZcT8/gHk8QQAxXn8VgaDqcUmU6O/r28nbVwXAqca2mRNtPAF5+zoP2MeN9Fy4NgC6RfcbgE7XITBRYTtOE3U3C2DVff7pk+PkUxgAbvtnPXJaD6DxulMLwOhPS/M3MQkgg1ZFrIXnmfaZoOfpKiFgzeZD/WuKqQEGrfJYkyWf6vlG3xUgTuscnkNkQsb599q124kdpMUjCa/XARHs1gZymVtGt3wLkiFv8rUgTxitYCex5EVGec0Y9VmoDTFBSQte2TfXGXlf7hbdaUM9Sk7fisEN9qfBBTK+FZcvM9fQSdkl2vj4W2oX/bRogO3XasiNH7R0eW7fgRM834ImTg+Lg6BEnx4vz81rhr+MYPBBQg1v8GndEOrthxaCTxNAOut8WKLGZQl+MPz88Q9tAO/hVuSeqQAAAABJRU5ErkJggg==
9595
Content string
9696
// Filename
9797
// required: true
98-
// example: AttachedFile.txt
98+
// example: mailpit.png
9999
Filename string
100+
// Optional Content Type for the the attachment.
101+
// If this field is not set (or empty) then the content type is automatically detected.
102+
// required: false
103+
// example: image/png
104+
ContentType string
105+
// Optional Content-ID (`cid`) for attachment.
106+
// If this field is set then the file is attached inline.
107+
// required: false
108+
// example: mailpit-logo
109+
ContentID string
100110
}
101111

102112
// Mailpit tags
@@ -269,9 +279,15 @@ func (d SendRequest) Send(remoteAddr string) (string, error) {
269279
if err != nil {
270280
return "", fmt.Errorf("error decoding base64 attachment \"%s\": %s", a.Filename, err.Error())
271281
}
272-
273-
mimeType := http.DetectContentType(b)
274-
msg = msg.AddAttachment(b, mimeType, a.Filename)
282+
contentType := http.DetectContentType(b)
283+
if a.ContentType != "" {
284+
contentType = a.ContentType
285+
}
286+
if a.ContentID != "" {
287+
msg = msg.AddInline(b, contentType, a.Filename, a.ContentID)
288+
} else {
289+
msg = msg.AddAttachment(b, contentType, a.Filename)
290+
}
275291
}
276292
}
277293

server/server_test.go

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -243,6 +243,12 @@ func TestAPIv1Send(t *testing.T) {
243243
{
244244
"Content": "VGhpcyBpcyBhIHBsYWluIHRleHQgYXR0YWNobWVudA==",
245245
"Filename": "Attached File.txt"
246+
},
247+
{
248+
"Content": "iVBORw0KGgoAAAANSUhEUgAAAEEAAAA8CAMAAAAOlSdoAAAACXBIWXMAAAHrAAAB6wGM2bZBAAAAS1BMVEVHcEwRfnUkZ2gAt4UsSF8At4UtSV4At4YsSV4At4YsSV8At4YsSV4At4YsSV4sSV4At4YsSV4At4YtSV4At4YsSV4At4YtSV8At4YsUWYNAAAAGHRSTlMAAwoXGiktRE5dbnd7kpOlr7zJ0d3h8PD8PCSRAAACWUlEQVR42pXT4ZaqIBSG4W9rhqQYocG+/ys9Y0Z0Br+x3j8zaxUPewFh65K+7yrIMeIY4MT3wPfEJCidKXEMnLaVkxDiELiMz4WEOAZSFghxBIypCOlKiAMgXfIqTnBgSm8CIQ6BImxEUxEckClVQiHGj4Ba4AQHikAIClwTE9KtIghAhUJwoLkmLnCiAHJLRKgIMsEtVUKbBUIwoAg2C4QgQBE6l4VCnApBgSKYLLApCnCa0+96AEMW2BQcmC+Pr3nfp7o5Exy49gIADcIqUELGfeA+bp93LmAJp8QJoEcN3C7NY3sbVANixMyI0nku20/n5/ZRf3KI2k6JEDWQtxcbdGuAqu3TAXG+/799Oyyas1B1MnMiA+XyxHp9q0PUKGPiRAau1fZbLRZV09wZcT8/gHk8QQAxXn8VgaDqcUmU6O/r28nbVwXAqca2mRNtPAF5+zoP2MeN9Fy4NgC6RfcbgE7XITBRYTtOE3U3C2DVff7pk+PkUxgAbvtnPXJaD6DxulMLwOhPS/M3MQkgg1ZFrIXnmfaZoOfpKiFgzeZD/WuKqQEGrfJYkyWf6vlG3xUgTuscnkNkQsb599q124kdpMUjCa/XARHs1gZymVtGt3wLkiFv8rUgTxitYCex5EVGec0Y9VmoDTFBSQte2TfXGXlf7hbdaUM9Sk7fisEN9qfBBTK+FZcvM9fQSdkl2vj4W2oX/bRogO3XasiNH7R0eW7fgRM834ImTg+Lg6BEnx4vz81rhr+MYPBBQg1v8GndEOrthxaCTxNAOut8WKLGZQl+MPz88Q9tAO/hVuSeqQAAAABJRU5ErkJggg==",
249+
"Filename": "logo.png",
250+
"ContentID": "inline-cid",
251+
"ContentType": "overridden/type"
246252
}
247253
],
248254
"ReplyTo": [
@@ -294,6 +300,10 @@ func TestAPIv1Send(t *testing.T) {
294300
assertEqual(t, `Tag 1,Tag 2`, strings.Join(msg.Tags, ","), "wrong Tags")
295301
assertEqual(t, 1, len(msg.Attachments), "wrong Attachment count")
296302
assertEqual(t, `Attached File.txt`, msg.Attachments[0].FileName, "wrong Attachment name")
303+
assertEqual(t, `text/plain`, msg.Attachments[0].ContentType, "wrong Content-Type")
304+
assertEqual(t, 1, len(msg.Inline), "wrong inline Attachment count")
305+
assertEqual(t, `logo.png`, msg.Inline[0].FileName, "wrong Attachment name")
306+
assertEqual(t, `overridden/type`, msg.Inline[0].ContentType, "wrong Content-Type")
297307

298308
attachmentBytes, err := clientGet(ts.URL + "/api/v1/message/" + resp.ID + "/part/" + msg.Attachments[0].PartID)
299309
if err != nil {

server/ui/api/v1/swagger.json

Lines changed: 14 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -1634,12 +1634,22 @@
16341634
"Content": {
16351635
"description": "Base64-encoded string of the file content",
16361636
"type": "string",
1637-
"example": "VGhpcyBpcyBhIHBsYWluIHRleHQgYXR0YWNobWVudA=="
1637+
"example": "iVBORw0KGgoAAAANSUhEUgAAAEEAAAA8CAMAAAAOlSdoAAAACXBIWXMAAAHrAAAB6wGM2bZBAAAAS1BMVEVHcEwRfnUkZ2gAt4UsSF8At4UtSV4At4YsSV4At4YsSV8At4YsSV4At4YsSV4sSV4At4YsSV4At4YtSV4At4YsSV4At4YtSV8At4YsUWYNAAAAGHRSTlMAAwoXGiktRE5dbnd7kpOlr7zJ0d3h8PD8PCSRAAACWUlEQVR42pXT4ZaqIBSG4W9rhqQYocG+/ys9Y0Z0Br+x3j8zaxUPewFh65K+7yrIMeIY4MT3wPfEJCidKXEMnLaVkxDiELiMz4WEOAZSFghxBIypCOlKiAMgXfIqTnBgSm8CIQ6BImxEUxEckClVQiHGj4Ba4AQHikAIClwTE9KtIghAhUJwoLkmLnCiAHJLRKgIMsEtVUKbBUIwoAg2C4QgQBE6l4VCnApBgSKYLLApCnCa0+96AEMW2BQcmC+Pr3nfp7o5Exy49gIADcIqUELGfeA+bp93LmAJp8QJoEcN3C7NY3sbVANixMyI0nku20/n5/ZRf3KI2k6JEDWQtxcbdGuAqu3TAXG+/799Oyyas1B1MnMiA+XyxHp9q0PUKGPiRAau1fZbLRZV09wZcT8/gHk8QQAxXn8VgaDqcUmU6O/r28nbVwXAqca2mRNtPAF5+zoP2MeN9Fy4NgC6RfcbgE7XITBRYTtOE3U3C2DVff7pk+PkUxgAbvtnPXJaD6DxulMLwOhPS/M3MQkgg1ZFrIXnmfaZoOfpKiFgzeZD/WuKqQEGrfJYkyWf6vlG3xUgTuscnkNkQsb599q124kdpMUjCa/XARHs1gZymVtGt3wLkiFv8rUgTxitYCex5EVGec0Y9VmoDTFBSQte2TfXGXlf7hbdaUM9Sk7fisEN9qfBBTK+FZcvM9fQSdkl2vj4W2oX/bRogO3XasiNH7R0eW7fgRM834ImTg+Lg6BEnx4vz81rhr+MYPBBQg1v8GndEOrthxaCTxNAOut8WKLGZQl+MPz88Q9tAO/hVuSeqQAAAABJRU5ErkJggg=="
1638+
},
1639+
"ContentID": {
1640+
"description": "Optional Content-ID (`cid`) for attachment.\nIf this field is set then the file is attached inline.",
1641+
"type": "string",
1642+
"example": "mailpit-logo"
1643+
},
1644+
"ContentType": {
1645+
"description": "Optional Content Type for the the attachment.\nIf this field is not set (or empty) then the content type is automatically detected.",
1646+
"type": "string",
1647+
"example": "image/png"
16381648
},
16391649
"Filename": {
16401650
"description": "Filename",
16411651
"type": "string",
1642-
"example": "AttachedFile.txt"
1652+
"example": "mailpit.png"
16431653
}
16441654
}
16451655
}
@@ -1698,7 +1708,7 @@
16981708
"HTML": {
16991709
"description": "Message body (HTML)",
17001710
"type": "string",
1701-
"example": "\u003cp style=\"font-family: arial\"\u003eMailpit is \u003cb\u003eawesome\u003c/b\u003e!\u003c/p\u003e"
1711+
"example": "\u003cdiv style=\"text-align:center\"\u003e\u003cp style=\"font-family: arial; font-size: 24px;\"\u003eMailpit is \u003cb\u003eawesome\u003c/b\u003e!\u003c/p\u003e\u003cp\u003e\u003cimg src=\"cid:mailpit-logo\" /\u003e\u003c/p\u003e\u003c/div\u003e"
17021712
},
17031713
"Headers": {
17041714
"description": "Optional headers in {\"key\":\"value\"} format",
@@ -1751,7 +1761,7 @@
17511761
"Text": {
17521762
"description": "Message body (text)",
17531763
"type": "string",
1754-
"example": "This is the text body"
1764+
"example": "Mailpit is awesome!"
17551765
},
17561766
"To": {
17571767
"description": "\"To\" recipients",

0 commit comments

Comments
 (0)