How I migrated a WordPress site to Cloudflare Pages using AI (and what broke) A developer migrated a WordPress site to Cloudflare Pages using AI assistance, with Claude converting XML exports to Markdown and generating migration scripts. The process revealed that 80% of posts converted cleanly, while 20% required manual fixes for shortcodes like contact forms and galleries. WordPress does everything. That's both the problem and the solution. I've spent years in the WordPress ecosystem: plugins, themes, WooCommerce, Gutenberg, the whole thing. It's a mature, powerful platform that handles complex cases really well. I'm not here to say it's bad. But with AI there's a category of site where it became overkill: landing pages, product sites, company pages, portfolios. Pages that are essentially static content with a contact form. You're running a full PHP stack, managing plugin updates, paying for server hosting, all for a page that could be pure HTML served from the edge. For those cases, I've started migrating everything. Destination: Cloudflare Pages. The process: largely AI-assisted. Here's how it went. A few reasons it won over Vercel, Netlify, and GitHub Pages for this project: If you're already on Cloudflare for DNS most people are , the migration path is shorter than it looks. First step: understand what you actually have. Export everything from WordPress Admin → Tools → Export → All content Downloads an XML file The XML export gives you posts, pages, categories, tags, authors, and media references. It doesn't give you the actual media files — those come separately. I fed the XML export to Claude and asked it to: The output was a clean inventory in about 30 seconds. On a large site this alone saves hours. Prompt I used: Here's a WordPress XML export. Give me: 1. A table of all posts: title, slug, publish date, category, word count 2. A list of all shortcodes used across the content 3. Any embedded iframes or third-party scripts you can identify Format as markdown. WordPress stores content as HTML with shortcodes mixed in. Astro wants Markdown with frontmatter. Claude handled most of this conversion automatically. Prompt for content conversion: Convert this WordPress post HTML to Markdown with Astro frontmatter. Preserve all headings, links, images, and formatting. Output the frontmatter with: title, description, pubDate, slug, tags. Flag any shortcodes you can't convert with a MANUAL REVIEW NEEDED comment. For 80% of posts, the output was clean and ready to use. The remaining 20% had: contact-form-7 — needed replacement gallery ids="..." — needed manual rebuildThe shortcode problem is where most WordPress migrations get stuck. More on the form replacement below. npm create astro@latest my-site cd my-site npx astro add cloudflare The Cloudflare adapter handles the build output format for Pages. After that, connect your GitHub repo to Cloudflare Pages and it deploys on every push. Basic astro.config.mjs : python import { defineConfig } from 'astro/config' import cloudflare from '@astrojs/cloudflare' export default defineConfig { output: 'static', adapter: cloudflare , } For a pure static site, output: 'static' is what you want. No server-side rendering, no cold starts, pure HTML at the edge. I asked Claude to generate a migration script that: .md files for each post with proper frontmatter js // Claude-generated migration script simplified import { parseStringPromise } from 'xml2js' import { writeFileSync, mkdirSync } from 'fs' import { slugify } from './utils' const xml = readFileSync 'export.xml', 'utf-8' const data = await parseStringPromise xml const posts = data.rss.channel 0 .item for const post of posts { const title = post.title 0 const content = post 'content:encoded' 0 const date = post.pubDate 0 const slug = post 'wp:post name' 0 const frontmatter = --- title: "${title}" pubDate: ${new Date date .toISOString } slug: "${slug}" ---\n\n writeFileSync src/content/blog/${slug}.md , frontmatter + convertToMarkdown content } For media, I kept the files in the Cloudflare Pages public folder and updated the references in the Markdown files. For larger sites with heavy media, R2 is worth looking at, but for most company pages and landing pages, serving assets directly from Pages is enough. Claude generated the find-and-replace script for URL rewriting. This is where every WordPress-to-static migration hits the same wall. Contact Form 7, Gravity Forms, WPForms — they all depend on PHP running on the server. Move to static and they stop working entirely. There's no server to process the submission. The options I evaluated: Write a Cloudflare Worker — possible, but you're now maintaining a backend for what is essentially a contact form. Wiring up email delivery, validation, spam protection. More moving parts than I wanted. Use a form backend service — drop one endpoint into the form, the service handles everything. No server, no maintenance. I went with FormRoute https://formroute.dev . The migration from Contact Form 7 was straightforward: Before Contact Form 7 shortcode : contact-form-7 id="123" title="Contact form" After Astro component : --- // src/components/ContactForm.astro ---