#!/usr/bin/env node import inquirer from 'inquirer'; import { fileURLToPath } from 'url'; import { dirname } from 'path'; import { execSync } from 'child_process'; import fs from 'fs'; import path from 'path'; import crypto from 'crypto'; const __filename = fileURLToPath(import.meta.url); const __dirname = dirname(__filename); const REPO_URL = 'https://github.com/neynarxyz/create-farcaster-mini-app.git'; const SCRIPT_VERSION = JSON.parse(fs.readFileSync(path.join(__dirname, '..', 'package.json'), 'utf8')).version; // ANSI color codes const purple = '\x1b[35m'; const yellow = '\x1b[33m'; const blue = '\x1b[34m'; const reset = '\x1b[0m'; const dim = '\x1b[2m'; const bright = '\x1b[1m'; const italic = '\x1b[3m'; function printWelcomeMessage() { console.log(` ${purple}╔═══════════════════════════════════════════════════╗${reset} ${purple}║ ║${reset} ${purple}║${reset} ${bright}Welcome to Mini Apps Quickstart by Neynar${reset} ${purple}║${reset} ${purple}║${reset} ${dim}the quickest way to build Farcaster mini apps${reset} ${purple}║${reset} ${purple}║ ║${reset} ${purple}╚═══════════════════════════════════════════════════╝${reset} ${blue}Version:${reset} ${SCRIPT_VERSION} ${blue}Repository:${reset} ${dim}${REPO_URL}${reset} Let's create your mini app! 🚀 `); } async function queryNeynarApp(apiKey) { if (!apiKey) { return null; } try { const response = await fetch( `https://api.neynar.com/portal/app_by_api_key?starter_kit=true`, { headers: { 'x-api-key': apiKey } } ); const data = await response.json(); return data; } catch (error) { console.error('Error querying Neynar app data:', error); return null; } } // Export the main CLI function for programmatic use export async function init() { printWelcomeMessage(); // Ask about Neynar usage let useNeynar = true; let neynarApiKey = null; let neynarClientId = null; let neynarAppName = null; let neynarAppLogoUrl = null; while (useNeynar) { const neynarAnswers = await inquirer.prompt([ { type: 'confirm', name: 'useNeynar', message: '🪐 Neynar is an API that makes it easy to build on Farcaster.\n\n' + 'Benefits of using Neynar in your mini app:\n' + '- Pre-configured webhook handling (no setup required)\n' + '- Automatic mini app analytics in your dev portal\n' + '- Send manual notifications from dev.neynar.com\n' + '- Built-in rate limiting and error handling\n\n' + `${purple}${bright}${italic}A demo API key is included if you would like to try out Neynar before signing up!${reset}\n\n` + 'Would you like to use Neynar in your mini app?', default: true } ]); if (!neynarAnswers.useNeynar) { useNeynar = false; break; } const neynarKeyAnswer = await inquirer.prompt([ { type: 'password', name: 'neynarApiKey', message: 'Enter your Neynar API key (or press enter to skip):', default: null } ]); if (neynarKeyAnswer.neynarApiKey) { neynarApiKey = neynarKeyAnswer.neynarApiKey; } else { const useDemoKey = await inquirer.prompt([ { type: 'confirm', name: 'useDemo', message: 'Would you like to try the demo Neynar API key?', default: true } ]); if (useDemoKey.useDemo) { console.warn('\n⚠️ Note: the demo key is for development purposes only and is aggressively rate limited.'); console.log('For production, please sign up for a Neynar account at https://neynar.com/ and configure the API key in your .env or .env.local file with NEYNAR_API_KEY.'); console.log(`\n${purple}${bright}${italic}Neynar now has a free tier! See https://neynar.com/#pricing for details.\n${reset}`); neynarApiKey = 'FARCASTER_V2_FRAMES_DEMO'; } } if (!neynarApiKey) { console.log('\n⚠️ No valid API key provided. Would you like to try again?'); const { retry } = await inquirer.prompt([ { type: 'confirm', name: 'retry', message: 'Try configuring Neynar again?', default: true } ]); if (!retry) { useNeynar = false; break; } continue; } const appInfo = await queryNeynarApp(neynarApiKey); if (appInfo) { neynarClientId = appInfo.app_uuid; neynarAppName = appInfo.app_name; neynarAppLogoUrl = appInfo.logo_url; } if (!neynarClientId) { const { retry } = await inquirer.prompt([ { type: 'confirm', name: 'retry', message: '⚠️ Could not find a client ID for this API key. Would you like to try configuring Neynar again?', default: true } ]); if (!retry) { useNeynar = false; break; } continue; } // If we get here, we have both API key and client ID break; } const defaultFrameName = (neynarAppName && !neynarAppName.toLowerCase().includes('demo')) ? neynarAppName : undefined; const answers = await inquirer.prompt([ { type: 'input', name: 'projectName', message: 'What is the name of your mini app?', default: defaultFrameName, validate: (input) => { if (input.trim() === '') { return 'Project name cannot be empty'; } return true; } }, { type: 'input', name: 'description', message: 'Give a one-line description of your mini app (optional):', default: 'A Farcaster mini app created with Neynar' }, { type: 'list', name: 'primaryCategory', message: 'It is strongly recommended to choose a primary category and tags to help users discover your mini app.\n\nSelect a primary category:', choices: [ new inquirer.Separator(), { name: 'Skip (not recommended)', value: null }, new inquirer.Separator(), { name: 'Games', value: 'games' }, { name: 'Social', value: 'social' }, { name: 'Finance', value: 'finance' }, { name: 'Utility', value: 'utility' }, { name: 'Productivity', value: 'productivity' }, { name: 'Health & Fitness', value: 'health-fitness' }, { name: 'News & Media', value: 'news-media' }, { name: 'Music', value: 'music' }, { name: 'Shopping', value: 'shopping' }, { name: 'Education', value: 'education' }, { name: 'Developer Tools', value: 'developer-tools' }, { name: 'Entertainment', value: 'entertainment' }, { name: 'Art & Creativity', value: 'art-creativity' } ], default: null }, { type: 'input', name: 'tags', message: 'Enter tags for your mini app (separate with spaces or commas, optional):', default: '', filter: (input) => { if (!input.trim()) return []; // Split by both spaces and commas, trim whitespace, and filter out empty strings return input .split(/[,\s]+/) .map(tag => tag.trim()) .filter(tag => tag.length > 0); } }, { type: 'input', name: 'buttonText', message: 'Enter the button text for your mini app:', default: 'Launch Mini App', validate: (input) => { if (input.trim() === '') { return 'Button text cannot be empty'; } return true; } } ]); // Ask about localhost vs tunnel const hostingAnswer = await inquirer.prompt([ { type: 'confirm', name: 'useTunnel', message: 'Would you like to test on mobile and/or test the app with Warpcast developer tools?\n' + `⚠️ ${yellow}${italic}Both mobile testing and the Warpcast debugger require setting up a tunnel to serve your app from localhost to the broader internet.\n${reset}` + 'Configure a tunnel for mobile testing and/or Warpcast developer tools?', default: true } ]); answers.useTunnel = hostingAnswer.useTunnel; const projectName = answers.projectName; const projectDirName = projectName.replace(/\s+/g, '-').toLowerCase(); const projectPath = path.join(process.cwd(), projectDirName); console.log(`\nCreating a new mini app in ${projectPath}`); // Clone the repository try { console.log(`\nCloning repository from ${REPO_URL}...`); // Use separate commands for better cross-platform compatibility execSync(`git clone ${REPO_URL} "${projectPath}"`, { stdio: 'inherit', shell: process.platform === 'win32' }); execSync('git fetch origin main', { cwd: projectPath, stdio: 'inherit', shell: process.platform === 'win32' }); execSync('git reset --hard origin/main', { cwd: projectPath, stdio: 'inherit', shell: process.platform === 'win32' }); } catch (error) { console.error('\n❌ Error: Failed to create project directory.'); console.error('Please make sure you have write permissions and try again.'); process.exit(1); } // Remove the .git directory console.log('\nRemoving .git directory...'); fs.rmSync(path.join(projectPath, '.git'), { recursive: true, force: true }); // Remove package-lock.json console.log('\nRemoving package-lock.json...'); const packageLockPath = path.join(projectPath, 'package-lock.json'); if (fs.existsSync(packageLockPath)) { fs.unlinkSync(packageLockPath); } // Update package.json console.log('\nUpdating package.json...'); const packageJsonPath = path.join(projectPath, 'package.json'); let packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8')); packageJson.name = projectName; packageJson.version = '0.1.0'; delete packageJson.author; delete packageJson.keywords; delete packageJson.repository; delete packageJson.license; delete packageJson.bin; delete packageJson.files; delete packageJson.dependencies; delete packageJson.devDependencies; // Add dependencies packageJson.dependencies = { "@farcaster/auth-client": ">=0.3.0 <1.0.0", "@farcaster/auth-kit": ">=0.6.0 <1.0.0", "@farcaster/frame-core": ">=0.0.29 <1.0.0", "@farcaster/frame-node": ">=0.0.18 <1.0.0", "@farcaster/frame-sdk": ">=0.0.31 <1.0.0", "@farcaster/frame-wagmi-connector": ">=0.0.19 <1.0.0", "@farcaster/mini-app-solana": "^0.0.5", "@radix-ui/react-label": "^2.1.1", "@solana/wallet-adapter-react": "^0.15.38", "@tanstack/react-query": "^5.61.0", "@upstash/redis": "^1.34.3", "class-variance-authority": "^0.7.1", "clsx": "^2.1.1", "dotenv": "^16.4.7", "lucide-react": "^0.469.0", "mipd": "^0.0.7", "next": "15.0.3", "next-auth": "^4.24.11", "react": "^18", "react-dom": "^18", "tailwind-merge": "^2.6.0", "tailwindcss-animate": "^1.0.7", "viem": "^2.23.6", "wagmi": "^2.14.12", "zod": "^3.24.2" }; packageJson.devDependencies = { "@types/node": "^20", "@types/react": "^18", "@types/react-dom": "^18", "crypto": "^1.0.1", "eslint": "^8", "eslint-config-next": "15.0.3", "localtunnel": "^2.0.2", "pino-pretty": "^13.0.0", "postcss": "^8", "tailwindcss": "^3.4.1", "typescript": "^5" }; // Add Neynar dependencies if selected if (useNeynar) { packageJson.dependencies['@neynar/nodejs-sdk'] = '^2.19.0'; packageJson.dependencies['@neynar/react'] = '^0.9.7'; } fs.writeFileSync(packageJsonPath, JSON.stringify(packageJson, null, 2)); // Handle .env file console.log('\nSetting up environment variables...'); const envExamplePath = path.join(projectPath, '.env.example'); const envPath = path.join(projectPath, '.env.local'); if (fs.existsSync(envExamplePath)) { // Read the example file content const envExampleContent = fs.readFileSync(envExamplePath, 'utf8'); // Write it to .env.local fs.writeFileSync(envPath, envExampleContent); // Append all remaining environment variables fs.appendFileSync(envPath, `\nNEXT_PUBLIC_FRAME_NAME="${answers.projectName}"`); fs.appendFileSync(envPath, `\nNEXT_PUBLIC_FRAME_DESCRIPTION="${answers.description}"`); fs.appendFileSync(envPath, `\nNEXT_PUBLIC_FRAME_PRIMARY_CATEGORY="${answers.primaryCategory}"`); fs.appendFileSync(envPath, `\nNEXT_PUBLIC_FRAME_TAGS="${answers.tags.join(',')}"`); fs.appendFileSync(envPath, `\nNEXT_PUBLIC_FRAME_BUTTON_TEXT="${answers.buttonText}"`); fs.appendFileSync(envPath, `\nNEXTAUTH_SECRET="${crypto.randomBytes(32).toString('hex')}"`); if (useNeynar && neynarApiKey && neynarClientId) { fs.appendFileSync(envPath, `\nNEYNAR_API_KEY="${neynarApiKey}"`); fs.appendFileSync(envPath, `\nNEYNAR_CLIENT_ID="${neynarClientId}"`); } else if (useNeynar) { console.log('\n⚠️ Could not find a Neynar client ID and/or API key. Please configure Neynar manually in .env.local with NEYNAR_API_KEY and NEYNAR_CLIENT_ID'); } fs.appendFileSync(envPath, `\nUSE_TUNNEL="${answers.useTunnel}"`); fs.unlinkSync(envExamplePath); console.log('\nCreated .env.local file from .env.example'); } else { console.log('\n.env.example does not exist, skipping copy and remove operations'); } // Update README console.log('\nUpdating README...'); const readmePath = path.join(projectPath, 'README.md'); const prependText = `\n\n`; if (fs.existsSync(readmePath)) { const originalReadmeContent = fs.readFileSync(readmePath, { encoding: 'utf8' }); const updatedReadmeContent = prependText + originalReadmeContent; fs.writeFileSync(readmePath, updatedReadmeContent); } else { fs.writeFileSync(readmePath, prependText); } // Install dependencies console.log('\nInstalling dependencies...'); execSync('npm cache clean --force', { cwd: projectPath, stdio: 'inherit', shell: process.platform === 'win32' }); execSync('npm install', { cwd: projectPath, stdio: 'inherit', shell: process.platform === 'win32' }); // Remove the bin directory console.log('\nRemoving bin directory...'); const binPath = path.join(projectPath, 'bin'); if (fs.existsSync(binPath)) { fs.rmSync(binPath, { recursive: true, force: true }); } // Initialize git repository console.log('\nInitializing git repository...'); execSync('git init', { cwd: projectPath }); execSync('git add .', { cwd: projectPath }); execSync('git commit -m "initial commit from @neynar/create-farcaster-mini-app"', { cwd: projectPath }); // Calculate border length based on message length const message = `✨🪐 Successfully created frame ${projectName} with git and dependencies installed! 🪐✨`; const borderLength = message.length; const borderStars = '✨'.repeat((borderLength / 2) + 1); console.log(`\n${borderStars}`); console.log(`${message}`); console.log(`${borderStars}`); console.log('\nTo run the app:'); console.log(` cd ${projectName}`); console.log(' npm run dev\n'); }