stevensDemohandleUSPSEmail.ts5 matches
85console.log(e.text);
8687// Get Anthropic API key from environment
88const apiKey = Deno.env.get("ANTHROPIC_API_KEY");
89if (!apiKey) {
90console.error("Anthropic API key is not configured for this val.");
91return;
92}
9394// Initialize Anthropic client
95const anthropic = new Anthropic({ apiKey });
9697// Process each image attachment serially
stevensDemohandleTelegramMessage.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});
stevensDemogetWeather.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({
stevensDemogenerateFunFacts.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
stevensDemo.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`
stevensDemoApp.tsx8 matches
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;
sqliteExplorerAppREADME.md1 match
13## Authentication
1415Login to your SQLite Explorer with [password authentication](https://www.val.town/v/pomdtr/password_auth) with your [Val Town API Token](https://www.val.town/settings/api) as the password.
1617## Todos / Plans
sqliteExplorerAppmain.tsx2 matches
27<head>
28<title>SQLite Explorer</title>
29<link rel="preconnect" href="https://fonts.googleapis.com" />
3031<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
32<link
33href="https://fonts.googleapis.com/css2?family=Fira+Code:wght@300..700&family=Source+Sans+3:ital,wght@0,200..900;1,200..900&display=swap"
34rel="stylesheet"
35/>
lotusReflectionsGetUsermain.tsx6 matches
34// Environment Variables to set in Val.Town for this specific val:
5// - NOTION_KEY: Your Notion API Key
6// - OPENAI_KEY: Your OpenAI API Key
7// - NOTION_USERS_DATABASE_ID: The Database ID for your "Reflection Users" database
8// - NOTION_DATABASE_ID: The Database ID for your "Reflections" database (submissions)
103};
104105const userSearchResponse = await fetch(`https://api.notion.com/v1/databases/${NOTION_USERS_DB_ID}/query`, {
106method: "POST",
107headers: {
142143const reflectionSearchResponse = await fetch(
144`https://api.notion.com/v1/databases/${NOTION_REFLECTIONS_DB_ID}/query`,
145{
146method: "POST",
206console.log("getUserReflectionContext: Calling OpenAI for personalized welcome...");
207try {
208const openAIResponse = await fetch("https://api.openai.com/v1/chat/completions", {
209method: "POST",
210headers: { "Content-Type": "application/json", "Authorization": `Bearer ${OPENAI_KEY}` },
214if (!openAIResponse.ok) {
215const errorBody = await openAIResponse.text();
216console.error("getUserReflectionContext: OpenAI API Error:", openAIResponse.status, errorBody);
217welcomeMessage = DEFAULT_RETURNING_USER_MESSAGE;
218} else {
lotusReflectionsmain.tsx17 matches
45// Environment Variables to set in Val.Town:
6// - OPENAI_KEY: Your OpenAI API Key
7// - NOTION_KEY: Your Notion API Key
8// - NOTION_DATABASE_ID: The Database ID for your "Reflections" database (submissions)
9// - NOTION_USERS_DATABASE_ID: The Database ID for your "Reflection Users" database
180};
181try {
182const openAIResponse = await fetch("https://api.openai.com/v1/chat/completions", {
183method: "POST",
184headers: { "Content-Type": "application/json", "Authorization": `Bearer ${OPENAI_KEY}` },
187if (!openAIResponse.ok) {
188const errorBody = await openAIResponse.text();
189console.error("Main Submission Handler: OpenAI API Error:", openAIResponse.status, errorBody);
190}
191else {
221}
222} catch (aiError) {
223console.error("Main Submission Handler: Error calling OpenAI API:", aiError);
224}
225239if (normalizedUserNumberFromForm) {
240try {
241const searchResponse = await fetch(`https://api.intercom.io/contacts/search`, {
242method: "POST",
243headers: {
251if (!searchResponse.ok) {
252const errorBody = await searchResponse.text();
253console.error("Main Submission Handler: Intercom Search API Error:", searchResponse.status, errorBody);
254}
255else {
307}
308309const eventResponse = await fetch("https://api.intercom.io/events", {
310method: "POST",
311headers: {
327else console.log("Main Submission Handler: Intercom event submitted successfully.");
328329const contactDetailsResponse = await fetch(`https://api.intercom.io/contacts/${intercomContactId}`, {
330headers: {
331"Authorization": `Bearer ${INTERCOM_KEY}`,
380// last_notion_reflection_url will be added later after Notion save
381382const updateResponse = await fetch(`https://api.intercom.io/contacts/${intercomContactId}`, {
383method: "PUT",
384headers: {
411}
412try {
413const tagResponse = await fetch(`https://api.intercom.io/contacts/${intercomContactId}/tags`, {
414method: "POST",
415headers: {
434}
435} catch (taggingError) {
436console.error("Main Submission Handler: Error during tag application API call:", taggingError);
437}
438} else {
475};
476console.log("Main Submission Handler: Notion User Query Payload:", JSON.stringify(userQueryPayload, null, 2));
477const userSearchResponse = await fetch(`https://api.notion.com/v1/databases/${NOTION_USERS_DB_ID}/query`, {
478method: "POST",
479headers: {
499if (intercomProfileUrl && userSearchData.results[0].properties.IntercomProfile?.url !== intercomProfileUrl) {
500console.log("Main Submission Handler: Attempting to update IntercomProfile URL on existing Notion User.");
501await fetch(`https://api.notion.com/v1/pages/${notionUserPageId}`, {
502method: "PATCH",
503headers: {
528JSON.stringify(newUserProperties, null, 2),
529);
530const createUserResponse = await fetch("https://api.notion.com/v1/pages", {
531method: "POST",
532headers: {
592);
593594const reflectionNotionResponse = await fetch("https://api.notion.com/v1/pages", {
595method: "POST",
596headers: {
629},
630};
631const updateNotionUrlResponse = await fetch(`https://api.intercom.io/contacts/${intercomContactId}`, {
632method: "PUT",
633headers: {