This project is a proof of concept that integrates OpenAI’s API with a user’s personal notes to narrow the context of any query to the information stored within said notes. The goal is to create a focused, context-aware assistant that can help users by answering questions based solely on their stored data.
The project has a variety of potential use cases, such as helping students study, providing an internal chatbot for company documentation, or supporting memory-impaired individuals with life events and important details. Its broad target audience is anyone who needs context-specific information from their own notes, with the AI being limited to only the content the user has saved.
The inspiration behind the project came from observing multiple clients struggle with onboarding new employees and maintaining up-to-date internal documentation. This solution addresses both issues by offering a streamlined way to access and interact with critical information.
One of the initial challenges I faced was related to using Next.js’s server actions in conjunction
with client-side hooks like useActionState
and trying to invalidate routes after a successful
mutation. The default behavior of route invalidation would trigger a full route refresh, which
didn’t give me the fine-grained control I needed. To solve this, I switched to React Query for
better caching and more flexible invalidation controls, which greatly improved the experience.
Another challenge came when working with Supabase. Their documentation had conflicting instructions about separating the server and browser clients, which led to some confusion. Fortunately, Supabase’s latest connection utility, available in the table management UI, helped clarify this and provided the solution.
A final challenge was rendering and styling a markdown preview of a user’s notes. I initially struggled to find a clean, efficient solution, but using the ReactMarkdown package along with Tailwind CSS allowed me to easily style the markdown and ensure a smooth user experience. The render props provided by ReactMarkdown made customizing the display straightforward.
The simplicity of the project structure plays a key role in its scalability. The application is organized in a way that easily accommodates future growth:
While I initially experimented with Prisma ORM to abstract away platform-specific details and make the app more agnostic, I ran into issues with Prisma’s migrations. The migrations would occasionally wipe unspecified tables and produce errors, likely due to some misconfiguration on my part. Ultimately, the added complexity wasn’t worth the benefits, so I decided to stick with Supabase’s native tools.
Regarding performance, the primary bottleneck arises from Supabase’s authentication and database speeds, which can sometimes result in noticeable lag. However, overall, the architecture is designed to handle growth without significant performance concerns, given the project’s simplicity and modularity.
There’s a lot of room for optimization and fine-tuning moving forward. One potential improvement is running the AI model locally, which could help mitigate the cost of providing large contexts for each query. This would give users more flexibility without being constrained by token limitations. Alternatively, instead of providing raw notes, it might make sense to fine-tune a model with the user’s notes for better performance and more accurate responses.
Next steps include introducing an organizational structure, allowing users to separate their personal notes from shared organization-related notes. This will also involve adding a user management system to support multiple users within an organization, along with role-based access controls. These controls would ensure that users with lower roles can’t edit or delete critical resources, maintaining data integrity.