LLM rules for Cursor and Windsurf | Next.js Supabase
Use these rules for Cursor and Windsurf to generate the best possible context for your LLMs
The following content is a guideline for Cursor and Codeium Windsurf to help you generate the best possible context for your LLMs. It should also work well for other coding tools, such as Windsurf.
- Cursor: create a file
.cursorrules
file in your project root and paste the rules below. - Windsurf: create a file
.windsurfrules
file in your project root and paste the rules at the bottom of this page.
Please Remember: These tools are brilliant, but not perfect. Always double check the results and make sure they are relevant and useful.
Cursor Rules for Makerkit
Cursor doesn't have a specified limit, so we recommend using the following content:
# Makerkit GuidelinesYou are an expert programming assistant focusing on:- Expertise: React, Supabase, TypeScript, Next.js 15, Shadcn UI, Tailwind CSS in a Turborepo project- Focus: Code clarity, Readability, Best practices, Maintainability- Style: Expert level, factual, solution-focused- Libraries: TypeScript, React Hook Form, React Query, Zod, Lucide React## Application Scope<INSERT_HERE_YOUR_APPLICATION_SCOPE_HERE>## Project StructureThe below is the Makerkit's Next.js App Router structure.```- apps-- web--- app---- home # protected routes------ (user) # user workspace------ [account] # team workspace---- (marketing) # marketing pages---- auth # auth pages--- components # global components--- config # global config--- lib # global utils--- content # markdoc content--- styles--- supabase # supabase root```## Configuration- When adding paths under the user workspace, update the file `apps/web/config/personal-account-navigation.config.tsx`- When adding paths under the team workspace, update the file `apps/web/config/team-account-navigation.config.tsx`## Database- Supabase uses Postgres- We strive to create a safe, robust, performant schema- Accounts are the general concept of a user account, defined by the having the same ID as Supabase Auth's users (personal). They can be a team account or a personal account.- Generally speaking, other tables will be used to store data related to the account. For example, a table `notes` would have a foreign key `account_id` to link it to an account.- Using RLS, we must ensure that only the account owner can access the data. Always write safe RLS policies and ensure that the policies are enforced.- Unless specified, always enable RLS when creating a table. Propose the required RLS policies ensuring the safety of the data.- Always consider any required constraints and triggers are in place for data consistency- Always consider the compromises you need to make and explain them so I can make an educated decision. Follow up with the considerations make and explain them.- Always consider the security of the data and explain the security implications of the data.- Always use Postgres schemas explicitly (e.g., `public.accounts`)- Data types should always be inferred using the `Database` types from `@kit/supabase/database````tsximport type { Tables } from '@kit/supabase/database';type Bookmark = Tables<'bookmarks'>;```## Personal Account ContextThe user/personal account context in the application lives under the path `app/home/(user)`. Under this context, we identify the user using Supabase Auth.We can use the `requireUserInServerComponent` to retrieve the relative Supabase User object and identify the user.### Client ComponentsIn a Client Component, we can access the `UserWorkspaceContext` and use the `user` object to identify the user.We can use it like this:```tsximport { useUserWorkspace } from '@kit/accounts/hooks/use-user-workspace';```## Team Account ContextThe team account context in the application lives under the path `app/home/[account]`. The `[account]` segment is the slug of the team account, from which we can identify the team.### Accessing the Account Workspace Data in Client ComponentsThe data fetched from the account workspace API is available in the team context. You can access this data using the useAccountWorkspace hook.```tsx'use client';import { useTeamAccountWorkspace } from '@kit/team-accounts/hooks/use-team-account-workspace';export default function SomeComponent() { const { account, user, accounts } = useTeamAccountWorkspace(); // use account, user, and accounts}```The useTeamAccountWorkspace hook returns the same data structure as the loadTeamWorkspace function.NB: the hooks is not to be used is Server Components, only in Client Components. Additionally, this is only available in the pages under /home/[account] layout.### Team PagesThese pages are dedicated to the team account, which means they are only accessible to team members. To access these pages, the user must be authenticated and belong to the team.## UI ComponentsReusable UI components are defined in the "packages/ui" package named "@kit/ui".By exporting the component from the "exports" field, we can import it using the "@kit/ui/{component-name}" format.### Code Standards- Files - Always use kebab-case- Naming - Functions/Vars: camelCase - Constants: UPPER_SNAKE_CASE - Types/Classes: PascalCase- TypeScript - Prefer types over interfaces - Use type inference whenever possible - Avoid any, any[], unknown, or any other generic type - Use spaces between code blocks to improve readability### Styling- Styling is done using Tailwind CSS. We use the "cn" function from the "@kit/ui/utils" package to generate class names.- Avoid fixes classes such as "bg-gray-500". Instead, use Shadcn classes such as "bg-background", "text-secondary-foreground", "text-muted-foreground", etc.### Data Fetching- In a Server Component context, please use the Supabase Client directly for data fetching- In a Client Component context, please use the `useQuery` hook from the "@tanstack/react-query" packageData Flow works in the following way:1. Server Component uses the Supabase Client to fetch data.2. Data is rendered in Server Components or passed down to Client Components when absolutely necessary to use a client component (e.g. when using React Hooks or any interaction with the DOM).```tsximport { getSupabaseServerClient } from '@kit/supabase/server-client';async function ServerComponent() { const client = getSupabaseServerClient(); const { data, error } = await client.from('notes').select('*'); // use data}```or pass down the data to a Client Component:```tsximport { getSupabaseServerClient } from '@kit/supabase/server-client';export default function ServerComponent() { const supabase = getSupabaseServerClient(); const { data, error } = await supabase.from('notes').select('*'); if (error) { return <SomeErrorComponent error={error} />; } return <SomeClientComponent data={data} />;}```#### Supabase Clients- In a Server Component context, use the `getSupabaseServerClient` function from the "@kit/supabase/server-client" package.- In a Client Component context, use the `useSupabase` hook from the "@kit/supabase/hooks/use-supabase" package.##### Admin ActionsOnly in rare cases suggest using the Admin client `getSupabaseServerAdminClient` when needing to bypass RLS from the package `@kit/supabase/server-admin-client`.#### React QueryWhen using `useQuery`, make sure to define the data fetching hook. Create two components: one that fetches the data and one that displays the data.## Server Actions- For Data Mutations from Client Components, always use Server Actions- Always name the server actions file as "server-actions.ts"- Always name exported Server Actions suffixed as "Action", ex. "createPostAction"- Always use the `enhanceAction` function from the "@kit/supabase/actions" package.```tsx'use server';import { z } from 'zod';import { enhanceAction } from '@kit/next/actions';const ZodSchema = z.object({ email: z.string().email(), password: z.string().min(6),});export const myServerAction = enhanceAction( async function (data, user) { // 1. "data" is already a valid ZodSchema and it's safe to use // 2. "user" is the authenticated user // ... your code here return { success: true, }; }, { auth: true, schema: ZodSchema, },);```## Route Handler / API Routes- Use Route Handlers when data fetching from Client Components- To create API routes (route.ts), always use the `enhanceRouteHandler` function from the "@kit/supabase/routes" package.```tsximport { NextResponse } from 'next/server';import { z } from 'zod';import { enhanceRouteHandler } from '@kit/next/routes';const ZodSchema = z.object({ email: z.string().email(), password: z.string().min(6),});export const POST = enhanceRouteHandler( async function ({ body, user, request }) { // 1. "body" is already a valid ZodSchema and it's safe to use // 2. "user" is the authenticated user // 3. "request" is NextRequest // ... your code here return NextResponse.json({ success: true, }); }, { schema: ZodSchema, },);// example of unauthenticated route (careful!)export const GET = enhanceRouteHandler( async function ({ user, request }) { // 1. "user" is null, as "auth" is false and we don't require authentication // 2. "request" is NextRequest // ... your code here return NextResponse.json({ success: true, }); }, { auth: false, },);```Consider logging asynchronous requests using the `@kit/shared/logger` package in a structured way to provide context to the logs in both server actions and route handlers.```tsxconst ctx = { name: 'my-server-action', // use a meaningful name userId: user.id, // use the authenticated user's ID};logger.info(ctx, 'Request started...');const { data, error } = await supabase.from('notes').select('*');if (error) { logger.error(ctx, 'Request failed...'); // handle error} else { logger.info(ctx, 'Request succeeded...'); // use data}```### Related APIsWhen required, use the following APIs.1. Personal Account APIs:```tsximport { createAccountsApi } from '@kit/accounts/api';import { getSupabaseServerClient } from '@kit/supabase/server-client';async function ServerComponent() { const client = getSupabaseServerClient(); const api = createAccountsApi(client); // use api}```2. Team Account APIs:```tsximport { getSupabaseServerClient } from '@kit/supabase/server-client';import { createTeamAccountsApi } from '@kit/team-accounts/api';async function ServerComponent() { const client = getSupabaseServerClient(); const api = createTeamAccountsApi(client); // use api}```3. Auth API:```tsximport { redirect } from 'next/navigation';import { requireUser } from '@kit/supabase/require-user';import { getSupabaseServerClient } from '@kit/supabase/server-client';async function ServerComponent() { const client = getSupabaseServerClient(); const auth = await requireUser(client); // check if the user needs redirect if (auth.error) { redirect(auth.redirectTo); } // user is authed! const user = auth.data;}```4. Billing API:```tsximport { createBillingGatewayService } from '@kit/billing-gateway';const service = createBillingGatewayService('stripe');```## Creating PagesWhen creating new pages ensure:1. The page is exported using `withI18n` to enable i18n.2. The page has the required and correct metadata using the `metadata` or `generateMetadata` function.3. Don't worry about authentication, it will be handled automatically.```tsximport { Metadata } from 'next';import { withI18n } from '~/lib/i18n/with-i18n';export const metadata: Metadata = { title: 'A page title', description: 'A page description',};function Page() { // ...}export default withI18n(Page);```## Forms- Use React Hook Form for form validation and submission.- Use Zod for form validation.- Use the `zodResolver` function to resolve the Zod schema to the form.Follow the example below to create all forms:### Define the schemaZod schemas should be defined in the `schema` folder and exported, so we can reuse them across a Server Action and the client-side form:```tsx// _lib/schema/create-note.schema.tsimport { z } from 'zod';export const CreateNoteSchema = z.object({ title: z.string().min(1), content: z.string().min(1),});```### Create the Server Action```tsx// _lib/server/server-actions.ts'use server';import { z } from 'zod';import { enhanceAction } from '@kit/next/actions';import { CreateNoteSchema } from '../schema/create-note.schema';const CreateNoteSchema = z.object({ title: z.string().min(1), content: z.string().min(1),});export const createNoteAction = enhanceAction( async function (data, user) { // 1. "data" has been validated against the Zod schema, and it's safe to use // 2. "user" is the authenticated user // ... your code here return { success: true, }; }, { auth: true, schema: CreateNoteSchema, },);```Then create a client component to handle the form submission:```tsx// _components/create-note-form.tsx'use client';import { zodResolver } from '@hookform/resolvers/zod';import { useForm } from 'react-hook-form';import { z } from 'zod';import { Form, FormControl, FormField, FormItem, FormLabel, FormMessage,} from '@kit/ui/form';import { CreateNoteSchema } from '../_lib/schema/create-note.schema';export function CreateNoteForm() { const [pending, startTransition] = useTransition(); const form = useForm({ resolver: zodResolver(CreateNoteSchema), defaultValues: { title: '', content: '', }, }); const onSubmit = (data) => { startTransition(async () => { try { await createNoteAction(data); } catch { // handle error } }); }; return ( <form onSubmit={form.handleSubmit(onSubmit)}> <Form {...form}> <FormField name={'title'} render={({ field }) => ( <FormItem> <FormLabel> <span className={'text-sm font-medium'}>Title</span> </FormLabel> <FormControl> <input type={'text'} className={'w-full'} placeholder={'Title'} {...field} /> </FormControl> <FormMessage /> </FormItem> )} /> <FormField name={'content'} render={({ field }) => ( <FormItem> <FormLabel> <span className={'text-sm font-medium'}>Content</span> </FormLabel> <FormControl> <textarea className={'w-full'} placeholder={'Content'} {...field} /> </FormControl> <FormMessage /> </FormItem> )} /> <button disabled={pending} type={'submit'} className={'w-full'}> Submit </button> </Form> </form> );}```Always use `@kit/ui` for writing the UI of the form.## Error Handling- Logging using the `@kit/shared/logger` package- Don't swallow errors, always handle them appropriately- Handle promises and async/await gracefully- Consider the unhappy path and handle errors appropriately- Context without sensitive data## Translations- Use the `Trans` component from the `@kit/ui/trans` package to translate strings.- Translations are placed in the `apps/web/public/locales/{language}/{namespace}.json` path.- To add a new language, update the `languages` array in the `apps/web/lib/i18n/i18n.settings.ts` file.- Consider adding a new namespace for a new functionality/page.
Windsurf Rules for Makerkit
Windsurf has a limit of 6000 characters per file, so we recommend using the following content:
# Makerkit Guidelines## Project Stack- Framework: Next.js 15 App Router, TypeScript, React, Node.js- Backend: Supabase with Postgres- UI: Shadcn UI, Tailwind CSS- Key libraries: React Hook Form, React Query, Zod, Lucide React- Focus: Code clarity, Readability, Best practices, Maintainability## Project Structure```/apps/web/ /app /home # protected routes /(user) # user workspace /[account] # team workspace /(marketing) # marketing pages /auth # auth pages /components # global components /config # global config /lib # global utils /content # markdoc content /supabase # supabase root```## Core Principles### Data Flow1. Server Components - Use Supabase Client directly via `getSupabaseServerClient` - Handle errors with proper boundaries - Example: ```tsx async function ServerComponent() { const client = getSupabaseServerClient(); const { data, error } = await client.from('notes').select('*'); if (error) return <ErrorComponent error={error} />; return <ClientComponent data={data} />; } ```2. Client Components - Use React Query for data fetching - Implement proper loading states - Example: ```tsx function useNotes() { const { data, isLoading } = useQuery({ queryKey: ['notes'], queryFn: async () => { const { data } = await fetch('/api/notes'); return data; } }); return { data, isLoading }; } ```### Server Actions- Name files as "server-actions.ts" in `_lib/server` folders- Export with "Action" suffix- Use `enhanceAction`. Use a single parameter for data- Use `revalidatePath` to refresh the data after a submission- Example:```tsx'use server';import { enhanceAction } from '@kit/next/actions';import { getSupabaseServerClient } from '@kit/supabase/server-client';export const createNoteAction = enhanceAction( async function (data, user) { const client = getSupabaseServerClient(); const { error } = await client .from('notes') .insert({ ...data, user_id: user.id }); if (error) throw error; return { success: true }; }, { auth: true, schema: NoteSchema, });```From client components:```tsxawait createNoteAction(data);```### Route Handlers- Use `enhanceRouteHandler` to wrap route handlers- Use Route Handlers when data fetching from Client Components```tsximport { enhanceRouteHandler } from '@kit/next/routes';import { getSupabaseServerClient } from '@kit/supabase/server-client';export const POST = enhanceRouteHandler( async ({ request, data, user }) => { const client = getSupabaseServerClient(); // .... }, { auth: true, schema: NoteSchema, });```## Database & Security### RLS Policies- Strive to create a safe, robust, secure and consistent database schema- Always consider the compromises you need to make and explain them so I can make an educated decision. Follow up with the considerations make and explain them.- Enable RLS by default and propose the required RLS policies- `public.accounts` are the root tables for the application- Implement cascading deletes when appropriate- Ensure strong consistency considering triggers and constraints- Always use Postgres schemas explicitly (e.g., `public.accounts`)- Data types should always be inferred using the `Database` types from `@kit/supabase/database````tsximport type { Tables } from '@kit/supabase/database';type Bookmark = Tables<'bookmarks'>;```## Forms Pattern### 1. Schema Definition```tsx// schema/note.schema.tsimport { z } from 'zod';export const NoteSchema = z.object({ title: z.string().min(1).max(100), content: z.string().min(1), category: z.enum(['work', 'personal']),});```### 2. Form Component```tsx'use client';export function NoteForm() { const [pending, startTransition] = useTransition(); const form = useForm({ resolver: zodResolver(NoteSchema), defaultValues: { title: '', content: '', category: 'personal' } }); const onSubmit = (data: z.infer<typeof NoteSchema>) => { startTransition(async () => { try { await createNoteAction(data); form.reset(); } catch (error) { // Handle error } }); }; return ( <Form {...form}> <FormField name="title" render={({ field }) => ( <FormItem> <FormLabel>Title</FormLabel> <FormControl> <Input {...field} /> </FormControl> <FormMessage /> </FormItem> )} /> {/* Other fields */} </Form> );}```## Error Handling- Consider logging asynchronous requests in server code using the `@kit/shared/logger`- Handle promises and async/await gracefully- Consider the unhappy path and handle errors appropriately### Structured Logging```tsxconst ctx = { name: 'create-note', userId: user.id, noteId: note.id};logger.info(ctx, 'Creating new note...');try { await createNote(); logger.info(ctx, 'Note created successfully');} catch (error) { logger.error(ctx, 'Failed to create note', { error }); throw error;}```## Context ManagementIn client components, we can use the `useUserWorkspace` hook to access the user's workspace data.### Personal Account```tsx'use client';function PersonalDashboard() { const { workspace, user } = useUserWorkspace(); if (!workspace) return null; return ( <div> <h1>Welcome, {user.email}</h1> <SubscriptionStatus status={workspace.subscription_status} /> </div> );}```### Team AccountIn client components, we can use the `useTeamAccountWorkspace` hook to access the team account's workspace data. It only works under the `/home/[account]` route.```tsx'use client';function TeamDashboard() { const { account, user } = useTeamAccountWorkspace(); return ( <div> <h1>{account.name}</h1> <RoleDisplay role={account.role} /> <PermissionsList permissions={account.permissions} /> </div> );}```## UI Components- Reusable UI components are defined in the "packages/ui" package named "@kit/ui".- By exporting the component from the "exports" field, we can import it using the "@kit/ui/{component-name}" format.## Creating PagesWhen creating new pages ensure:- The page is exported using `withI18n(Page)` to enable i18n.- The page has the required and correct metadata using the `metadata` or `generateMetadata` function.- Don't worry about authentication, it's handled in the middleware.
You can add more rules to Windsurf using the global rules, which are a set of rules that are applied to all projects:
I use Typescript, React, Next.js App Router, Zod and Shadcn UI.## Typescript- My Typescript is always typed, but never returns explicit types.- My Typescript is strict.- I write small, self-contained, pure functions. I avoid mutations.- My code is simple, self-explanatory, and readable.- I write spaces between code blocks to make it easier to read.- Prefer types over interfaces- Use type inference whenever possible- Avoid any, any[], unknown, or any other generic type- Use spaces between code blocks to improve readability## React- I write small and reusable React components. I only use `useState` when required. If I do, I make sure to use a few as possible and group state together to avoid unnecessary re-renders.- I write small and reusable React components. I only use `useEffect` when required, which means rarely.- Components are written with clear intent and purpose, use props to pass data from parent to child and are reusable across the app.
Additional context for LLMs
You can generate a set of Markdown files using our little script and the open source Makerkit documentation.
You can then use the Markdown files as a knowledge base for your LLMs, such as Cursor, ChatGPT or Claude.
Clone the repository, then run the following command:
node index.js kits/next-supabase-turbo
To ensure a more precise context, always scope the generation for the kit you are interested in. For example, to generate the markdown files for the Remix Supabase Kit, run the following command:
node index.js kits/remix-supabase-turbo
You can also do it for the courses by running the following command:
node index.js courses
This will generate markdown files in the dist folder.
You can configure how many words per file by passing a second argument to the script, e.g.:
node index.js kits 4000
By default, it will generate 5000 words per file.