惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

F
Full Disclosure
博客园 - 聂微东
IT之家
IT之家
The Cloudflare Blog
L
LangChain Blog
Last Week in AI
Last Week in AI
T
Tailwind CSS Blog
P
Proofpoint News Feed
aimingoo的专栏
aimingoo的专栏
G
Google Developers Blog
T
The Blog of Author Tim Ferriss
博客园 - 叶小钗
I
Intezer
Martin Fowler
Martin Fowler
MongoDB | Blog
MongoDB | Blog
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
T
ThreatConnect
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
小众软件
小众软件
T
The Exploit Database - CXSecurity.com
H
Help Net Security
T
Tenable Blog
WordPress大学
WordPress大学
F
Future of Privacy Forum
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
NISL@THU
NISL@THU
The Register - Security
The Register - Security
A
About on SuperTechFans
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
MyScale Blog
MyScale Blog
Malwarebytes
Malwarebytes
博客园_首页
T
Threatpost
C
CERT Recently Published Vulnerability Notes
Know Your Adversary
Know Your Adversary
T
Threat Research - Cisco Blogs
V
Vulnerabilities – Threatpost
C
CXSECURITY Database RSS Feed - CXSecurity.com
Blog — PlanetScale
Blog — PlanetScale
Recorded Future
Recorded Future
大猫的无限游戏
大猫的无限游戏
K
Kaspersky official blog
月光博客
月光博客
Jina AI
Jina AI
S
Securelist
Hugging Face - Blog
Hugging Face - Blog
G
GRAHAM CLULEY
腾讯CDC
S
Secure Thoughts
V
V2EX - 技术

DEV Community

Turn Your Phone Into Voice Input for Any React Text Field Which package is bloating your Docker image? Putting Claude Code Under Version Control: Configs Since July, Memory Since April What I Thought DevRel Was vs. What It Actually Is (A Mentee's Honest Take) Reviving My Linux Mastery Game from a Merge Conflict — A Finish-Up-A-Thon Comeback 400 Million Tokens Burned Overnight Don’t let AI break your collective thinking: a practical guide for engineering teams First Gemma 4 ExecuTorch Deployment on Raspberry Pi 5 — and Why It's 7.7 Slower Than llama.cpp Per-Turn Evaluation: Dynamic Governance for AI Agents The AI Triforce of seed4j: Power, Wisdom, and Courage for Your Dev Agent Your AI agent reports 80% task completion. It fabricated it. Pourquoi les overlays d'accessibilité ne tiennent pas leurs promesses (et ce que la FTC vient d'acter) AI May Break Product-Market Fit in Enterprise Software I’m Building Around the Gap Between AI Output and Repo Truth How to Build a Stripe Customer Portal in Next.js SaaS On-Demand Pricing Feels Safe - Until You See the Bill Building an Internal Developer Portal with Backstage A Production Deployment Guide After the Last Song Sudoers Configuration in Linux Terraform + Terragrunt + Ansible: A Hands-On Learning Journey Switching Users in Linux (su, sudo) AI 智能体的鲁莽速度 Quick Win Card #01 — Ton backlog.md t'a menti (la cure en 30 secondes) Quick Win Card #01 — Your backlog.md lied to you (a 30-second cure) How to Manage an IT Team: Structure, Scaling, and Daily Workflows That Work Speccing Is the New Coding CAC 250만 원을 뚫기 위해 퍼널 세 곳을 뜯어고친 3개월 Creating My First Token on Solana Devnet as a Web2 Developer Five Salesforce Reports Every Nonprofit Leadership Team Should Have Beyond the West: What Eastern AI Models Mean for Enterprises, Developers, and Digital Sovereignty Class and Pseudo Class Git & GitLab Basics 고객은 우리를 사기꾼으로 봤다: 아무도 믿지 않는 신사업을 단 둘이서 검증한 3개월 Cron Not Working on Mac? How to Fix the macOS Sleep Trap with launchd Cache Everything: Advanced Caching Strategies in Vue 3 & Nuxt 4 Deploy a Node.js App to STACKIT Kubernetes Engine With Managed Redis & PostgreSQL Slopsquatting & Remote Prompts: Why I Built a 38,000 Ticker Engine with Zero NPM Dependencies 05/20: TCP/IP vs OSI Model: The Ultimate Comparison My New Adventures in IT # Mitigating Market Inefficiency in eSports: A Stochastic Approach to EA Sports FC25 Modeling Don't let a billion RAG docs drown your 25-result pipeline Experienced devs are slower with AI tools. Nobody wants to admit it. I built an MCP-native OSINT framework that lets AI agents investigate from your terminal AWS Nitro Enclaves vs Intel TDX: Why Attestation Root Matters for Regulated Workloads Vibe Coding: Revolution or Risk in Software Development? - SmarterArticles S1E6 JSON Schema Explained: Validate Your API Data Before It Breaks Production Harness Tells Your Agent What to Do. GUI Agents Let It Actually Do It. Is AI actually replacing developers? Customizing Docker Images: Write Your First Dockerfile (2026) €40 n8n vs 28% weekly Anthropic quota. Which /goal layer should you actually run? Reviving glyph-v8: From a Forgotten Prototype to STRIDE - a Field-Aware Integer Coder 04/20: Data Encapsulation: How a Message Becomes Bits on the Wire Hướng Dẫn Thiết Lập Reasoning Proxy DeepSeek V4-Pro với Cursor (2026) Sofi Log #012: Agentic GDP — Solana Pay.sh & x402 Protocol Spec Input Types, Attributes, Self-Closing Tags, Hover Effect Absolute vs Relative Paths File Types (Regular, Directory, Link, Device, Socket, Pipe) From Arduino IDE to AVR GCC | AVR Bare Metal #1 Using Bitcoin as collateral without wrapping it: the design of a BTC collateral vault Unreal Engine 5 Skill System Architecture using GAS and GameplayTags 5 Things I Wish I Knew Before Building with Hermes Agent Thoughts on Codingame 2026 Spring challenge OUT WITH THE OLD IN WITH THE NEW Why are simple 1099 tax calculators online so horribly bloated? So I built my own "Why You're Not Getting Callbacks (It's Not Your Skills)" # How I Built a Retail Demand Forecasting App with Python and Streamlit Why We Deliberately Crush Lithium Batteries (UN38.3 Crush Testing Explained) Command History & Completion The Three-Body Problem: AI Code, Supply Chain Attacks, and the Talent Exodus 로컬 LLM 셋업 가이드 (v27) Building Better .NET Worker Services with Cursor Rules Generate Professional PDF Invoices via REST API — JSON In, PDF Out Redis: Big Keys Destroem o Desempenho Compartilhado Agentic AI for Cybersecurity: Autonomous Threat Detection and Response How to Automate Android Without Appium Cron vs systemd daemon: which one for Node.js? Designing XSLT transforms with parameters and multiple inputs I Downloaded Gemma4:e2b On My Macbook in 2 steps Building an Autonomous SRE Agent: From Raw Telemetry to Safe, AI-Driven Remediation The EU AI Act in 2026: Reading the Law After the Omnibus I had zero coding knowledge. Here is "RetroTube", a 2010 YouTube sandbox prototype I built using AI! How to Validate Environment Variables in TypeScript (and Why You Should) I Built a CLI Tool That Writes Better Git Commits Than I Do Transfer Fees, Metadata, and Soulbound Tokens: My First Real Token Experiments on Solana Stop Using Fetch() in React: A Better Way To Call Your Backend Creando un Tetris con JavaScript VI: Complicando el juego. DeepSeek's API Price Cut Changed My Claude Code and ChatGPT Math [Boost] Perl 🐪 Weekly #774 - Perl is too HOT How to Track AI Usage Without Losing Revenue (Complete Guide) 77 Rules Later: What Graduating Our First Stack Actually Looked Like RAG 시스템 실전 구축 (v26) When Premature Scaling Leads to Operator Burnout Multi-Repo Microservice Changes Are a Coordination Problem. I Solved It With AI Agent Teams. The Next Frontier: How Multi-Agent Systems are Redefining Productivity The Kimwolf Bust Just Outed Android Webcams as Botnet Fodder — Here's the Question Every Repurposed-Phone Camera Setup Has to Answer I'm an autonomous AI agent. I shipped 18 fixes to myself in one session. Building a Secure Future with Zero Trust Security Architecture Asynchronous Functions in Dart How I migrated magic-link login from Resend to AWS SES + Lambda five days before launch
Building a Multi-Channel Content Syndication Pipeline with EmDash Plugins
Tony Nguyen · 2026-05-25 · via DEV Community

Tony Nguyen

I recently built a content syndication plugin for EmDash that automatically distributes blog posts to Dev.to, LinkedIn, Medium, Hacker News, and email newsletters from a single publish action. Here's how the architecture works and what I learned about multi-platform API orchestration.

The Problem: Format Fragmentation and Timing Drift

Manual cross-posting breaks down in practice because each platform expects different formats and has different constraints:

  • Format fragmentation — HTML on your site, Markdown on Dev.to, rich text on LinkedIn, plain text for HN, HTML for email
  • Timing drift — manual workflows slip by days or weeks, defeating the purpose of coordinated launches
  • Metadata mismatch — canonical URLs, tags, and excerpts need to be correct per platform for SEO
  • No centralized tracking — you can't measure which channel drives the most traffic without a unified analytics layer

Architecture Overview

The pipeline runs entirely on Cloudflare Workers via EmDash's plugin system with four components:

Component Purpose Technology
Publish Hook Trigger on post status change EmDash plugin middleware
Format Renderer Convert to platform-specific formats Template engine + markdown-it
Channel Adapter Platform-specific API client Fetch API + OAuth tokens
Queue Manager Retry failed syndications D1 queue table
Analytics Tracker Log syndication events D1 events table

No external cron jobs or queue infrastructure needed — Workers' Queues (or KV with TTL) handles the orchestration.

Step 1: The Publish Hook

The plugin registers a middleware that fires on afterPostSave when status flips to 'published':

// emdash-plugin-syndication/hooks.js
export default {
  async afterPostSave(post, context) {
    if (post.status === 'published' && post.wasDraft) {
      await context.env.SYNDICATION_QUEUE.put(
        `syndicate:${post.slug}`,
        JSON.stringify({
          slug: post.slug,
          title: post.title,
          body: post.body,
          excerpt: post.excerpt,
          tags: post.tags,
          publishedAt: post.published_at,
          channels: ['devto', 'linkedin', 'medium', 'hn']
        }),
        { expirationTtl: 86400 }
      );
    }
    return post;
  }
};

Enter fullscreen mode Exit fullscreen mode

Key design decision: using post.wasDraft prevents re-syndication on edits to already-published posts. Without this guard, every content update would re-trigger the pipeline and duplicate content across platforms.

Step 2: Format Adapter Pattern

Each platform gets its own adapter that converts the internal HTML body to the target format:

const adapters = {
  devto: (post) => ({
    body_markdown: htmlToMarkdown(post.body),
    tags: post.tags.slice(0, 4),  // Dev.to max 4 tags
    canonical_url: `https://ai-kit.net/blog/${post.slug}`,
    published: true
  }),
  linkedin: (post) => ({
    commentator: 'urn:li:person:ai-kit',
    content: {
      article: {
        title: post.title,
        description: post.excerpt,
        source: 'https://ai-kit.net',
        thumbnailUrl: `https://ai-kit.net/og/${post.slug}.png`,
        canonicalUrl: `https://ai-kit.net/blog/${post.slug}`
      }
    },
    distribution: {
      feedDistribution: 'MAIN_FEED',
      targetEntities: []
    }
  }),
  medium: (post) => ({
    title: post.title,
    contentFormat: 'markdown',
    content: htmlToMarkdown(post.body),
    tags: post.tags.slice(0, 3),  // Medium max 3 tags
    publishStatus: 'public',
    canonicalUrl: `https://ai-kit.net/blog/${post.slug}`
  })
};

Enter fullscreen mode Exit fullscreen mode

Platform-specific quirks I ran into:

  • LinkedIn strips code blocks — their rich text API doesn't support <pre><code>. I convert code blocks to inline code spans, which is imperfect but preserves readability
  • Medium expects Gist embeds for code — plain code fences in Medium import create formatting issues. The adapter optionally wraps code blocks as Gist URLs
  • Dev.to loves code fences — standard triple-backtick fences work perfectly with syntax highlighting
  • Hacker News is plain text only — 2000 character limit, no markup, no images. Append the original story URL for context
  • LinkedIn API has rate limits — 100 posts per 24 hours, ~1 post per 14 minutes sustained

Step 3: Token Refresh Pattern

OAuth tokens expire and each platform handles expiry differently. Here's the refresh wrapper I built:

async function getValidToken(channel, context) {
  const token = await context.env.SECRETS.get(`${channel}_token`);
  const expiresAt = await context.env.SECRETS.get(`${channel}_expires_at`);

  if (token && expiresAt && Date.now() < parseInt(expiresAt)) {
    return token;
  }

  // Refresh token
  const refreshToken = await context.env.SECRETS.get(`${channel}_refresh_token`);
  const response = await refreshAccessToken(channel, refreshToken);

  await context.env.SECRETS.put(`${channel}_token`, response.access_token);
  await context.env.SECRETS.put(
    `${channel}_expires_at`,
    String(Date.now() + response.expires_in * 1000)
  );

  return response.access_token;
}

Enter fullscreen mode Exit fullscreen mode

One gotcha: LinkedIn's access tokens expire every 60 days with no refresh token for the Community Management API. You need the Marketing Developer Platform OAuth 2.0 flow which provides refresh tokens. Dev.to and Medium tokens are long-lived (not expiring), so their expires_at is set far in the future as a simple sentinel.

Step 4: Error Tracking with D1

Each syndication attempt is logged to a D1 events table for debugging and observability:

CREATE TABLE syndication_events (
  id INTEGER PRIMARY KEY AUTOINCREMENT,
  post_slug TEXT NOT NULL,
  channel TEXT NOT NULL,
  status TEXT NOT NULL,  -- 'success', 'failed', 'retrying'
  url TEXT,
  error TEXT,
  attempted_at INTEGER NOT NULL,
  retry_count INTEGER DEFAULT 0
);
CREATE INDEX idx_syndication_post ON syndication_events(post_slug);

Enter fullscreen mode Exit fullscreen mode

The pipeline retries failed channels up to 3 times with exponential backoff (30s, 2min, 10min). After exhausting retries, the event is marked as permanently failed and a notification is dispatched via Telegram or email.

Design Decisions Worth Calling Out

Staggered syndication timing

I intentionally sequence channels to optimize indexing and avoid spamming overlapping audiences:

  1. Dev.to and Medium first — they index quickly and cache syndicated copies
  2. LinkedIn next — slower to appear in feeds, so publishing earlier doesn't help
  3. Email digests last — avoids sending notifications to subscribers who may have already seen the post on another platform

Rate limiting per channel

Each platform has different API rate limits. Dev.to allows 5 API calls per minute, LinkedIn has daily post limits. The plugin maintains a per-channel rate limiter using D1 counters:

async function checkRateLimit(channel, context) {
  const key = `ratelimit:${channel}:${Math.floor(Date.now() / 60000)}`;
  const count = await context.env.SYNDICATION_QUEUE.get(key) || 0;
  const limit = RATE_LIMITS[channel];  // { max: 5, window: 60000 }
  if (count >= limit.max) throw new RateLimitError(channel);
  await context.env.SYNDICATION_QUEUE.put(key, count + 1, { expirationTtl: 120 });
}

Enter fullscreen mode Exit fullscreen mode

Canonical URL enforcement

Every syndicated copy includes a canonical_url pointing back to the original. This is critical for SEO — without it, syndicated copies can outrank the original for search queries. Dev.to and Medium support canonical URLs natively. LinkedIn's article API requires setting canonicalUrl in the payload.

Dry-run mode

Before hitting live APIs, the plugin includes a dry-run mode that returns the formatted payload without posting. You can preview exactly what each platform will receive in EmDash's admin UI:

if (options.dryRun) {
  return { channel, payload: adapter(post) };
}

Enter fullscreen mode Exit fullscreen mode

What I'd Do Differently Next Time

  1. Use a message queue from day one — I started with simple Promise.all across all channels. First LinkedIn API timeout blocked Dev.to and Medium. Sequential processing with a proper queue (Workers Queues) fixed this.
  2. Platform-specific test fixtures — each platform has subtle JSON schema differences that surface as 400s at runtime. Mocking the actual API responses in tests would have caught these earlier.
  3. Graceful degradation per channel — one platform being down shouldn't stop syndication to others. Each channel should be fully isolated in its own Worker invocation.

Metrics to Track

  • Syndication velocity — time from initial publish to full syndication across all channels (target: under 10 minutes)
  • Channel performance — which syndication channel drives the most referrer traffic back
  • Failure rate — percentage of attempts requiring retries. High failure rates indicate token issues or API changes
  • SEO impact — monitor whether syndicated copies outrank your canonical page. If they do, strengthen canonical tags or delay syndication by 24 hours

Building a content syndication plugin this way turns a single-platform CMS into a multi-channel distribution engine. Each published post automatically reaches Dev.to's developer audience, LinkedIn's professional network, Medium's general readership, Hacker News's tech community, and email subscribers — with zero manual effort after the initial publish click.