Skip to content

Commit 096af4f

Browse files
authored
feat(imap): added support for imap trigger (#2663)
* feat(tools): added support for imap trigger * feat(imap): added parity, tested * ack PR comments * final cleanup
1 parent dc3de95 commit 096af4f

File tree

22 files changed

+1529
-26
lines changed

22 files changed

+1529
-26
lines changed

apps/docs/components/icons.tsx

Lines changed: 55 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -295,6 +295,56 @@ 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+
<rect
309+
x='3'
310+
y='4'
311+
width='18'
312+
height='16'
313+
rx='2'
314+
stroke='currentColor'
315+
strokeWidth='2'
316+
strokeLinecap='round'
317+
strokeLinejoin='round'
318+
/>
319+
<path
320+
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'
321+
stroke='currentColor'
322+
strokeWidth='2'
323+
strokeLinecap='round'
324+
strokeLinejoin='round'
325+
/>
326+
<line
327+
x1='7'
328+
y1='16'
329+
x2='7'
330+
y2='16'
331+
stroke='currentColor'
332+
strokeWidth='2'
333+
strokeLinecap='round'
334+
/>
335+
<line
336+
x1='10'
337+
y1='16'
338+
x2='10'
339+
y2='16'
340+
stroke='currentColor'
341+
strokeWidth='2'
342+
strokeLinecap='round'
343+
/>
344+
</svg>
345+
)
346+
}
347+
298348
export function CodeIcon(props: SVGProps<SVGSVGElement>) {
299349
return (
300350
<svg
@@ -4284,20 +4334,12 @@ export function RssIcon(props: SVGProps<SVGSVGElement>) {
42844334

42854335
export function SpotifyIcon(props: SVGProps<SVGSVGElement>) {
42864336
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=''
4337+
<svg {...props} xmlns='http://www.w3.org/2000/svg' viewBox='0 0 496 512'>
4338+
<path
4339+
fill='#1ed760'
4340+
d='M248 8C111.1 8 0 119.1 0 256s111.1 248 248 248 248-111.1 248-248S384.9 8 248 8Z'
43004341
/>
4342+
<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' />
43014343
</svg>
43024344
)
43034345
}

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: 101 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,101 @@
1+
import { createLogger } from '@sim/logger'
2+
import { ImapFlow } from 'imapflow'
3+
import { type NextRequest, NextResponse } from 'next/server'
4+
import { getSession } from '@/lib/auth'
5+
6+
const logger = createLogger('ImapMailboxesAPI')
7+
8+
interface ImapMailboxRequest {
9+
host: string
10+
port: number
11+
secure: boolean
12+
rejectUnauthorized: boolean
13+
username: string
14+
password: string
15+
}
16+
17+
export async function POST(request: NextRequest) {
18+
const session = await getSession()
19+
if (!session?.user?.id) {
20+
return NextResponse.json({ success: false, message: 'Unauthorized' }, { status: 401 })
21+
}
22+
23+
try {
24+
const body = (await request.json()) as ImapMailboxRequest
25+
const { host, port, secure, rejectUnauthorized, username, password } = body
26+
27+
if (!host || !username || !password) {
28+
return NextResponse.json(
29+
{ success: false, message: 'Missing required fields: host, username, password' },
30+
{ status: 400 }
31+
)
32+
}
33+
34+
const client = new ImapFlow({
35+
host,
36+
port: port || 993,
37+
secure: secure ?? true,
38+
auth: {
39+
user: username,
40+
pass: password,
41+
},
42+
tls: {
43+
rejectUnauthorized: rejectUnauthorized ?? true,
44+
},
45+
logger: false,
46+
})
47+
48+
try {
49+
await client.connect()
50+
51+
const listResult = await client.list()
52+
const mailboxes = listResult.map((mailbox) => ({
53+
path: mailbox.path,
54+
name: mailbox.name,
55+
delimiter: mailbox.delimiter,
56+
}))
57+
58+
await client.logout()
59+
60+
mailboxes.sort((a, b) => {
61+
if (a.path === 'INBOX') return -1
62+
if (b.path === 'INBOX') return 1
63+
return a.path.localeCompare(b.path)
64+
})
65+
66+
return NextResponse.json({
67+
success: true,
68+
mailboxes,
69+
})
70+
} catch (error) {
71+
try {
72+
await client.logout()
73+
} catch {
74+
// Ignore logout errors
75+
}
76+
throw error
77+
}
78+
} catch (error) {
79+
const errorMessage = error instanceof Error ? error.message : 'Unknown error'
80+
logger.error('Error fetching IMAP mailboxes:', errorMessage)
81+
82+
let userMessage = 'Failed to connect to IMAP server'
83+
if (
84+
errorMessage.includes('AUTHENTICATIONFAILED') ||
85+
errorMessage.includes('Invalid credentials')
86+
) {
87+
userMessage = 'Invalid username or password. For Gmail, use an App Password.'
88+
} else if (errorMessage.includes('ENOTFOUND') || errorMessage.includes('getaddrinfo')) {
89+
userMessage = 'Could not find IMAP server. Please check the hostname.'
90+
} else if (errorMessage.includes('ECONNREFUSED')) {
91+
userMessage = 'Connection refused. Please check the port and SSL settings.'
92+
} else if (errorMessage.includes('certificate') || errorMessage.includes('SSL')) {
93+
userMessage =
94+
'TLS/SSL error. Try disabling "Verify TLS Certificate" for self-signed certificates.'
95+
} else if (errorMessage.includes('timeout')) {
96+
userMessage = 'Connection timed out. Please check your network and server settings.'
97+
}
98+
99+
return NextResponse.json({ success: false, message: userMessage }, { status: 500 })
100+
}
101+
}
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+
}

apps/sim/app/workspace/[workspaceId]/w/[workflowId]/components/panel/components/editor/components/sub-block/components/dropdown/dropdown.tsx

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -113,7 +113,13 @@ export function Dropdown({
113113
const value = isPreview ? previewValue : propValue !== undefined ? propValue : storeValue
114114

115115
const singleValue = multiSelect ? null : (value as string | null | undefined)
116-
const multiValues = multiSelect ? (value as string[] | null | undefined) || [] : null
116+
const multiValues = multiSelect
117+
? Array.isArray(value)
118+
? value
119+
: value
120+
? [value as string]
121+
: []
122+
: null
117123

118124
const fetchOptionsIfNeeded = useCallback(async () => {
119125
if (!fetchOptions || isPreview || disabled) return

0 commit comments

Comments
 (0)