So IFTTT shipped MCP support. That means you can control your automations, list applets, edit triggers, run queries... all through the Model Context Protocol. In theory, any MCP-capable AI assistant can now talk directly to IFTTT.
In practice? Not quite.
Right now, IFTTT officially supports only Claude and ChatGPT as AI assistant integrations. You go to Settings β Connectors in Claude, or Settings β Connected Apps in ChatGPT, and IFTTT is right there. But if your AI assistant isn't on that short list? You're on your own.
Here's the situation. My AI assistant (Amazon Quick) speaks MCP via stdio. It launches a local process and communicates over stdin/stdout using JSON-RPC. Simple. Clean. Works great for local tools.
IFTTT's MCP server lives at https://ifttt.com/mcp
and uses Streamable HTTP transport. It expects authenticated HTTP POST requests and responds with either JSON or Server-Sent Events streams.
Two completely different transport layers. They don't talk to each other.
So what do you do? You build a proxy.
Well... "you" build a proxy. In my case, I described the problem to Amazon Quick (my AI assistant) and it wrote the entire proxy for me. All ~500 lines of it.
I guided the architecture, debugged alongside it, and steered the fixes when things broke. But the actual code? That was all Quick guiding Kiro. This whole post is really about what happens when you pair an AI coding assistant with a well-defined integration problem.
The proxy is a ~500-line Node.js script that sits between them:
ββββββββββββββ stdio βββββββββββββ HTTPS ββββββββββββ
β β JSON-RPC β β POST β β
β Amazon β βββββββββΆ β MCP β βββββββΆ β IFTTT β
β Quick β β Proxy β β MCP β
β β βββββββββ β (Node) β βββββββ β (Remote) β
β β JSON-RPC β β SSE/JSONβ β
ββββββββββββββ βββββββ¬ββββββ ββββββββββββ
local β remote
ββββββββ΄βββββββ
β OAuth 2.1 β
β PKCE + Auto β
β Refresh β
βββββββββββββββ
It reads JSON-RPC messages from stdin, forwards them as authenticated HTTPS requests to IFTTT, handles whatever response format comes back (direct JSON or SSE stream), and writes the response to stdout for Quick to consume.
The full flow:
Sounds straightforward? It mostly is. But two gotchas took me while to debug. Let me walk you through them.
First things first. IFTTT requires OAuth authentication. The proxy has an --auth
mode that handles the entire flow:
async function authenticate() {
const codeVerifier = generateCodeVerifier();
const codeChallenge = generateCodeChallenge(codeVerifier);
const state = generateState();
const authParams = new URLSearchParams({
client_id: CLIENT_ID,
code_challenge: codeChallenge,
code_challenge_method: 'S256',
redirect_uri: REDIRECT_URI,
resource: 'https://ifttt.com/mcp',
response_type: 'code',
scope: 'mcp',
state: state,
});
// Opens browser, starts local callback server on port 3118
// Exchanges code for token using PKCE verifier
// Saves token to ~/.quickwork/ifttt-token.json
}
Run node index.js --auth
once, authenticate in your browser, and the token gets saved locally. After that, the proxy handles refresh automatically. You never think about auth again.
The token management is simple but important:
function isTokenExpired(tokenData) {
if (!tokenData || !tokenData.access_token) return true;
if (!tokenData.expires_in) return false;
const expiresAt = tokenData.obtained_at + (tokenData.expires_in * 1000);
return Date.now() > expiresAt - 60000; // 1 minute buffer
}
That 60-second buffer matters. You don't want a request to fail because the token expires mid-flight.
So here's where it got interesting.
My first version of the proxy was dead simple. Read from stdin, POST to IFTTT, buffer the response, write to stdout. Classic request/response.
It worked great for tools/list
. IFTTT returned a nice 200 OK with a JSON body listing all available tools. I was feeling good.
Then I called my_applets
.
Nothing came back. No error. No response. Just... silence.
After adding some debug logging, I discovered IFTTT was returning HTTP 202 Accepted with an empty body. The actual response? It was coming back as a Server-Sent Events stream. But my buffered HTTP client was already done. It saw the empty body, closed the connection, and moved on.
The fix is a streaming-aware HTTP client that checks the Content-Type
header:
function httpsStreamingRequest(url, options, body, timeoutMs = 60000) {
return new Promise((resolve, reject) => {
const req = https.request(reqOptions, (res) => {
const contentType = res.headers['content-type'] || '';
const isSSE = contentType.includes('text/event-stream');
if (isSSE) {
// Keep the connection open, collect SSE events
let sseBuffer = '';
res.setEncoding('utf8');
res.on('data', (chunk) => { sseBuffer += chunk; });
res.on('end', () => {
resolve({
status: res.statusCode,
isSSE: true,
events: parseSSEBody(sseBuffer),
});
});
} else {
// Standard buffered response
let data = '';
res.on('data', (chunk) => { data += chunk; });
res.on('end', () => {
resolve({ status: res.statusCode, isSSE: false, body: data });
});
}
});
req.setTimeout(timeoutMs, () => {
req.destroy(new Error(`Request timed out after ${timeoutMs}ms`));
});
if (body) req.write(body);
req.end();
});
}
The SSE parser itself is straightforward. Events are separated by double newlines, data lines start with data:
:
function parseSSEBody(body) {
const events = [];
const blocks = body.split('\n\n');
for (const block of blocks) {
let eventData = '';
for (const line of block.split('\n')) {
if (line.startsWith('data: ')) {
eventData += line.substring(6);
} else if (line.startsWith('data:')) {
eventData += line.substring(5);
}
}
if (eventData) {
try { events.push(JSON.parse(eventData)); } catch (e) {}
}
}
return events;
}
After this fix, my_applets
worked beautifully. IFTTT returned 12 applets, all properly structured. I was back to feeling good.
For about 10 minutes.
So the proxy was getting responses. IFTTT was sending back data. But Amazon Quick was still showing... nothing. Or more precisely, it was throwing a vague "Tool execution failed" error.
I pulled the raw JSON-RPC response to see what IFTTT was actually sending:
{
"jsonrpc": "2.0",
"id": 1,
"result": {
"content": [],
"isError": false,
"structuredContent": {
"applets": [...]
}
}
}
See it? The content
array is empty. The actual data is in structuredContent
.
According to the MCP spec, tool results go in the content
array as TextContent
or ImageContent
objects. That's what Amazon Quick reads. IFTTT decided to put their data in a custom structuredContent
field instead, leaving content
as an empty array.
The fix is a response transformer that runs before writing to stdout:
function transformToolResponse(jsonRpcResponse) {
if (!jsonRpcResponse || !jsonRpcResponse.result) return jsonRpcResponse;
const result = jsonRpcResponse.result;
if (
result.structuredContent &&
(!result.content || result.content.length === 0)
) {
result.content = [
{
type: 'text',
text: JSON.stringify(result.structuredContent, null, 2),
},
];
}
return jsonRpcResponse;
}
12 lines. That's all it took. But finding the problem? That was the hard part.
With both gotchas solved, the main proxy loop is clean:
async function proxyMcpRequest(jsonRpcMessage) {
const token = await getValidToken();
const headers = {
'Content-Type': 'application/json',
'Authorization': `Bearer ${token}`,
'Accept': 'application/json, text/event-stream',
};
if (mcpSessionId) {
headers['Mcp-Session-Id'] = mcpSessionId;
}
let response = await httpsStreamingRequest(IFTTT_MCP_URL, {
method: 'POST', headers
}, JSON.stringify(jsonRpcMessage));
// Capture session ID for subsequent requests
if (response.sessionId) {
mcpSessionId = response.sessionId;
}
// Handle 401 - try token refresh
if (response.status === 401) {
cachedToken = await refreshToken(cachedToken);
headers['Authorization'] = `Bearer ${cachedToken.access_token}`;
response = await httpsStreamingRequest(IFTTT_MCP_URL, {
method: 'POST', headers
}, JSON.stringify(jsonRpcMessage));
}
return response;
}
The Accept: application/json, text/event-stream
header is important. It tells IFTTT "I can handle both formats." Without it, you might not get the SSE stream at all.
The proxy registers itself in the MCP config as a simple stdio server:
{
"mcpServers": {
"ifttt": {
"command": "node",
"args": ["/path/to/ifttt-mcp-proxy/index.js"]
}
}
}
That's it. Amazon Quick launches the process, pipes JSON-RPC to stdin, reads responses from stdout. The proxy handles everything in between: auth, streaming, format translation, token refresh.
With this proxy running, I can do all of this from my AI assistant using natural language:
No browser. No IFTTT web UI. Just conversational access to my entire automation setup.
A few takeaways if you're building something similar:
The MCP spec has transport flexibility. Stdio and Streamable HTTP are both valid, but they don't interoperate automatically. If you're connecting a stdio client to an HTTP server, you need a proxy.
If you're working with MCP on AWS, Amazon Bedrock Agents supports MCP servers natively for remote tool use... so you might not need a custom proxy if you're already in that ecosystem.
SSE is sneaky. When a server returns 202 Accepted, your instinct is "okay, no content." But with SSE, the content is coming... just not the way you expect. Always check Content-Type
before closing the connection.
Not everyone implements the spec the same way. IFTTT's use of structuredContent
instead of content[]
is technically non-standard. Your proxy might need to normalize responses.
OAuth 2.1 + PKCE is worth the complexity. No client secrets stored on disk, proper token rotation, and it works great for local tools that need to authenticate with remote services.
AI assistants are shockingly good at integration plumbing. I didn't write a single line of this proxy by hand. I described the problem to Amazon Quick, and it generated the entire thing... the OAuth flow, the streaming HTTP client, the SSE parser, the response transformer.
When something broke, I described the symptoms and it diagnosed and fixed the issue. The whole thing went from "IFTTT has MCP support" to "fully working native integration" in about an hour of back-and-forth conversation. That's the real story here. I've written more about this dynamic between developer and AI coding assistant... it's a relationship worth understanding.
Tools like the AWS Toolkit for AI Agents are making this kind of AI-assisted building the norm rather than the exception.
The full proxy is about 500 lines of zero-dependency Node.js. No npm install needed. Just node
and the built-in http
, https
, and crypto
modules.
The complete source code is on GitHub.
I would be very interested to hear your thoughts or comments, so if you've built something similar or found a different approach, ping me on X or LinkedIn or feel free to leave a comment below.
And if you're trying to connect other remote MCP servers to a local client...
your mileage may vary, but the pattern should be the same.