- Node/Express/TypeScript API under /api/v1 with JWT auth (login, refresh, logout, /me) - Prisma schema: vendors, users, roles, products, categories, taxes, transactions - SQLite for local dev; Postgres via docker-compose for production - Full CRUD routes for vendors, users, categories, taxes, products with Zod validation and RBAC - Paginated list endpoints scoped per vendor; refresh token rotation - React/TypeScript admin SPA (Vite): login, protected routing, sidebar layout - Pages: Dashboard, Catalog (tabbed Products/Categories/Taxes), Users, Vendor Settings - Shared UI: Table, Modal, FormField, Btn, PageHeader components - Multi-stage Dockerfile; docker-compose with Postgres healthcheck - Seed script with demo vendor and owner account - INSTRUCTIONS.md, ROADMAP.md, .claude/launch.json for dev server config Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
48 lines
885 B
JavaScript
48 lines
885 B
JavaScript
/*!
|
|
* express
|
|
* Copyright(c) 2009-2013 TJ Holowaychuk
|
|
* Copyright(c) 2013 Roman Shtylman
|
|
* Copyright(c) 2014-2015 Douglas Christopher Wilson
|
|
* MIT Licensed
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
/**
|
|
* Module dependencies.
|
|
*/
|
|
|
|
var merge = require('utils-merge')
|
|
var parseUrl = require('parseurl');
|
|
var qs = require('qs');
|
|
|
|
/**
|
|
* @param {Object} options
|
|
* @return {Function}
|
|
* @api public
|
|
*/
|
|
|
|
module.exports = function query(options) {
|
|
var opts = merge({}, options)
|
|
var queryparse = qs.parse;
|
|
|
|
if (typeof options === 'function') {
|
|
queryparse = options;
|
|
opts = undefined;
|
|
}
|
|
|
|
if (opts !== undefined && opts.allowPrototypes === undefined) {
|
|
// back-compat for qs module
|
|
opts.allowPrototypes = true;
|
|
}
|
|
|
|
return function query(req, res, next){
|
|
if (!req.query) {
|
|
var val = parseUrl(req).query;
|
|
req.query = queryparse(val, opts);
|
|
}
|
|
|
|
next();
|
|
};
|
|
};
|