mailpit-api 1.0.0 → 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,12 +1,12 @@
1
1
  {
2
2
  "name": "mailpit-api",
3
- "version": "1.0.0",
3
+ "version": "1.0.1",
4
4
  "description": "A NodeJS client library, written in TypeScript, to interact with the Mailpit API.",
5
5
  "repository": {
6
6
  "type": "git",
7
7
  "url": "https://github.com/mpspahr/mailpit-api.git"
8
8
  },
9
- "main": "index.ts",
9
+ "main": "src/index.ts",
10
10
  "scripts": {
11
11
  "test": "echo \"TODO: Add tests\" && exit 0",
12
12
  "build": "tsc"
package/src/index.ts ADDED
@@ -0,0 +1,467 @@
1
+ import axios, { AxiosInstance } from "axios";
2
+
3
+ interface MailpitInfo {
4
+ Database: string;
5
+ DatabaseSize: number;
6
+ LatestVersion: string;
7
+ Messages: number;
8
+ RuntimeStats: {
9
+ Memory: number;
10
+ MessagesDeleted: number;
11
+ SMTPAccepted: number;
12
+ SMTPAcceptedSize: number;
13
+ SMTPIgnored: number;
14
+ SMTPRejected: number;
15
+ Uptime: number;
16
+ };
17
+ Tags: {
18
+ [key: string]: number;
19
+ };
20
+ Unread: number;
21
+ Version: string;
22
+ }
23
+
24
+ interface MailpitConfiguration {
25
+ DuplicatesIgnored: boolean;
26
+ Label: string;
27
+ SpamAssassin: boolean;
28
+ MessageRelay: {
29
+ AllowedRecipients: string;
30
+ Enabled: boolean;
31
+ ReturnPath: string;
32
+ SMTPServer: string;
33
+ };
34
+ }
35
+
36
+ interface MailpitMessageSummary {
37
+ Attachments: [
38
+ {
39
+ ContentID: string;
40
+ ContentType: string;
41
+ FileName: string;
42
+ PartID: string;
43
+ Size: number;
44
+ },
45
+ ];
46
+ Bcc: [
47
+ {
48
+ Address: string;
49
+ Name: string;
50
+ },
51
+ ];
52
+ Cc: [
53
+ {
54
+ Address: string;
55
+ Name: string;
56
+ },
57
+ ];
58
+ Date: string;
59
+ From: {
60
+ Address: string;
61
+ Name: string;
62
+ };
63
+ HTML: string;
64
+ ID: string;
65
+ Inline: [
66
+ {
67
+ ContentID: string;
68
+ ContentType: string;
69
+ FileName: string;
70
+ PartID: string;
71
+ Size: number;
72
+ },
73
+ ];
74
+ MessageID: string;
75
+ ReplyTo: [
76
+ {
77
+ Address: string;
78
+ Name: string;
79
+ },
80
+ ];
81
+ ReturnPath: string;
82
+ Size: number;
83
+ Subject: string;
84
+ Tags: [string];
85
+ Text: string;
86
+ To: [
87
+ {
88
+ Address: string;
89
+ Name: string;
90
+ },
91
+ ];
92
+ }
93
+
94
+ interface MailpitMessagesSummary {
95
+ messages: [MailpitMessageSummary];
96
+ }
97
+
98
+ interface MailpitMessageHeaders {
99
+ [key: string]: string;
100
+ }
101
+
102
+ interface MailpitSendRequest {
103
+ Attachments: [
104
+ {
105
+ Content: string;
106
+ Filename: string;
107
+ },
108
+ ];
109
+ Bcc: [string];
110
+ Cc: [
111
+ {
112
+ Email: string;
113
+ Name: string;
114
+ },
115
+ ];
116
+ From: {
117
+ Email: string;
118
+ Name: string;
119
+ };
120
+ HTML: string;
121
+ Headers: {
122
+ [key: string]: string;
123
+ };
124
+ ReplyTo: [
125
+ {
126
+ Email: string;
127
+ Name: string;
128
+ },
129
+ ];
130
+ Subject: string;
131
+ Tags: [string];
132
+ Text: string;
133
+ To: [
134
+ {
135
+ Email: string;
136
+ Name: string;
137
+ },
138
+ ];
139
+ }
140
+
141
+ interface MailpitSendMessageConfirmation {
142
+ ID: string;
143
+ }
144
+
145
+ interface MailpitHTMLCheckResponse {
146
+ Platforms: {
147
+ [key: string]: [string];
148
+ };
149
+ Total: {
150
+ Nodes: number;
151
+ Partial: number;
152
+ Supported: number;
153
+ Tests: number;
154
+ Unsupported: number;
155
+ };
156
+ Warnings: [
157
+ {
158
+ Category: "css" | "html";
159
+ Description: string;
160
+ Keywords: string;
161
+ NotesByNumber: {
162
+ [key: string]: string;
163
+ };
164
+ Results: [
165
+ {
166
+ Family: string;
167
+ Name: string;
168
+ NoteNumber: string;
169
+ Platform: string;
170
+ Support: "yes" | "no" | "partial";
171
+ Version: string;
172
+ },
173
+ ];
174
+ Score: {
175
+ Found: number;
176
+ Partial: number;
177
+ Supported: number;
178
+ Unsupported: number;
179
+ };
180
+ Slug: string;
181
+ Tags: [string];
182
+ Title: string;
183
+ URL: string;
184
+ },
185
+ ];
186
+ }
187
+
188
+ interface MailpitLinkCheckResponse {
189
+ Errors: number;
190
+ Links: [
191
+ {
192
+ Status: string;
193
+ StatusCode: number;
194
+ URL: string;
195
+ },
196
+ ];
197
+ }
198
+
199
+ interface MailpitSpamAssassinResponse {
200
+ Errors: number;
201
+ IsSpam: boolean;
202
+ Rules: [
203
+ {
204
+ Description: string;
205
+ Name: string;
206
+ Score: number;
207
+ },
208
+ ];
209
+ Score: number;
210
+ }
211
+
212
+ interface MailpitReadStatusRequest {
213
+ IDs: [string];
214
+ Read: boolean;
215
+ }
216
+
217
+ interface MailpitDeleteRequest {
218
+ IDs: [string];
219
+ }
220
+
221
+ interface MailpitSearch {
222
+ query: string;
223
+ start: number;
224
+ limit: number;
225
+ tz: string;
226
+ }
227
+
228
+ interface MailpitSearchDelete {
229
+ query: string;
230
+ tz: string;
231
+ }
232
+
233
+ interface MailpitSetTagsRequest {
234
+ IDs: [string];
235
+ Tags: [string];
236
+ }
237
+
238
+ class MailpitClient {
239
+ private axiosInstance: AxiosInstance;
240
+
241
+ constructor(baseURL: string) {
242
+ this.axiosInstance = axios.create({
243
+ baseURL: baseURL,
244
+ validateStatus: function (status) {
245
+ return status === 200;
246
+ },
247
+ });
248
+ }
249
+
250
+ private async handleRequest<T>(
251
+ request: () => Promise<{ data: T }>,
252
+ ): Promise<T> {
253
+ try {
254
+ const response = await request();
255
+ return response.data;
256
+ } catch (error) {
257
+ if (axios.isAxiosError(error)) {
258
+ if (error.response) {
259
+ // Server responded with a status other than 2xx
260
+ throw new Error(
261
+ `Mailpit API Error: ${error.response.status} ${error.response.statusText}: ${JSON.stringify(error.response.data)}`,
262
+ );
263
+ } else if (error.request) {
264
+ // Request was made but no response was received
265
+ throw new Error(
266
+ "Mailpit API Error: No response received from server.",
267
+ );
268
+ } else {
269
+ // Something happened in setting up the request
270
+ throw new Error(`Mailpit API Error: ${error.message}`);
271
+ }
272
+ } else {
273
+ throw new Error("Unexpected Error: " + error);
274
+ }
275
+ }
276
+ }
277
+
278
+ // Message
279
+ public async getInfo(): Promise<MailpitInfo> {
280
+ return this.handleRequest(() =>
281
+ this.axiosInstance.get<MailpitInfo>("/api/v1/info"),
282
+ );
283
+ }
284
+
285
+ public async getConfiguration(): Promise<MailpitConfiguration> {
286
+ return this.handleRequest(() =>
287
+ this.axiosInstance.get<MailpitConfiguration>("/api/v1/webui"),
288
+ );
289
+ }
290
+
291
+ public async getMessageSummary(id: string): Promise<MailpitMessageSummary> {
292
+ return this.handleRequest(() =>
293
+ this.axiosInstance.get<MailpitMessageSummary>(`/api/v1/message/${id}`),
294
+ );
295
+ }
296
+
297
+ public async getMessageHeaders(id: string): Promise<MailpitMessageHeaders> {
298
+ return this.handleRequest(() =>
299
+ this.axiosInstance.get<MailpitMessageHeaders>(
300
+ `/api/v1/message/${id}/headers`,
301
+ ),
302
+ );
303
+ }
304
+
305
+ public async getMessageAttachment(
306
+ id: string,
307
+ partID: string,
308
+ ): Promise<string> {
309
+ return this.handleRequest(() =>
310
+ this.axiosInstance.get<string>(`/api/v1/message/${id}/part/${partID}`),
311
+ );
312
+ }
313
+
314
+ public async getMessageSource(id: string): Promise<string> {
315
+ return this.handleRequest(() =>
316
+ this.axiosInstance.get<string>(`/api/v1/message/${id}/raw`),
317
+ );
318
+ }
319
+
320
+ public async getAttachmentThumbnail(
321
+ id: string,
322
+ partID: string,
323
+ ): Promise<string> {
324
+ return this.handleRequest(() =>
325
+ this.axiosInstance.get<string>(
326
+ `/api/v1/message/${id}/part/${partID}/thumb`,
327
+ ),
328
+ );
329
+ }
330
+
331
+ public async releaseMessage(
332
+ id: string,
333
+ releaseRequest: { To: string[] },
334
+ ): Promise<string> {
335
+ return this.handleRequest(() =>
336
+ this.axiosInstance.post<string>(
337
+ `/api/v1/message/${id}/release`,
338
+ releaseRequest,
339
+ ),
340
+ );
341
+ }
342
+
343
+ public async sendMessage(
344
+ sendReqest: MailpitSendRequest,
345
+ ): Promise<MailpitSendMessageConfirmation> {
346
+ return this.handleRequest(() =>
347
+ this.axiosInstance.post<MailpitSendMessageConfirmation>(
348
+ `/api/v1/send`,
349
+ sendReqest,
350
+ ),
351
+ );
352
+ }
353
+
354
+ // Other
355
+ public async htmlCheck(id: string): Promise<MailpitHTMLCheckResponse> {
356
+ return this.handleRequest(() =>
357
+ this.axiosInstance.get<MailpitHTMLCheckResponse>(
358
+ `/api/v1/message/${id}/html-check`,
359
+ ),
360
+ );
361
+ }
362
+
363
+ public async linkCheck(id: string): Promise<MailpitLinkCheckResponse> {
364
+ return this.handleRequest(() =>
365
+ this.axiosInstance.get<MailpitLinkCheckResponse>(
366
+ `/api/v1/message/${id}/link-check`,
367
+ ),
368
+ );
369
+ }
370
+
371
+ public async spamAssassinCheck(
372
+ id: string,
373
+ ): Promise<MailpitSpamAssassinResponse> {
374
+ return this.handleRequest(() =>
375
+ this.axiosInstance.get<MailpitSpamAssassinResponse>(
376
+ `/api/v1/message/${id}/sa-check`,
377
+ ),
378
+ );
379
+ }
380
+
381
+ // Messages
382
+ public async listMessages(): Promise<MailpitMessagesSummary> {
383
+ return this.handleRequest(() =>
384
+ this.axiosInstance.get<MailpitMessagesSummary>(`/api/v1/messages`),
385
+ );
386
+ }
387
+
388
+ public async setReadStatus(
389
+ readStatus: MailpitReadStatusRequest,
390
+ ): Promise<string> {
391
+ return this.handleRequest(() =>
392
+ this.axiosInstance.put<string>(`/api/v1/messages`, readStatus),
393
+ );
394
+ }
395
+
396
+ public async deleteMessages(
397
+ deleteRequest: MailpitDeleteRequest,
398
+ ): Promise<string> {
399
+ return this.handleRequest(() =>
400
+ this.axiosInstance.delete<string>(`/api/v1/messages`, {
401
+ data: deleteRequest,
402
+ }),
403
+ );
404
+ }
405
+
406
+ // See https://mailpit.axllent.org/docs/usage/search-filters/
407
+ public async searchMessages(
408
+ search: MailpitSearch,
409
+ ): Promise<MailpitMessagesSummary> {
410
+ return this.handleRequest(() =>
411
+ this.axiosInstance.put<MailpitMessagesSummary>(`/api/v1/search`, search),
412
+ );
413
+ }
414
+
415
+ // See https://mailpit.axllent.org/docs/usage/search-filters/
416
+ public async deleteMessagesBySearch(
417
+ search: MailpitSearchDelete,
418
+ ): Promise<string> {
419
+ return this.handleRequest(() =>
420
+ this.axiosInstance.delete<string>(`/api/v1/search`, { data: search }),
421
+ );
422
+ }
423
+
424
+ // Tags
425
+ public async getTags(): Promise<[string]> {
426
+ return this.handleRequest(() =>
427
+ this.axiosInstance.get<[string]>(`/api/v1/tags`),
428
+ );
429
+ }
430
+
431
+ public async setTags(request: MailpitSetTagsRequest): Promise<string> {
432
+ return this.handleRequest(() =>
433
+ this.axiosInstance.put<string>(`/api/v1/tags`, request),
434
+ );
435
+ }
436
+
437
+ public async renameTag(tag: string, newTagName: string): Promise<string> {
438
+ const encodedTag = encodeURI(tag);
439
+ return this.handleRequest(() =>
440
+ this.axiosInstance.put<string>(`/api/v1/tags/${encodedTag}`, {
441
+ Name: newTagName,
442
+ }),
443
+ );
444
+ }
445
+
446
+ public async deleteTag(tag: string): Promise<string> {
447
+ const encodedTag = encodeURI(tag);
448
+ return this.handleRequest(() =>
449
+ this.axiosInstance.delete<string>(`/api/v1/tags/${encodedTag}`),
450
+ );
451
+ }
452
+
453
+ // Testing
454
+ public async renderMessageHTML(id: string): Promise<string> {
455
+ return this.handleRequest(() =>
456
+ this.axiosInstance.get<string>(`/view/${id}.html`),
457
+ );
458
+ }
459
+
460
+ public async renderMessageText(id: string): Promise<string> {
461
+ return this.handleRequest(() =>
462
+ this.axiosInstance.get<string>(`/view/${id}.txt`),
463
+ );
464
+ }
465
+ }
466
+
467
+ export default MailpitClient;
@@ -1,33 +0,0 @@
1
- # This workflow will run tests using node and then publish a package to GitHub Packages when a release is created
2
- # For more information see: https://docs.github.com/en/actions/publishing-packages/publishing-nodejs-packages
3
-
4
- name: Node.js Package
5
-
6
- on:
7
- release:
8
- types: [created]
9
-
10
- jobs:
11
- build:
12
- runs-on: ubuntu-latest
13
- steps:
14
- - uses: actions/checkout@v4
15
- - uses: actions/setup-node@v4
16
- with:
17
- node-version: 20
18
- - run: npm ci
19
- - run: npm test
20
-
21
- publish-npm:
22
- needs: build
23
- runs-on: ubuntu-latest
24
- steps:
25
- - uses: actions/checkout@v4
26
- - uses: actions/setup-node@v4
27
- with:
28
- node-version: 20
29
- registry-url: https://registry.npmjs.org/
30
- - run: npm ci
31
- - run: npm publish
32
- env:
33
- NODE_AUTH_TOKEN: ${{secrets.npm_token}}
package/.prettierignore DELETED
@@ -1,3 +0,0 @@
1
- # Ignore artifacts:
2
- build
3
- coverage
package/.prettierrc DELETED
@@ -1 +0,0 @@
1
- {}
package/eslint.config.mjs DELETED
@@ -1,10 +0,0 @@
1
- import globals from "globals";
2
- import pluginJs from "@eslint/js";
3
- import tseslint from "typescript-eslint";
4
-
5
- export default [
6
- { files: ["**/*.{js,mjs,cjs,ts}"] },
7
- { languageOptions: { globals: globals.browser } },
8
- pluginJs.configs.recommended,
9
- ...tseslint.configs.recommended,
10
- ];