├── readme.md ├── package.json ├── tsconfig.json ├── src └── index.ts └── .gitignore /readme.md: -------------------------------------------------------------------------------- 1 | # Farcaster/Warpcast Frames with Express 2 | 3 | Simple express app to post frames. 4 | 5 | Test your frames with the [warpcast embed page](https://warpcast.com/~/developers/embeds) 6 | - You cannot test the redirect (loading a new frame via button click) in the embed, you have to tweet 7 | 8 | ## Frames Quirks 9 | - Frames don't like to render if the route is "/", so target e.g. "/frame" 10 | - Frames require `````` 11 | - Frames don't like to render at port 3000? 12 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "frames-express", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "test": "echo \"Error: no test specified\" && exit 1", 8 | "build": "tsc -p .", 9 | "dev": "nodemon --watch 'src/**/*.ts' --exec 'ts-node' src/index.ts", 10 | "start": "node dist/index.js" 11 | }, 12 | "dependencies": { 13 | "cors": "^2.8.5", 14 | "express": "^4.17.1" 15 | }, 16 | "devDependencies": { 17 | "@types/cors": "^2.8.17", 18 | "@types/express": "^4.17.9", 19 | "@types/node": "^14.14.10", 20 | "nodemon": "^2.0.6", 21 | "ts-node": "^10.9.1", 22 | "typescript": "^5.3.3" 23 | }, 24 | "keywords": [], 25 | "author": "", 26 | "license": "ISC", 27 | "engines": { 28 | "node": "16.x" 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es6", // Compile to ES6 4 | "module": "commonjs", // Use commonjs module system 5 | "lib": ["es6"], // Include the ES6 library 6 | "outDir": "./dist", // Output to 'dist' folder 7 | "rootDir": "./src", // Root directory is 'src' 8 | "strict": true, // Enable all strict type-checking options 9 | "moduleResolution": "node", // Module resolution strategy 10 | "esModuleInterop": true, // Enables __importDefault helper 11 | "skipLibCheck": true, // Skip type checking of all declaration files (*.d.ts) 12 | "forceConsistentCasingInFileNames": true // Disallow inconsistently-cased references to the same file 13 | }, 14 | "include": [ 15 | "src/**/*.ts" // Include all TypeScript files in src 16 | ], 17 | "exclude": [ 18 | "node_modules", // Exclude the node_modules directory 19 | "**/*.spec.ts" // Exclude test files 20 | ] 21 | } -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import express from 'express'; 2 | import cors from 'cors'; 3 | 4 | const app = express(); 5 | app.use(express.json()); 6 | app.use(cors({ 7 | origin: '*' 8 | })); 9 | 10 | const port = 8080; 11 | 12 | interface IFrameProps { 13 | frame?: string; 14 | imageUrl: string; 15 | buttons?: string[]; 16 | postUrl?: string; 17 | } 18 | 19 | function generateFarcasterFrameMetaTag({ frame, imageUrl, postUrl, buttons }: IFrameProps): string { 20 | // Default to vNext 21 | if (!frame) { 22 | frame = "vNext" 23 | } 24 | // Ensure there are at most four buttons 25 | if (buttons && buttons.length > 4) { 26 | throw new Error("Maximum of four buttons are allowed per frame."); 27 | } 28 | 29 | // Generate Open Graph tags for image, redirect, and buttons 30 | let metaTag = `\n`; 31 | metaTag += `\n`; 32 | 33 | if (buttons) { 34 | buttons.forEach((button, index) => { 35 | metaTag += `\n`; 36 | }); 37 | } 38 | 39 | // post URL if exists 40 | if (postUrl) { 41 | metaTag += ` \n` 42 | } 43 | 44 | return metaTag; 45 | } 46 | 47 | function frameGenerator(frameProps: IFrameProps): string { 48 | 49 | const metaTag = generateFarcasterFrameMetaTag(frameProps); 50 | 51 | const html = ` 52 | 53 |
54 | 55 |