stevennshandleTelegramMessage.ts7 matches
9293/**
94* Format chat history for Anthropic API
95*/
96function formatChatHistoryForAI(history) {
321bot.on("message", async (ctx) => {
322try {
323// Get Anthropic API key from environment
324const apiKey = Deno.env.get("ANTHROPIC_API_KEY");
325if (!apiKey) {
326console.error("Anthropic API key is not configured.");
327ctx.reply(
328"I apologize, but I'm not properly configured at the moment. Please inform the household administrator."
332333// Initialize Anthropic client
334const anthropic = new Anthropic({ apiKey });
335336// Get message text and user info
502// Set webhook if it is not set yet
503if (!isEndpointSet) {
504await bot.api.setWebhook(req.url, {
505secret_token: SECRET_TOKEN,
506});
stevennsgetWeather.ts5 matches
27async function generateConciseWeatherSummary(weatherDay) {
28try {
29// Get API key from environment
30const apiKey = Deno.env.get("ANTHROPIC_API_KEY");
31if (!apiKey) {
32console.error("Anthropic API key is not configured.");
33return null;
34}
3536// Initialize Anthropic client
37const anthropic = new Anthropic({ apiKey });
3839const response = await anthropic.messages.create({
stevennsgenerateFunFacts.ts5 matches
77async function generateFunFacts(previousFacts) {
78try {
79// Get API key from environment
80const apiKey = Deno.env.get("ANTHROPIC_API_KEY");
81if (!apiKey) {
82console.error("Anthropic API key is not configured.");
83return null;
84}
8586// Initialize Anthropic client
87const anthropic = new Anthropic({ apiKey });
8889// Format previous facts for the prompt
stevenns.cursorrules10 matches
20### 2. HTTP Vals
2122- Create web APIs and endpoints
23- Handle HTTP requests and responses
24- Example structure:
66- Generate code in TypeScript
67- Add appropriate TypeScript types and interfaces for all data structures
68- Prefer official SDKs or libraries than writing API calls directly
69- Ask the user to supply API or library documentation if you are at all unsure about it
70- **Never bake in secrets into the code** - always use environment variables
71- Include comments explaining complex logic (avoid commenting obvious operations)
190- For AI-generated images, use: `https://maxm-imggenurl.web.val.run/the-description-of-your-image`
191- **Storage:** DO NOT use the Deno KV module for storage
192- **Browser APIs:** DO NOT use the `alert()`, `prompt()`, or `confirm()` methods
193- **Weather Data:** Use open-meteo for weather data (doesn't require API keys) unless otherwise specified
194- **View Source:** Add a view source link with `import.meta.url.replace("esm.town", "val.town")` and include `target="_top"` attribute
195- **Error Debugging:** Add `<script src="https://esm.town/v/std/catch"></script>` to HTML to capture client-side errors
196- **Error Handling:** Only use try...catch when there's a clear local resolution; avoid catches that merely log or return 500s - let errors bubble up with full context
197- **Environment Variables:** Use `Deno.env.get('keyname')` and minimize their use - prefer APIs without keys
198- **Imports:** Use `https://esm.sh` for npm and Deno dependencies to ensure compatibility on server and browser
199- **Storage Strategy:** Only use backend storage if explicitly required; prefer simple static client-side sites
230231### Backend (Hono) Best Practices
232- Hono is the recommended API framework (similar to Express, Flask, or Sinatra)
233- Main entry point should be `backend/index.ts`
234- **Static asset serving:** Use the utility functions to read and serve project files:
251});
252```
253- Create RESTful API routes for CRUD operations
254- Be careful with error handling as Hono tends to swallow errors
255- Always include this snippet at the top-level Hono app to re-throwing errors to see full stack traces:
268- Use React 18.2.0 consistently in all imports and the `@jsxImportSource` pragma
269- Follow the React component pattern from the example project
270- Handle API calls properly with proper error catching
271272### Database Patterns
299- For files in the project, use `readFile` helpers
3003015. **API Design:**
302- `fetch` handler is the entry point for HTTP vals
303- Run the Hono app with `export default app.fetch // This is the entry point for HTTP vals`
10import { NotebookView } from "./NotebookView.tsx";
1112const API_BASE = "/api/memories";
13const MEMORIES_PER_PAGE = 20; // Increased from 7 to 20 memories per page
149091// Fetch avatar image
92fetch("/api/images/stevens.jpg")
93.then((response) => {
94if (response.ok) return response.blob();
104105// Fetch wood background
106fetch("/api/images/wood.jpg")
107.then((response) => {
108if (response.ok) return response.blob();
133setError(null);
134try {
135const response = await fetch(API_BASE);
136if (!response.ok) {
137throw new Error(`HTTP error! status: ${response.status}`);
176177try {
178const response = await fetch(API_BASE, {
179method: "POST",
180headers: { "Content-Type": "application/json" },
199200try {
201const response = await fetch(`${API_BASE}/${id}`, {
202method: "DELETE",
203});
231232try {
233const response = await fetch(`${API_BASE}/${editingMemory.id}`, {
234method: "PUT",
235headers: { "Content-Type": "application/json" },
606<div className="font-pixel text-[#f8f1e0]">
607<style jsx>{`
608@import url("https://fonts.googleapis.com/css2?family=Pixelify+Sans&display=swap");
609610@tailwind base;
6* loading animation, EN/ES localization.
7* Uses 'npm:pdf.js-extract' for PDF extraction.
8* Serves HTML UI & API endpoint from the same Val.
9* Assumes 'openai' secret is set in Val Town environment variables.
10*
215<title data-translate="pageTitle">${currentLocaleStrings.pageTitle}</title>
216<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0/css/all.min.css">
217<link rel="preconnect" href="https://fonts.googleapis.com">
218<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
219<link href="https://fonts.googleapis.com/css2?family=Cinzel:wght@600&family=Segoe+UI:wght@400;600&display=swap" rel="stylesheet">
220<style>
221/* --- Full CSS from original, or templatized CSS variables --- */
651async function extractPdfTextNative(data: ArrayBuffer, fileName: string, log: LogEntry[]): Promise<string | null> { /* ... original ... */ }
652653// --- Helper Function: Call OpenAI API (Uses APP_CONFIG for model) ---
654async function callOpenAI(
655openai: OpenAI,
857* and `{{app_config.document_format_accepted_label}}` (e.g. "PDF") for UI text.
858*
859* 5. OpenAI API Key:
860* Ensure your environment (e.g., Val Town secrets) has the `OPENAI_API_KEY` (or the appropriate
861* environment variable name for the `OpenAI` library) set.
862*
863* 6. Deployment:
864* This script is designed to be deployed as a single serverless function (e.g., on Val Town).
865* It serves both the HTML interface and the backend API endpoint.
866*
867* --- Customization Notes ---
76// --- Core Lead Generation Logic ---
7778// WARNING: Scraping search engines like DuckDuckGo is fragile and might break.
79async function searchWebsites(searchQuery: string, jobId: string): Promise<{ name: string; url: string }[]> {
80const searchUrl = `https://html.duckduckgo.com/html/?q=${encodeURIComponent(searchQuery)}`;
125126async function extractEmailsFromWebsite(websiteUrl: string, leadId: string): Promise<string | null> {
127console.log(`Scraping ${websiteUrl} for lead ${leadId}`);
128try {
129// Add http:// if no scheme
139if (!response.ok) {
140console.warn(`Failed to fetch ${properUrl} for lead ${leadId}: ${response.status}`);
141await updateLeadStatus(leadId, "error_scraping_fetch", {}, `HTTP ${response.status} for ${properUrl}`);
142return null;
143}
183return null;
184} catch (error) {
185console.error(`Error scraping ${websiteUrl} (processed as ${websiteUrl}) for lead ${leadId}:`, error);
186await updateLeadStatus(leadId, "error_scraping", {}, `Scrape exception: ${error.message}`);
187return null;
188}
279} catch (error) {
280console.error(`Error drafting email for ${emailAddress} (lead ${leadId}):`, error);
281await updateLeadStatus(leadId, "error_drafting", {}, `OpenAI API error: ${error.message}`);
282return null;
283}
286async function processJob(jobId: string, searchQuery: string) {
287await updateJobStatus(jobId, "processing_search");
288const openaiApiKey = Deno.env.get("openai");
289if (!openaiApiKey) {
290console.error("OpenAI API key secret 'openai' is not set.");
291await updateJobStatus(jobId, "failed_config", "OpenAI API key not configured.");
292return;
293}
294const openai = new OpenAI({ apiKey: openaiApiKey });
295296const websites = await searchWebsites(searchQuery, jobId);
380.status-pending, .status-processing_search, .status-processing_leads { background-color: #ffc107; color: #212529;}
381.status-completed { background-color: #28a745; }
382.status-failed_config, .status-error_searching, .status-completed_no_sites, .status-error_scraping, .status-error_drafting, .status-error_scraping_fetch, .status-error_drafting_empty_response, .status-failed_uncaught_exception { background-color: #dc3545; }
383.status-email_extracted, .status-email_drafted { background-color: #17a2b8; }
384.status-no_email_found, .status-pending_scrape { background-color: #6c757d; }
484metrics.total = leads.length;
485metrics.emailFound = leads.filter(l =>
486l.email_address && l.status !== "error_scraping_fetch" && l.status !== "error_scraping"
487).length;
488metrics.emailDrafted = leads.filter(l => l.drafted_email_body && l.status !== "error_drafting").length;
MiniAppStarterneynar.ts2 matches
1const baseUrl = "https://api.neynar.com/v2/farcaster/";
23export async function fetchNeynarGet(path: string) {
7"Content-Type": "application/json",
8"x-neynar-experimental": "true",
9"x-api-key": "NEYNAR_API_DOCS",
10},
11});
MiniAppStarterindex.tsx2 matches
19}));
2021app.get("/api/counter/get", async c => c.json(await db.get("counter")));
22app.get("/api/counter/increment", async c => c.json(await db.set("counter", (await db.get("counter") || 0) + 1)));
2324app.get("/frontend/**/*", c => serveFile(c.req.path, import.meta.url));
MiniAppStarterimage.tsx3 matches
8485const loadEmoji = (code) => {
86// const api = `https://cdnjs.cloudflare.com/ajax/libs/twemoji/14.0.2/svg/${code.toLowerCase()}.svg`
87const api = `https://cdn.jsdelivr.net/gh/shuding/fluentui-emoji-unicode/assets/${code.toLowerCase()}_color.svg`;
88return fetch(api).then((r) => r.text());
89};
90