nuxt-auto-crud (nac 2.x)
A Nuxt.js module providing dynamic RESTful CRUD APIs derived directly from your Drizzle schemas, without writing any code for CRUD operations.
Core Features
- Zero-Codegen Dynamic RESTful CRUD APIs: nuxt-auto-crud leverages Drizzle ORM, Zod, Nuxt, and Nitro to eliminate the need for manual CRUD coding.
- Single Source of Truth (SSOT): Your Drizzle schemas (
server/db/schema) define the entire API structure and validation. - Constant Bundle Size: Since no code is generated, the bundle size remains virtually identical whether you have one table or one hundred (scaling only with your schema definitions).
Supported Databases
- SQLite (libSQL)
- MySQL
Installation Guide (SQLite)
Option A: Starter Template
npx nuxi init -t gh:clifordpereira/nac-starter my-app
cd my-app
nuxt db generate
nuxt dev
Option B: Manual Installation
bun create nuxt@latest my-app
cd my-app
npx nuxi module add hub
bun add drizzle-orm@beta @libsql/client nuxt-auto-crud
bun add -D drizzle-kit@beta typescript
Configuration
Update nuxt.config.ts:
export default defineNuxtConfig({
modules: [
'@nuxthub/core',
'nuxt-auto-crud'
],
hub: {
db: 'sqlite'
}
})
Schema Definition
Define your schema in server/db/schema.ts:
import { sqliteTable, text, integer, numeric } from 'drizzle-orm/sqlite-core'
export const products = sqliteTable('products', {
id: integer('id').primaryKey({ autoIncrement: true }),
name: text('name').notNull(),
sku: text('sku').notNull(),
price: numeric('price', { mode: 'number' }).notNull(),
stock: integer('stock').notNull(),
createdAt: integer({ mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
updatedAt: integer({ mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
})
Generate Migrations and Start Dev Server
nuxt db generate
nuxt dev
For MySQL installation instructions, visit INSTALLATION.md.
Critical Stability Update (v2.5+)
The "String-instead-of-Number" Bug is Fixed! Previous versions had an issue inheriting Drizzle-Zod's underlying behavior where
numericandintegercolumns erroneously parsed asstringtypes in the UI metadata. This caused type-mismatches with frontend form components and required manual parsing.Fixed in recent versions: The module now fully infers, converts, and enforces native database numeric types strictly as
numberpayloads. No more manual casting or frontend form component type hacks are required.
Data APIs (Dynamic RESTful CRUD)
Note: All endpoints follow the pattern ${nacEndpointPrefix}/:model. By default, this is /api/_nac/:model.
| Method | Endpoint | Action |
|---|---|---|
| GET | /api/_nac/:model |
List records |
| POST | /api/_nac/:model |
Create record with Zod validation |
| GET | /api/_nac/:model/:id |
Fetch single record |
| PATCH | /api/_nac/:model/:id |
Partial update with validation |
| DELETE | /api/_nac/:model/:id |
Delete record |
Example (products table):
| Action | HTTP Method | Endpoint | Example Result |
|---|---|---|---|
| Fetch All | GET |
/api/_nac/products |
List of all products |
| Create | POST |
/api/_nac/products |
New product record added |
| Fetch One | GET |
/api/_nac/products/1 |
Details of product with id: 1 |
| Update | PATCH |
/api/_nac/products/1 |
Partial update to product 1 |
| Delete | DELETE |
/api/_nac/products/1 |
Product 1 removed from DB |
Introspection & Metadata APIs
Use these endpoints to build dynamic UI components (like menus and forms) or provide context to AI agents. These use the _schemas and _meta reserved paths.
1. Discovery Endpoints
- List Resource Names:
GET /api/_nac/_schemas - Returns an array of all available table names. Useful for generating dynamic navigation menus.
- Resource Metadata:
GET /api/_nac/_schemas/:resource - Returns field definitions, validation rules, and
readonlystatus for a specific table. - Example:
GET /api/_nac/_schemas/productsreturns the schema for the products table.
Schema Interface
export interface Field {
name: string
type: string
required?: boolean
selectOptions?: string[]
references?: string
readonly?: boolean
}
export interface SchemaDefinition {
resource: string
labelField: string
fields: Field[]
}
Example Response
GET /api/_nac/_schemas/products
{
"resource": "products",
"labelField": "name",
"fields": [
{ "name": "id", "type": "number", "required": true, "readonly": true },
{ "name": "name", "type": "string", "required": true, "readonly": false },
{ "name": "sku", "type": "string", "required": true, "readonly": false },
{ "name": "price", "type": "number", "required": true, "readonly": false },
{ "name": "stock", "type": "number", "required": true, "readonly": false }
]
}
2. Agentic Discovery
- Manifest:
GET /api/_nac/_meta?format=md - Returns a token-efficient Markdown manifest for LLM context injection.
- Security: Requires
NUXT_AUTO_CRUD_AGENTIC_TOKEN(min 16 characters) in your.env.
Security & Configuration
Enabling authentication in the autoCrud config protects all nac routes (/api/_nac/*), except those explicitly defined in publicResources.
Access Control & Data Safety
apiHiddenFields: Globally hides sensitive columns from all API responses. Default: ['password', 'secret', 'token', 'resetToken', 'resetExpires', 'githubId', 'googleId'].formHiddenFields: Columns excluded from the frontend schema metadata to prevent user input. Defaults to apiHiddenFields plus system-managed fields likeid,uuid,createdAt,updatedAt,deletedAt,createdBy, andupdatedBy.formReadOnlyFields: Columns visible in the UI for context but protected from user modification (e.g., slug, status).- Response Scrubbing: If a field is in
apiHiddenFieldsor does not exist in the schema, it is silently stripped from the response even if listed inpublicResources.
Configuration Reference
| Key | Default | Description |
|---|---|---|
statusFiltering |
false |
Enables/disables automatic filtering of records based on the status column. |
realtime |
false |
Enables real-time broadcasting of all Create, Update, and Delete (CUD) operations via SSE. |
auth.authentication |
false |
Requires a valid session for all NAC routes. |
auth.authorization |
false |
Enables role/owner-based access checks. |
auth.ownerKey |
'createdBy' |
The column name used to identify the record creator. |
publicResources |
{} |
Defines tables and specific columns accessible without auth. |
apiHiddenFields |
NAC_API_HIDDEN_FIELDS |
Arrays of keys to exclude from all API responses. |
formHiddenFields |
NAC_FORM_HIDDEN_FIELDS |
Arrays of keys to exclude from dynamic forms. |
formReadOnlyFields |
NAC_FORM_READ_ONLY_FIELDS |
List of visible but non-editable fields (UI only). |
agenticToken |
'' |
Secret key used to secure the /_meta endpoint, preventing unauthorized AI agents from introspecting your schema. |
nacEndpointPrefix |
'/api/_nac' |
The base path for NAC routes. Access via useRuntimeConfig().public.autoCrud. |
schemaPath |
'server/db/schema' |
Location of your Drizzle schema files. |
Example nuxt.config.ts
autoCrud: {
statusFiltering: false,
realtime: false,
auth: {
authentication: false,
authorization: false,
ownerKey: 'createdBy',
},
publicResources: {
products: ['id', 'name', 'sku', 'price'],
},
apiHiddenFields: ['cost_price'],
formHiddenFields: ['created_at', 'updated_at'],
formReadOnlyFields: ['sku'], // Locked for user input after generation
agenticToken: '',
nacEndpointPrefix: '/api/_nac',
schemaPath: 'server/db/schema',
}
Note: Modify
nacEndpointPrefixorschemaPathonly if the Nuxt/Nitro conventions change.
Filtering & Performance Optimization
Automatic Status Filtering
If statusFiltering is enabled, nac applies global visibility constraints. When a status column exists, queries are automatically restricted to active records. This logic integrates with the authorization layer, allowing users to see their own records (regardless of status) if they possess the list_active permission.
Ownership & Permissions
While the implementing app handles the authentication & authorization layer, nac provides a standardized way to enforce record ownership and granular access.
If your middleware populates event.context.nac with resourcePermissions, nac automatically injects the necessary SQL filters.
Example: Restricting users to their own records
If the permissions array includes 'list_own', nac appends a filter where ownerKey (defaulting to createdBy) matches the userId.
If list_active is present, it applies a hybrid OR logic: users can see all active records OR any record they own, regardless of its status.
// Example: Setting context in your Auth Middleware
event.context.nac = {
userId: user.id,
resourcePermissions: user.permissions[model], // e.g., ['list_own', 'list_active']
record: null, // Optional: Pre-fetched record to prevent double-hitting the DB
}
Optimization: Skip Redundant Fetches
If your middleware has already fetched the record, pass it to event.context.nac.record (as shown above). nac will use this object instead of executing an additional database query.
Real-time Synchronization (SSE)
When realtime is enabled, all create, update, and delete operations are automatically broadcasted:
if (realtime) {
void broadcast({
table: model,
action: 'create',
primaryKey: newRecord.id,
data: newRecord,
})
}
Frontend Usage
NAC provides a useNacAutoCrudSSE composable to listen for these changes in your frontend:
useNacAutoCrudSSE(({ table, action, data: sseData, primaryKey }) => {
// Optional: Filter by specific table
if (table !== 'products') return
if (action === 'update') {
// updateRow(primaryKey, sseData)
}
if (action === 'create') {
// addRow(sseData)
}
if (action === 'delete') {
// removeRow(primaryKey)
}
})