Skip to content

Commit 5f8d769

Browse files
committed
feat(tools): added support for imap trigger
1 parent 79be435 commit 5f8d769

File tree

17 files changed

+1336
-14
lines changed

17 files changed

+1336
-14
lines changed

apps/docs/components/icons.tsx

Lines changed: 56 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -295,6 +295,57 @@ export function MailIcon(props: SVGProps<SVGSVGElement>) {
295295
)
296296
}
297297

298+
export function MailServerIcon(props: SVGProps<SVGSVGElement>) {
299+
return (
300+
<svg
301+
{...props}
302+
width='24'
303+
height='24'
304+
viewBox='0 0 24 24'
305+
fill='none'
306+
xmlns='http://www.w3.org/2000/svg'
307+
>
308+
{/* Server/inbox icon with mail symbol */}
309+
<rect
310+
x='3'
311+
y='4'
312+
width='18'
313+
height='16'
314+
rx='2'
315+
stroke='currentColor'
316+
strokeWidth='2'
317+
strokeLinecap='round'
318+
strokeLinejoin='round'
319+
/>
320+
<path
321+
d='M3 8L10.89 13.26C11.2187 13.4793 11.6049 13.5963 12 13.5963C12.3951 13.5963 12.7813 13.4793 13.11 13.26L21 8'
322+
stroke='currentColor'
323+
strokeWidth='2'
324+
strokeLinecap='round'
325+
strokeLinejoin='round'
326+
/>
327+
<line
328+
x1='7'
329+
y1='16'
330+
x2='7'
331+
y2='16'
332+
stroke='currentColor'
333+
strokeWidth='2'
334+
strokeLinecap='round'
335+
/>
336+
<line
337+
x1='10'
338+
y1='16'
339+
x2='10'
340+
y2='16'
341+
stroke='currentColor'
342+
strokeWidth='2'
343+
strokeLinecap='round'
344+
/>
345+
</svg>
346+
)
347+
}
348+
298349
export function CodeIcon(props: SVGProps<SVGSVGElement>) {
299350
return (
300351
<svg
@@ -4284,20 +4335,12 @@ export function RssIcon(props: SVGProps<SVGSVGElement>) {
42844335

42854336
export function SpotifyIcon(props: SVGProps<SVGSVGElement>) {
42864337
return (
4287-
<svg
4288-
{...props}
4289-
width='386'
4290-
height='386'
4291-
viewBox='100 100 186 186'
4292-
fill='none'
4293-
xmlns='http://www.w3.org/2000/svg'
4294-
xmlnsXlink='http://www.w3.org/1999/xlink'
4295-
>
4296-
<image
4297-
width='386'
4298-
height='386'
4299-
xlinkHref=''
4338+
<svg {...props} xmlns='http://www.w3.org/2000/svg' viewBox='0 0 496 512'>
4339+
<path
4340+
fill='#1ed760'
4341+
d='M248 8C111.1 8 0 119.1 0 256s111.1 248 248 248 248-111.1 248-248S384.9 8 248 8Z'
43004342
/>
4343+
<path d='M406.6 231.1c-5.2 0-8.4-1.3-12.9-3.9-71.2-42.5-198.5-52.7-280.9-29.7-3.6 1-8.1 2.6-12.9 2.6-13.2 0-23.3-10.3-23.3-23.6 0-13.6 8.4-21.3 17.4-23.9 35.2-10.3 74.6-15.2 117.5-15.2 73 0 149.5 15.2 205.4 47.8 7.8 4.5 12.9 10.7 12.9 22.6 0 13.6-11 23.3-23.2 23.3zm-31 76.2c-5.2 0-8.7-2.3-12.3-4.2-62.5-37-155.7-51.9-238.6-29.4-4.8 1.3-7.4 2.6-11.9 2.6-10.7 0-19.4-8.7-19.4-19.4s5.2-17.8 15.5-20.7c27.8-7.8 56.2-13.6 97.8-13.6 64.9 0 127.6 16.1 177 45.5 8.1 4.8 11.3 11 11.3 19.7-.1 10.8-8.5 19.5-19.4 19.5zm-26.9 65.6c-4.2 0-6.8-1.3-10.7-3.6-62.4-37.6-135-39.2-206.7-24.5-3.9 1-9 2.6-11.9 2.6-9.7 0-15.8-7.7-15.8-15.8 0-10.3 6.1-15.2 13.6-16.8 81.9-18.1 165.6-16.5 237 26.2 6.1 3.9 9.7 7.4 9.7 16.5s-7.1 15.4-15.2 15.4z' />
43014344
</svg>
43024345
)
43034346
}

apps/docs/components/ui/icon-mapping.ts

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,7 @@ import {
5858
LinkupIcon,
5959
MailchimpIcon,
6060
MailgunIcon,
61+
MailServerIcon,
6162
Mem0Icon,
6263
MicrosoftExcelIcon,
6364
MicrosoftOneDriveIcon,
@@ -165,6 +166,7 @@ export const blockTypeToIconMap: Record<string, IconComponent> = {
165166
huggingface: HuggingFaceIcon,
166167
hunter: HunterIOIcon,
167168
image_generator: ImageIcon,
169+
imap: MailServerIcon,
168170
incidentio: IncidentioIcon,
169171
intercom: IntercomIcon,
170172
jina: JinaAIIcon,
Lines changed: 40 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
---
2+
title: IMAP Email
3+
description: Trigger workflows when new emails arrive via IMAP (works with any email provider)
4+
---
5+
6+
import { BlockInfoCard } from "@/components/ui/block-info-card"
7+
8+
<BlockInfoCard
9+
type="imap"
10+
color="#6366F1"
11+
/>
12+
13+
{/* MANUAL-CONTENT-START:intro */}
14+
The IMAP Email trigger allows your Sim workflows to start automatically whenever a new email is received in any mailbox that supports the IMAP protocol. This works with Gmail, Outlook, Yahoo, and most other email providers.
15+
16+
With the IMAP trigger, you can:
17+
18+
- **Automate email processing**: Start workflows in real time when new messages arrive in your inbox.
19+
- **Filter by sender, subject, or folder**: Configure your trigger to react only to emails that match certain conditions.
20+
- **Extract and process attachments**: Automatically download and use file attachments in your automated flows.
21+
- **Parse and use email content**: Access the subject, sender, recipients, full body, and other metadata in downstream workflow steps.
22+
- **Integrate with any email provider**: Works with any service that provides standard IMAP access, without vendor lock-in.
23+
- **Trigger on unread, flagged, or custom criteria**: Set up advanced filters for the kinds of emails that start your workflows.
24+
25+
With Sim, the IMAP integration gives you the power to turn email into an actionable source of automation. Respond to customer inquiries, process notifications, kick off data pipelines, and more—directly from your email inbox, with no manual intervention.
26+
{/* MANUAL-CONTENT-END */}
27+
28+
29+
## Usage Instructions
30+
31+
Connect to any email server via IMAP protocol to trigger workflows when new emails are received. Supports Gmail, Outlook, Yahoo, and any other IMAP-compatible email provider.
32+
33+
34+
35+
36+
37+
## Notes
38+
39+
- Category: `triggers`
40+
- Type: `imap`

apps/docs/content/docs/en/tools/meta.json

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@
4242
"huggingface",
4343
"hunter",
4444
"image_generator",
45+
"imap",
4546
"incidentio",
4647
"intercom",
4748
"jina",
Lines changed: 99 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
import { createLogger } from '@sim/logger'
2+
import { ImapFlow } from 'imapflow'
3+
import { type NextRequest, NextResponse } from 'next/server'
4+
5+
const logger = createLogger('ImapMailboxesAPI')
6+
7+
interface ImapMailboxRequest {
8+
host: string
9+
port: number
10+
secure: boolean
11+
rejectUnauthorized: boolean
12+
username: string
13+
password: string
14+
}
15+
16+
export async function POST(request: NextRequest) {
17+
try {
18+
const body = (await request.json()) as ImapMailboxRequest
19+
const { host, port, secure, rejectUnauthorized, username, password } = body
20+
21+
if (!host || !username || !password) {
22+
return NextResponse.json(
23+
{ success: false, message: 'Missing required fields: host, username, password' },
24+
{ status: 400 }
25+
)
26+
}
27+
28+
const client = new ImapFlow({
29+
host,
30+
port: port || 993,
31+
secure: secure ?? true,
32+
auth: {
33+
user: username,
34+
pass: password,
35+
},
36+
tls: {
37+
rejectUnauthorized: rejectUnauthorized ?? true,
38+
},
39+
logger: false,
40+
})
41+
42+
try {
43+
await client.connect()
44+
45+
// List all mailboxes
46+
const listResult = await client.list()
47+
const mailboxes = listResult.map((mailbox) => ({
48+
path: mailbox.path,
49+
name: mailbox.name,
50+
delimiter: mailbox.delimiter,
51+
}))
52+
53+
await client.logout()
54+
55+
// Sort mailboxes: INBOX first, then alphabetically
56+
mailboxes.sort((a, b) => {
57+
if (a.path === 'INBOX') return -1
58+
if (b.path === 'INBOX') return 1
59+
return a.path.localeCompare(b.path)
60+
})
61+
62+
return NextResponse.json({
63+
success: true,
64+
mailboxes,
65+
})
66+
} catch (error) {
67+
// Make sure to close connection on error
68+
try {
69+
await client.logout()
70+
} catch {
71+
// Ignore logout errors
72+
}
73+
throw error
74+
}
75+
} catch (error) {
76+
const errorMessage = error instanceof Error ? error.message : 'Unknown error'
77+
logger.error('Error fetching IMAP mailboxes:', errorMessage)
78+
79+
// Provide user-friendly error messages
80+
let userMessage = 'Failed to connect to IMAP server'
81+
if (
82+
errorMessage.includes('AUTHENTICATIONFAILED') ||
83+
errorMessage.includes('Invalid credentials')
84+
) {
85+
userMessage = 'Invalid username or password. For Gmail, use an App Password.'
86+
} else if (errorMessage.includes('ENOTFOUND') || errorMessage.includes('getaddrinfo')) {
87+
userMessage = 'Could not find IMAP server. Please check the hostname.'
88+
} else if (errorMessage.includes('ECONNREFUSED')) {
89+
userMessage = 'Connection refused. Please check the port and SSL settings.'
90+
} else if (errorMessage.includes('certificate') || errorMessage.includes('SSL')) {
91+
userMessage =
92+
'TLS/SSL error. Try disabling "Verify TLS Certificate" for self-signed certificates.'
93+
} else if (errorMessage.includes('timeout')) {
94+
userMessage = 'Connection timed out. Please check your network and server settings.'
95+
}
96+
97+
return NextResponse.json({ success: false, message: userMessage }, { status: 500 })
98+
}
99+
}
Lines changed: 68 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
import { createLogger } from '@sim/logger'
2+
import { nanoid } from 'nanoid'
3+
import { type NextRequest, NextResponse } from 'next/server'
4+
import { verifyCronAuth } from '@/lib/auth/internal'
5+
import { acquireLock, releaseLock } from '@/lib/core/config/redis'
6+
import { pollImapWebhooks } from '@/lib/webhooks/imap-polling-service'
7+
8+
const logger = createLogger('ImapPollingAPI')
9+
10+
export const dynamic = 'force-dynamic'
11+
export const maxDuration = 180 // Allow up to 3 minutes for polling to complete
12+
13+
const LOCK_KEY = 'imap-polling-lock'
14+
const LOCK_TTL_SECONDS = 180 // Same as maxDuration (3 min)
15+
16+
export async function GET(request: NextRequest) {
17+
const requestId = nanoid()
18+
logger.info(`IMAP webhook polling triggered (${requestId})`)
19+
20+
let lockValue: string | undefined
21+
22+
try {
23+
const authError = verifyCronAuth(request, 'IMAP webhook polling')
24+
if (authError) {
25+
return authError
26+
}
27+
28+
lockValue = requestId // unique value to identify the holder
29+
const locked = await acquireLock(LOCK_KEY, lockValue, LOCK_TTL_SECONDS)
30+
31+
if (!locked) {
32+
return NextResponse.json(
33+
{
34+
success: true,
35+
message: 'Polling already in progress – skipped',
36+
requestId,
37+
status: 'skip',
38+
},
39+
{ status: 202 }
40+
)
41+
}
42+
43+
const results = await pollImapWebhooks()
44+
45+
return NextResponse.json({
46+
success: true,
47+
message: 'IMAP polling completed',
48+
requestId,
49+
status: 'completed',
50+
...results,
51+
})
52+
} catch (error) {
53+
logger.error(`Error during IMAP polling (${requestId}):`, error)
54+
return NextResponse.json(
55+
{
56+
success: false,
57+
message: 'IMAP polling failed',
58+
error: error instanceof Error ? error.message : 'Unknown error',
59+
requestId,
60+
},
61+
{ status: 500 }
62+
)
63+
} finally {
64+
if (lockValue) {
65+
await releaseLock(LOCK_KEY, lockValue).catch(() => {})
66+
}
67+
}
68+
}

0 commit comments

Comments
 (0)