ValTownForNotion1notionHelpers.ts23 matches
7});
89export async function createDatabasePagesParallel(databaseId: string, pages: []) {
10const creations = pages.map((page) =>
11notion.pages.create({
12parent: { database_id: databaseId },
13properties: {
14"Website": {
42}
4344// delete all database rows
45export async function deleteAllDatabasePagesParallel(databaseId: string) {
46const pageIds = await notion.databases.query({
47database_id: databaseId,
48});
49// Step 2: Delete all pages in parallel
59}
6061export async function getDatabaseId(databaseTitle: string) {
62// getDatabaseId(databaseTitle)
63const database = await notion.databases
64.query({ filter: { property: "title", title: { equals: databaseTitle } } });
65return database.results[0].id;
66}
6768export async function getDatabaseTitle(databaseId: string) {
69// getDatabaseTitle(databaseId)
70const database = await notion.databases.retrieve({ database_id: databaseId });
71// console.log(database.title[0].plain_text);
72return database.title[0].plain_text;
73}
7475export async function getDatabaseParentPageId(databaseId: string) {
76let currentBlockId = databaseId;
7778while (true) {
121}
122123export async function findChildDatabaseBlocks(rootBlockId: string, blockIdentifier: string) {
124const matchingBlocks = [];
125140// }
141if (
142block.type === "child_database" && block.child_database?.title && (!block.archived || !block.in_trash)
143) {
144const databaseTitle = await helpers.slugify(block.child_database?.title.toLowerCase()) || "";
145const identifier = await helpers.slugify(blockIdentifier?.toLowerCase());
146147if (databaseTitle === identifier) {
148console.log(databaseTitle, identifier);
149matchingBlocks.push(block);
150}
20const data = payload?.data;
21const pageId = data?.id;
22const databaseId = data?.parent?.database_id;
2324// this endpoint is for a specific database page and is triggered by a button in that database page
2526// the button that calls this endpoint sends a payload with the id of the database page, and the id of the parent database
27// (e.g., pageId and databaseId from the payload)
2829// different than other endpoints, this one does not get called by a reset button or the cron reset
30// instead, the reset button and the reset cron target the database, not the page
31// so, we don't have a reset call coming in with the x-blob-key custom header to this enpoint
32// nor do we have a x-container-title header that holds the title of the target object coming from the button payload
3334// both the reset button and the reset cron delete all database pages, and then add back fresh pages
35// the endpoint for that is /example-database-pages (note the "s" at the end)
3637// different than other endpoints, we get the id of the block to modify (ie, the database page) from the button payload
38// so, we don't need to do the things we do at other endpoints to get the id of target block
39// (e.g., search the blob cache, or traverse notion page objects)
4041// but, we do need to set the blob for the resets
42// and for that we'll need the id of the page in which this database lives
43// and an identifier for the database
4445// can that idenfifier be the databaseId rather than the database title?
46// the reset button sends the database title to the .../pages endpoint in order to find the id of the database
47// but the matching there will traverse the notion page database objects looking for a title match
48// which has nothing to do with the database title in the blob
4950// note: if you're wondering why none of the notion buttons have hard-coded ids for the targeting
53// as is, i can clone a page with examples for a guest and everything will work without anyone needing to edit button settings
5455// when a user adds a favicon to a database page, we need to capture in the blob cache that a page in this database has been updated
56// so that resets know which database to reset
57// but the payload of the favicon button in this page does _not_ send the id of the page in which the database lives
58// nor does it send the database title
59// so we'll need to get those to make the blob key
6067for (const item of items) {
68const blobData = await blob.getJSON(item.key);
69if (blobData?.id === databaseId) {
70// console.log(item.key, blobData);
71blobData.key = item.key;
8283// blob for cache
84// get the parent page id of this child_database
85// this is the user's page and we need it for the blob key
86const parentPageId = blobject?.user
87? blobject?.user?.page_id
88: await notionHelpers.getDatabaseParentPageId(databaseId);
8990// we also need the database title
91// even though we have the database id and it's tempting to use it here
92// we need the title in the blob key so that the reset button can work
93// b/c the reset button can only send the title in the x-container-title header
94// with no other way to tie the button to the database
9596// const databaseTitle = blobject?.key
97// ? (blobject?.key).split("--").pop()
98// : await notionHelpers.getDatabaseTitle(databaseId);
99const databaseTitle = await notionHelpers.getDatabaseTitle(databaseId);
100101console.log("databaseTitle:", databaseTitle);
102103// store webhook data in blob storage for resets
105"slug": slug,
106"clientPageId": parentPageId,
107// "containerId": databaseId, // slugify databaseTitle here?
108"containerId": await helpers.slugify(databaseTitle),
109});
110await blob.setJSON(blobKey, {
113page_id: parentPageId,
114},
115id: databaseId,
116date: new Date(),
117content: askingFor,
16const slug = await c.req.url.replace("https://", "");
17const askingFor = c.req.headers.get("x-asking-for") || "default"; // val.town or default
18const containerTitle = c.req.headers.get("x-container-title") || "good web sites"; // should map to the database title
19const payload = await c.req.json();
20const pageId = payload?.data?.id;
3839// when the reset button is clicked, it sends the x-container-title header
40// we use this header to a) find a blob or b) if no blob, find a database with a title that matches it
41// a) find a blob
42// b) find a database with a title that matches it
4344// blob
64// console.log("blockId:", blockId);
6566const databaseId = blobject?.id || (await notionHelpers.findChildDatabaseBlocks(pageId, containerTitle))[0].id;
67// console.log("databaseId:", databaseId);
6869// actions
70await notionHelpers.deleteAllDatabasePagesParallel(databaseId);
71// add back pages; fresh start for the next person to add favicons
72// create new database pages from default object
73await notionHelpers.createDatabasePagesParallel(databaseId, defaults);
74// store webhook data in blob storage for resets
75await blob.setJSON(blobKey, {
78page_id: pageId,
79},
80id: databaseId,
81date: new Date(),
82content: askingFor,
ValTownForNotionwebhookAPI4 matches
7import exampleCallout from "./routes/example-callout.ts";
8import exampleChildPages from "./routes/example-child-pages.ts";
9import exampleDatabasePage from "./routes/example-database-page.ts";
10import exampleDatabasePages from "./routes/example-database-pages.ts";
11import exampleGuestWelcome from "./routes/example-guest-welcome.ts";
12import propertyInjections from "./routes/property-injections.ts";
2425// Mount route modules
26app.route("/example/database/pages", exampleDatabasePages);
27app.route("/example/database/page", exampleDatabasePage);
28app.route("/example/child_pages", exampleChildPages);
29app.route("/example/callout", exampleCallout);
ValTownForNotionREADME.md2 matches
67- `auth.ts` - Authentication middleware
8- `example-database-pages.ts` - Handles `/example/database/pages` endpoint
9- `example-database-page.ts` - Handles `/example/database/page` endpoint
10- `example-child-pages.ts` - Handles `/example/child_pages` endpoint
11- `example-callout.ts` - Handles `/example/callout` endpoint
stevensDemosetupTelegramChatDb.ts2 matches
1// Script to set up the telegram_chats table in SQLite
2// Run this script manually to create the database table
34export default async function setupTelegramChatDb() {
25`);
2627return "Telegram chat database table created successfully.";
28} catch (error) {
29console.error("Error setting up telegram_chats table:", error);
stevensDemoREADME.md3 matches
13## Technical Architecture
1415**⚠️ important caveat: the admin dashboard doesn't have auth! currently it just relies on security by obscurity of people not knowing the url to a private val. this is not very secure. if you fork this project and put sensitive data in a database you should think carefully about how to secure it.**
1617Stevens has been designed with the utmost simplicity and extensibility, much like a well-organized household. At the heart of his operation lies a single "memories" table - a digital equivalent of a butler's meticulous records. This table serves as the foundation for all of Stevens' operations.
45- `dashboard`: the admin view for showing the memories notebook + visualizing imports
46- `dailyBriefing`: stuff related to sending a daily update via telegram
47- `dbUtils`: little one-off scripts for database stuff
4849## Hiring your own Stevens
57- For the Google Calendar integration you'll need `GOOGLE_CALENDAR_ACCOUNT_ID` and `GOOGLE_CALENDAR_CALENDAR_ID`. See [these instuctions](https://www.val.town/v/stevekrouse/pipedream) for details.
5859**important caveat: the admin dashboard doesn't have auth! currently it just relies on security by obscurity of people not knowing the url to a private val. this is not very secure, if you put sensitive data in a database you should think carefully about how to secure it.**
6061Overall it's a simple enough project that I encourage you to just copy the ideas and run in your own direction rather than try to use it as-is.
stevensDemoREADME.md2 matches
45* `index.ts` - this is the **entrypoint** for this whole project
6* `database/` - this contains the code for interfacing with the app's SQLite database table
78## Hono
26## CRUD API Routes
2728This app has two CRUD API routes: for reading and inserting into the messages table. They both speak JSON, which is standard. They import their functions from `/backend/database/queries.ts`. These routes are called from the React app to refresh and update data.
2930## Errors
stevensDemoREADME.md6 matches
1# Database
23This app uses [Val Town SQLite](https://docs.val.town/std/sqlite/) to manage data. Every Val Town account comes with a free SQLite database, hosted on [Turso](https://turso.tech/). This folder is broken up into two files:
45* `migrations.ts` - code to set up the database tables the app needs
6* `queries.ts` - functions to run queries against those tables, which are imported and used in the main Hono server in `/backend/index.ts`
78## Migrations
910In `backend/database/migrations.ts`, this app creates a new SQLite table `reactHonoStarter_messages` to store messages.
1112This "migration" runs once on every app startup because it's imported in `index.ts`. You can comment this line out for a slight (30ms) performance improvement on cold starts. It's left in so that users who fork this project will have the migration run correctly.
1314SQLite has much more limited support for altering existing tables as compared to other databases. Often it's easier to create new tables with the schema you want, and then copy the data over. Happily LLMs are quite good at those sort of database operations, but please reach out in the [Val Town Discord](https://discord.com/invite/dHv45uN5RY) if you need help.
1516## Queries
1718The queries file is where running the migrations happen in this app. It'd also be reasonable for that to happen in index.ts, or as is said above, for that line to be commented out, and only run when actual changes are made to your database schema.
1920The queries file exports functions to get and write data. It relies on shared types and data imported from the `/shared` directory.
stevensDemopopulateDemo.ts4 matches
374];
375376// Insert memories into the database
377async function insertDemoMemories() {
378try {
415await insertDemoMemories();
416417console.log("Demo database successfully populated!");
418return "Demo database successfully populated!";
419} catch (error) {
420console.error("Error populating demo database:", error);
421throw error;
422}