A ChatGPT App that brings high-quality, royalty-free photography from Pexels directly into your conversations. Built with the Model Context Protocol (MCP) and powered by Cloudflare Workers.
This project is a Model Context Protocol (MCP) server that integrates the Pexels API into ChatGPT. Users can search for professional stock photos directly in their chat conversations and view results in an interactive, responsive gallery widget.
- 🔍 Advanced Photo Search - Search by keywords, orientation, color palette, size, and more
- 🎨 Interactive Gallery Widget - Beautiful carousel interface with photo cards
- 🌓 Theme Aware - Automatically adapts to light/dark mode
- ♿ Accessible - Full screen reader support and keyboard navigation
- ⚡ Serverless - Runs on Cloudflare Workers edge network for global performance
- 📱 Responsive - Works seamlessly across desktop, tablet, and mobile
This project demonstrates a complete MCP application architecture:
┌─────────────────────────────────────────────────────────────┐
│ ChatGPT UI │
├─────────────────────────────────────────────────────────────┤
│ React Widget (Embedded Gallery) │
│ ↓ Uses OpenAI SDK │
├─────────────────────────────────────────────────────────────┤
│ MCP Protocol (SSE) │
│ ↓ Tool Invocation │
├─────────────────────────────────────────────────────────────┤
│ Cloudflare Worker (Durable Objects) │
│ ↓ HTTP Request │
├─────────────────────────────────────────────────────────────┤
│ Pexels API (https://api.pexels.com) │
└─────────────────────────────────────────────────────────────┘
Backend:
- Cloudflare Workers - Serverless edge computing
- Durable Objects - Stateful MCP server instances
- @modelcontextprotocol/sdk - MCP implementation
- TypeScript 5.9.3
Frontend:
- React 18.3.1 - UI framework
- embla-carousel-react - Carousel functionality
- lucide-react - Icon library
- esbuild - Fast bundler
External API:
- Pexels API - Free stock photography
pexels-app/
├── src/ # Backend (Cloudflare Worker)
│ ├── index.ts # MCP Durable Object & worker entry
│ ├── types.ts # Shared TypeScript types
│ ├── tools/
│ │ └── pexels.ts # Pexels search tool implementation
│ └── components/
│ ├── react-widget-inline.ts # Bundled React widget (auto-generated)
│ └── react-widget-resource.ts # HTML wrapper for widget
│
├── web/ # Frontend (React Widget)
│ ├── src/
│ │ ├── component.tsx # React entry point
│ │ ├── theme.tsx # Theme tokens & context
│ │ ├── components/
│ │ │ ├── App.tsx # Main gallery application
│ │ │ └── cards/ # Photo card components
│ │ └── hooks/
│ │ ├── use-openai-global.ts # OpenAI SDK integration
│ │ └── use-widget-state.ts # Persistent state management
│ └── dist/ # Build output
│
├── scripts/
│ └── inline-react-widget.js # Bundles React into Worker
│
├── docs/ # Comprehensive documentation
│ ├── ARCHITECTURE.md # System design
│ ├── DEPLOYMENT-GUIDE.md # Deployment instructions
│ └── ... # More guides
│
├── wrangler.jsonc # Cloudflare Worker config
├── .dev.vars.example # Environment variable template
└── package.json # Dependencies & scripts
- Node.js 18+ and npm
- Pexels API key (free)
- Cloudflare account (free tier available)
-
Clone the repository:
git clone https://github.com/yourusername/pexels-app.git cd pexels-app -
Install dependencies:
npm install cd web && npm install && cd ..
-
Configure environment variables:
cp .dev.vars.example .dev.vars
Edit
.dev.varsand add your Pexels API key:PEXELS_API_KEY=your_actual_api_key_here -
Build and run locally:
npm run dev
The MCP server will be available at
http://localhost:8787
-
Authenticate with Cloudflare:
npx wrangler login
-
Set production secrets:
npx wrangler secret put PEXELS_API_KEY # Enter your API key when prompted -
Deploy to Cloudflare:
npm run deploy
Your app will be live at
https://your-worker-name.workers.dev
Search the Pexels library with powerful filtering options:
Parameters:
| Parameter | Type | Required | Description |
|---|---|---|---|
query |
string | Yes | Search keywords (1-120 characters) |
page |
number | No | Page number (1-50, default: 1) |
perPage |
number | No | Results per page (1-30, default: 12) |
orientation |
string | No | landscape, portrait, or square |
size |
string | No | large, medium, or small |
color |
string | No | Hex code or color keyword |
locale |
string | No | ISO locale code (e.g., en-US) |
Example in ChatGPT:
User: Show me portrait photos of mountain landscapes
[ChatGPT invokes: pexels.searchPhotos({
query: "mountain landscapes",
orientation: "portrait",
perPage: 15
})]
[Interactive gallery widget displays with 15 photos]
Output:
- Text summary: "Found X Pexels photos for 'query'"
- Interactive gallery widget with photo cards
- Each card shows:
- High-quality photo preview
- Photographer attribution
- "View on Pexels" button (opens in new tab)
# Development
npm start # Start local dev server
npm run dev # Same as start
# Building
npm run build:widget # Build React widget only
npm run deploy # Build everything and deploy
# React Development (in web/ directory)
cd web
npm run build # Build React bundle
npm run watch # Watch mode for React changes
# Type Checking
npm run type-check # Check TypeScript types- Make changes to React widget (
web/src/) - Build widget:
cd web && npm run build - Inline bundle:
node scripts/inline-react-widget.js - Test locally:
npm run dev - Deploy:
npm run deploy
For faster iteration, use watch mode in a separate terminal:
cd web && npm run watchDevelopment (.dev.vars):
PEXELS_API_KEY=your_api_key_hereProduction (Cloudflare Secrets):
npx wrangler secret put PEXELS_API_KEY
npx wrangler secret list # View configured secretsKey settings in wrangler.jsonc:
{
"name": "pexels",
"main": "src/index.ts",
"compatibility_date": "2025-03-10",
"durable_objects": {
"bindings": [{
"class_name": "MyMCP",
"name": "MCP_OBJECT"
}]
},
"vars": {
"PEXELS_API_BASE_URL": "https://api.pexels.com/v1"
}
}Comprehensive documentation is available in the docs/ directory:
- ARCHITECTURE.md - How frontend and backend connect
- DEPLOYMENT-GUIDE.md - Complete deployment instructions
- PROJECT-TOUR.md - File-by-file walkthrough
- APP_DESIGN_GUIDELINES.md - OpenAI design principles
- METADATA-OPTIMIZATION.md - MCP metadata tuning
- FUTURE-UI-UX-GUIDE.md - UI styling best practices
- WRANGLER-SECRET-COMMANDS.md - Secret management
This project follows OpenAI's ChatGPT App design guidelines:
- Conversational - Seamlessly integrated into chat flow
- Intelligent - Context-aware tool invocation
- Simple - Focused, single-purpose interactions
- Responsive - Fast, lightweight, edge-optimized
- Accessible - Screen reader support, keyboard navigation
- System-First Design - Inherits ChatGPT's colors, fonts, and spacing
- Theme Tokens - Dynamic light/dark theme support
- Transparent Backgrounds - Blends naturally with host environment
- Component-Scoped Styles - No global CSS to avoid conflicts
- Minimal Branding - Only subtle accent colors on CTAs
The tool makes requests to the Pexels API v1:
Endpoint: GET https://api.pexels.com/v1/search
Authentication: Bearer token via Authorization header
Rate Limits: 200 requests/hour (free tier)
Response Format: Normalized from snake_case to camelCase for TypeScript
For full Pexels API documentation, visit: https://www.pexels.com/api/documentation/
useOpenAiGlobal(key) - Access OpenAI SDK globals
const theme = useOpenAiGlobal('theme'); // 'light' | 'dark'
const toolOutput = useOpenAiGlobal('toolOutput'); // Tool result datauseWidgetState(defaultState) - Persistent widget state
const [state, setState] = useWidgetState({ page: 1 });
// State survives widget remountsuseThemeTokens() - Access theme design tokens
const tokens = useThemeTokens();
// tokens.colors.background, tokens.fonts.body, etc.- Full TypeScript coverage with strict mode
- Zod validation for all tool inputs
- Shared types between frontend and backend
- Automatic type inference from OpenAI SDK
Graceful handling of:
- API authentication failures
- Network errors
- Missing configuration
- Invalid parameters
- Empty search results
- User-friendly error messages in widget
The widget enforces a strict CSP:
default-src 'none';
script-src 'self' 'unsafe-inline';
style-src 'self' 'unsafe-inline';
img-src 'self' data: https://images.pexels.com;
frame-ancestors 'none';
- API keys stored as Cloudflare secrets (never in code)
.dev.vars.exampleprovided as template- Input validation with Zod schemas
This project is licensed under the MIT License - see the LICENSE file for details.
- Pexels - For providing free, high-quality stock photography
- Cloudflare Workers - For serverless infrastructure
- OpenAI - For ChatGPT and the MCP specification
- Model Context Protocol - For the protocol specification
Built with ❤️ using the Model Context Protocol
