ndomo 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (247) hide show
  1. package/.bun-version +1 -0
  2. package/.dockerignore +79 -0
  3. package/.editorconfig +18 -0
  4. package/.env.example +19 -0
  5. package/.github/CODEOWNERS +8 -0
  6. package/.github/ISSUE_TEMPLATE/bug_report.yml +62 -0
  7. package/.github/ISSUE_TEMPLATE/config.yml +2 -0
  8. package/.github/ISSUE_TEMPLATE/feature_request.yml +34 -0
  9. package/.github/dependabot.yml +36 -0
  10. package/.github/pull_request_template.md +24 -0
  11. package/.github/release.yml +30 -0
  12. package/.github/workflows/gitleaks.yml +28 -0
  13. package/.github/workflows/release-please.yml +27 -0
  14. package/.github/workflows/smoke.yml +29 -0
  15. package/.husky/commit-msg +1 -0
  16. package/CHANGELOG.md +114 -0
  17. package/Dockerfile +32 -0
  18. package/README.es.md +174 -0
  19. package/README.md +187 -0
  20. package/agents/chronicler.md +98 -0
  21. package/agents/ci-smith.md +136 -0
  22. package/agents/craftsman.md +341 -0
  23. package/agents/deploy-smith.md +138 -0
  24. package/agents/foreman.md +377 -0
  25. package/agents/go-smith.md +164 -0
  26. package/agents/guild.md +188 -0
  27. package/agents/inspector.md +83 -0
  28. package/agents/js-smith.md +127 -0
  29. package/agents/ops-scout.md +173 -0
  30. package/agents/painter.md +200 -0
  31. package/agents/python-smith.md +120 -0
  32. package/agents/ranger.md +307 -0
  33. package/agents/release-smith.md +165 -0
  34. package/agents/rust-smith.md +159 -0
  35. package/agents/sage.md +178 -0
  36. package/agents/scout.md +144 -0
  37. package/agents/scribe.md +156 -0
  38. package/agents/smith.md +201 -0
  39. package/agents/vue-smith.md +155 -0
  40. package/agents/warden.md +216 -0
  41. package/agents/zig-smith.md +156 -0
  42. package/bin/ndomo-analyses.ts +4 -0
  43. package/bin/ndomo-status.ts +4 -0
  44. package/biome.json +57 -0
  45. package/bun.lock +514 -0
  46. package/commitlint.config.js +3 -0
  47. package/config/ndomo.config.json +258 -0
  48. package/config/ndomo.schema.json +166 -0
  49. package/docs/agents.md +375 -0
  50. package/docs/bugs/plan-create-orphan-fk.md +131 -0
  51. package/docs/bugs/task_create_batch-order-index-collision.md +158 -0
  52. package/docs/configuration.md +276 -0
  53. package/docs/database.md +364 -0
  54. package/docs/features/feature-flexible-builder-v1.md +724 -0
  55. package/docs/features/feature-flexible-builder-v2.md +882 -0
  56. package/docs/features/feature-flexible-builder.md +974 -0
  57. package/docs/http-server.md +244 -0
  58. package/docs/installation.md +259 -0
  59. package/docs/integrations.md +129 -0
  60. package/docs/operations/anti-pattern-sub-agent-verify-2026-06-21.md +32 -0
  61. package/docs/operations/audit-v1.md +417 -0
  62. package/docs/operations/audit-v2.md +197 -0
  63. package/docs/operations/audit-v3.md +306 -0
  64. package/docs/operations/db-optimize-foundations.md +123 -0
  65. package/docs/operations/verify-gate-architecture.md +82 -0
  66. package/docs/workflows.md +448 -0
  67. package/opencode.json +5 -0
  68. package/package.json +65 -0
  69. package/release-please-config.json +11 -0
  70. package/scripts/dev-bust-cache.sh +164 -0
  71. package/scripts/install.sh +688 -0
  72. package/scripts/smoke-e2e.ts +704 -0
  73. package/scripts/smoke-hot.ts +417 -0
  74. package/scripts/smoke-http.sh +228 -0
  75. package/scripts/smoke-v4.ts +256 -0
  76. package/scripts/smoke-v5.ts +397 -0
  77. package/scripts/smoke.sh +9 -0
  78. package/scripts/uninstall.sh +224 -0
  79. package/skills/api-security-best-practices/SKILL.md +915 -0
  80. package/skills/bash-scripting/SKILL.md +201 -0
  81. package/skills/bun/SKILL.md +313 -0
  82. package/skills/cavecrew/SKILL.md +82 -0
  83. package/skills/caveman/SKILL.md +74 -0
  84. package/skills/caveman-review/README.md +33 -0
  85. package/skills/caveman-review/SKILL.md +55 -0
  86. package/skills/find-skills/SKILL.md +142 -0
  87. package/skills/frontend-design/LICENSE.txt +177 -0
  88. package/skills/frontend-design/SKILL.md +55 -0
  89. package/skills/golang-patterns/SKILL.md +674 -0
  90. package/skills/golang-security/SKILL.md +185 -0
  91. package/skills/golang-security/evals/evals.json +595 -0
  92. package/skills/golang-security/references/architecture.md +268 -0
  93. package/skills/golang-security/references/checklist.md +80 -0
  94. package/skills/golang-security/references/cookies.md +200 -0
  95. package/skills/golang-security/references/cryptography.md +424 -0
  96. package/skills/golang-security/references/filesystem.md +285 -0
  97. package/skills/golang-security/references/injection.md +315 -0
  98. package/skills/golang-security/references/logging.md +163 -0
  99. package/skills/golang-security/references/memory-safety.md +241 -0
  100. package/skills/golang-security/references/network.md +253 -0
  101. package/skills/golang-security/references/secrets.md +189 -0
  102. package/skills/golang-security/references/third-party.md +159 -0
  103. package/skills/golang-security/references/threat-modeling.md +189 -0
  104. package/skills/golang-testing/SKILL.md +720 -0
  105. package/skills/grill-me/SKILL.md +7 -0
  106. package/skills/javascript-testing-patterns/SKILL.md +537 -0
  107. package/skills/javascript-testing-patterns/references/advanced-testing-patterns.md +513 -0
  108. package/skills/modern-javascript-patterns/SKILL.md +43 -0
  109. package/skills/modern-javascript-patterns/references/advanced-patterns.md +487 -0
  110. package/skills/modern-javascript-patterns/references/details.md +457 -0
  111. package/skills/python-anti-patterns/SKILL.md +349 -0
  112. package/skills/python-design-patterns/SKILL.md +85 -0
  113. package/skills/python-design-patterns/references/details.md +353 -0
  114. package/skills/python-error-handling/SKILL.md +193 -0
  115. package/skills/python-error-handling/references/details.md +171 -0
  116. package/skills/python-testing-patterns/SKILL.md +278 -0
  117. package/skills/python-testing-patterns/references/advanced-patterns.md +411 -0
  118. package/skills/python-testing-patterns/references/details.md +349 -0
  119. package/skills/rust-patterns/SKILL.md +500 -0
  120. package/skills/rust-testing/SKILL.md +501 -0
  121. package/skills/security-review/SKILL.md +504 -0
  122. package/skills/security-review/cloud-infrastructure-security.md +361 -0
  123. package/skills/vue-best-practices/SKILL.md +154 -0
  124. package/skills/vue-best-practices/references/animation-class-based-technique.md +254 -0
  125. package/skills/vue-best-practices/references/animation-state-driven-technique.md +291 -0
  126. package/skills/vue-best-practices/references/component-async.md +97 -0
  127. package/skills/vue-best-practices/references/component-data-flow.md +307 -0
  128. package/skills/vue-best-practices/references/component-fallthrough-attrs.md +174 -0
  129. package/skills/vue-best-practices/references/component-keep-alive.md +137 -0
  130. package/skills/vue-best-practices/references/component-slots.md +216 -0
  131. package/skills/vue-best-practices/references/component-suspense.md +228 -0
  132. package/skills/vue-best-practices/references/component-teleport.md +108 -0
  133. package/skills/vue-best-practices/references/component-transition-group.md +128 -0
  134. package/skills/vue-best-practices/references/component-transition.md +125 -0
  135. package/skills/vue-best-practices/references/composables.md +290 -0
  136. package/skills/vue-best-practices/references/directives.md +162 -0
  137. package/skills/vue-best-practices/references/perf-avoid-component-abstraction-in-lists.md +159 -0
  138. package/skills/vue-best-practices/references/perf-v-once-v-memo-directives.md +182 -0
  139. package/skills/vue-best-practices/references/perf-virtualize-large-lists.md +187 -0
  140. package/skills/vue-best-practices/references/plugins.md +166 -0
  141. package/skills/vue-best-practices/references/reactivity.md +344 -0
  142. package/skills/vue-best-practices/references/render-functions.md +201 -0
  143. package/skills/vue-best-practices/references/sfc.md +310 -0
  144. package/skills/vue-best-practices/references/state-management.md +135 -0
  145. package/skills/vue-best-practices/references/updated-hook-performance.md +187 -0
  146. package/skills/vue-pinia-best-practices/SKILL.md +21 -0
  147. package/skills/vue-pinia-best-practices/reference/pinia-no-active-pinia-error.md +248 -0
  148. package/skills/vue-pinia-best-practices/reference/pinia-setup-store-return-all-state.md +227 -0
  149. package/skills/vue-pinia-best-practices/reference/pinia-store-destructuring-breaks-reactivity.md +193 -0
  150. package/skills/vue-pinia-best-practices/reference/state-url-for-ephemeral-filters.md +238 -0
  151. package/skills/vue-pinia-best-practices/reference/state-use-pinia-for-large-apps.md +262 -0
  152. package/skills/vue-pinia-best-practices/reference/store-method-binding-parentheses.md +191 -0
  153. package/skills/zig-0.16/SKILL.md +840 -0
  154. package/skills/zig-0.16/scripts/check-zig-version.sh +21 -0
  155. package/src/cli/analyses.ts +280 -0
  156. package/src/cli/index.ts +108 -0
  157. package/src/cli/serve.ts +192 -0
  158. package/src/cli/smoke.ts +131 -0
  159. package/src/cli/status.test.ts +204 -0
  160. package/src/cli/status.ts +263 -0
  161. package/src/cli/vacuum.test.ts +82 -0
  162. package/src/cli/vacuum.ts +96 -0
  163. package/src/config/schema.test.ts +88 -0
  164. package/src/config/schema.ts +64 -0
  165. package/src/db/analyses-migration.test.ts +210 -0
  166. package/src/db/analyses.test.ts +466 -0
  167. package/src/db/analyses.ts +375 -0
  168. package/src/db/auto-checkpoint.ts +131 -0
  169. package/src/db/client.test.ts +129 -0
  170. package/src/db/client.ts +55 -0
  171. package/src/db/fts-escape.ts +20 -0
  172. package/src/db/incidents.test.ts +201 -0
  173. package/src/db/incidents.ts +93 -0
  174. package/src/db/index.ts +86 -0
  175. package/src/db/migrations-v13.test.ts +141 -0
  176. package/src/db/migrations-v8.test.ts +301 -0
  177. package/src/db/migrations.ts +147 -0
  178. package/src/db/plan-archive.test.ts +180 -0
  179. package/src/db/plan-archive.ts +274 -0
  180. package/src/db/plan-create.test.ts +276 -0
  181. package/src/db/plan-create.ts +78 -0
  182. package/src/db/plan-files.test.ts +289 -0
  183. package/src/db/plan-update-status.ts +287 -0
  184. package/src/db/plans.test.ts +490 -0
  185. package/src/db/plans.ts +534 -0
  186. package/src/db/resolve-project-dir.test.ts +143 -0
  187. package/src/db/resolve-project-dir.ts +75 -0
  188. package/src/db/rollbacks.test.ts +150 -0
  189. package/src/db/rollbacks.ts +67 -0
  190. package/src/db/schema.ts +907 -0
  191. package/src/db/sessions.test.ts +80 -0
  192. package/src/db/sessions.ts +135 -0
  193. package/src/db/shutdown.test.ts +147 -0
  194. package/src/db/shutdown.ts +45 -0
  195. package/src/db/tasks.test.ts +921 -0
  196. package/src/db/tasks.ts +747 -0
  197. package/src/db/types.ts +619 -0
  198. package/src/http/__tests__/auth.test.ts +196 -0
  199. package/src/http/__tests__/routes.test.ts +465 -0
  200. package/src/http/__tests__/sse.test.ts +317 -0
  201. package/src/http/auth.ts +72 -0
  202. package/src/http/middleware/cors.ts +53 -0
  203. package/src/http/middleware/security-headers.ts +21 -0
  204. package/src/http/routes/events.ts +112 -0
  205. package/src/http/routes/health.ts +51 -0
  206. package/src/http/routes/plans.ts +66 -0
  207. package/src/http/routes/sessions.ts +50 -0
  208. package/src/http/routes/tasks.ts +60 -0
  209. package/src/http/server.ts +95 -0
  210. package/src/http/sse.ts +116 -0
  211. package/src/index.ts +37 -0
  212. package/src/lib.ts +65 -0
  213. package/src/mem/scoped.ts +65 -0
  214. package/src/orchestrator/background.test.ts +268 -0
  215. package/src/orchestrator/background.ts +293 -0
  216. package/src/orchestrator/memory-hook.ts +182 -0
  217. package/src/orchestrator/reconciler.ts +123 -0
  218. package/src/orchestrator/scheduler.test.ts +300 -0
  219. package/src/orchestrator/scheduler.ts +243 -0
  220. package/src/plugin.test.ts +2574 -0
  221. package/src/plugin.ts +1690 -0
  222. package/src/sdk/client.ts +66 -0
  223. package/src/worktrees/manager.ts +236 -0
  224. package/src/worktrees/state.ts +87 -0
  225. package/tests/integration/ranger-flow.test.ts +257 -0
  226. package/tools/analysis_archive.ts +28 -0
  227. package/tools/analysis_create.ts +55 -0
  228. package/tools/analysis_get.ts +33 -0
  229. package/tools/analysis_link_plan.ts +44 -0
  230. package/tools/analysis_list.ts +48 -0
  231. package/tools/analysis_search.ts +36 -0
  232. package/tools/analysis_update.ts +44 -0
  233. package/tools/plan_approve.ts +31 -0
  234. package/tools/plan_create.ts +58 -0
  235. package/tools/plan_get.ts +40 -0
  236. package/tools/plan_list.ts +37 -0
  237. package/tools/plan_search.ts +34 -0
  238. package/tools/plan_update_status.ts +71 -0
  239. package/tools/session_checkpoint.ts +31 -0
  240. package/tools/session_end.ts +26 -0
  241. package/tools/session_start.ts +43 -0
  242. package/tools/task_create_batch.ts +70 -0
  243. package/tools/task_list.ts +35 -0
  244. package/tools/task_next_for_agent.ts +30 -0
  245. package/tools/task_search.ts +34 -0
  246. package/tools/task_update_status.ts +37 -0
  247. package/tsconfig.json +31 -0
@@ -0,0 +1,504 @@
1
+ ---
2
+ name: security-review
3
+ description: Use this skill when adding authentication, handling user input, working with secrets, creating API endpoints, or implementing payment/sensitive features. Provides comprehensive security checklist and patterns.
4
+ metadata:
5
+ origin: ECC
6
+ ---
7
+
8
+ # Security Review Skill
9
+
10
+ This skill ensures all code follows security best practices and identifies potential vulnerabilities.
11
+
12
+ ## When to Activate
13
+
14
+ - Implementing authentication or authorization
15
+ - Handling user input or file uploads
16
+ - Creating new API endpoints
17
+ - Working with secrets or credentials
18
+ - Implementing payment features
19
+ - Storing or transmitting sensitive data
20
+ - Integrating third-party APIs
21
+
22
+ ## Security Checklist
23
+
24
+ ### 1. Secrets Management
25
+
26
+ #### FAIL: NEVER Do This
27
+ ```typescript
28
+ const apiKey = "sk-proj-xxxxx" // Hardcoded secret
29
+ const dbPassword = "password123" // In source code
30
+ ```
31
+
32
+ #### PASS: ALWAYS Do This
33
+ ```typescript
34
+ const apiKey = process.env.OPENAI_API_KEY
35
+ const dbUrl = process.env.DATABASE_URL
36
+
37
+ // Verify secrets exist
38
+ if (!apiKey) {
39
+ throw new Error('OPENAI_API_KEY not configured')
40
+ }
41
+ ```
42
+
43
+ #### Verification Steps
44
+ - [ ] No hardcoded API keys, tokens, or passwords
45
+ - [ ] All secrets in environment variables
46
+ - [ ] `.env.local` in .gitignore
47
+ - [ ] No secrets in git history
48
+ - [ ] Production secrets in hosting platform (Vercel, Railway)
49
+
50
+ ### 2. Input Validation
51
+
52
+ #### Always Validate User Input
53
+ ```typescript
54
+ import { z } from 'zod'
55
+
56
+ // Define validation schema
57
+ const CreateUserSchema = z.object({
58
+ email: z.string().email(),
59
+ name: z.string().min(1).max(100),
60
+ age: z.number().int().min(0).max(150)
61
+ })
62
+
63
+ // Validate before processing
64
+ export async function createUser(input: unknown) {
65
+ try {
66
+ const validated = CreateUserSchema.parse(input)
67
+ return await db.users.create(validated)
68
+ } catch (error) {
69
+ if (error instanceof z.ZodError) {
70
+ return { success: false, errors: error.errors }
71
+ }
72
+ throw error
73
+ }
74
+ }
75
+ ```
76
+
77
+ #### File Upload Validation
78
+ ```typescript
79
+ function validateFileUpload(file: File) {
80
+ // Size check (5MB max)
81
+ const maxSize = 5 * 1024 * 1024
82
+ if (file.size > maxSize) {
83
+ throw new Error('File too large (max 5MB)')
84
+ }
85
+
86
+ // Type check
87
+ const allowedTypes = ['image/jpeg', 'image/png', 'image/gif']
88
+ if (!allowedTypes.includes(file.type)) {
89
+ throw new Error('Invalid file type')
90
+ }
91
+
92
+ // Extension check
93
+ const allowedExtensions = ['.jpg', '.jpeg', '.png', '.gif']
94
+ const extension = file.name.toLowerCase().match(/\.[^.]+$/)?.[0]
95
+ if (!extension || !allowedExtensions.includes(extension)) {
96
+ throw new Error('Invalid file extension')
97
+ }
98
+
99
+ return true
100
+ }
101
+ ```
102
+
103
+ #### Verification Steps
104
+ - [ ] All user inputs validated with schemas
105
+ - [ ] File uploads restricted (size, type, extension)
106
+ - [ ] No direct use of user input in queries
107
+ - [ ] Whitelist validation (not blacklist)
108
+ - [ ] Error messages don't leak sensitive info
109
+
110
+ ### 3. SQL Injection Prevention
111
+
112
+ #### FAIL: NEVER Concatenate SQL
113
+ ```typescript
114
+ // DANGEROUS - SQL Injection vulnerability
115
+ const query = `SELECT * FROM users WHERE email = '${userEmail}'`
116
+ await db.query(query)
117
+ ```
118
+
119
+ #### PASS: ALWAYS Use Parameterized Queries
120
+ ```typescript
121
+ // Safe - parameterized query
122
+ const { data } = await supabase
123
+ .from('users')
124
+ .select('*')
125
+ .eq('email', userEmail)
126
+
127
+ // Or with raw SQL
128
+ await db.query(
129
+ 'SELECT * FROM users WHERE email = $1',
130
+ [userEmail]
131
+ )
132
+ ```
133
+
134
+ #### Verification Steps
135
+ - [ ] All database queries use parameterized queries
136
+ - [ ] No string concatenation in SQL
137
+ - [ ] ORM/query builder used correctly
138
+ - [ ] Supabase queries properly sanitized
139
+
140
+ ### 4. Authentication & Authorization
141
+
142
+ #### JWT Token Handling
143
+ ```typescript
144
+ // FAIL: WRONG: localStorage (vulnerable to XSS)
145
+ localStorage.setItem('token', token)
146
+
147
+ // PASS: CORRECT: httpOnly cookies
148
+ res.setHeader('Set-Cookie',
149
+ `token=${token}; HttpOnly; Secure; SameSite=Strict; Max-Age=3600`)
150
+ ```
151
+
152
+ #### Authorization Checks
153
+ ```typescript
154
+ export async function deleteUser(userId: string, requesterId: string) {
155
+ // ALWAYS verify authorization first
156
+ const requester = await db.users.findUnique({
157
+ where: { id: requesterId }
158
+ })
159
+
160
+ if (requester.role !== 'admin') {
161
+ return NextResponse.json(
162
+ { error: 'Unauthorized' },
163
+ { status: 403 }
164
+ )
165
+ }
166
+
167
+ // Proceed with deletion
168
+ await db.users.delete({ where: { id: userId } })
169
+ }
170
+ ```
171
+
172
+ #### Row Level Security (Supabase)
173
+ ```sql
174
+ -- Enable RLS on all tables
175
+ ALTER TABLE users ENABLE ROW LEVEL SECURITY;
176
+
177
+ -- Users can only view their own data
178
+ CREATE POLICY "Users view own data"
179
+ ON users FOR SELECT
180
+ USING (auth.uid() = id);
181
+
182
+ -- Users can only update their own data
183
+ CREATE POLICY "Users update own data"
184
+ ON users FOR UPDATE
185
+ USING (auth.uid() = id);
186
+ ```
187
+
188
+ #### Verification Steps
189
+ - [ ] Tokens stored in httpOnly cookies (not localStorage)
190
+ - [ ] Authorization checks before sensitive operations
191
+ - [ ] Row Level Security enabled in Supabase
192
+ - [ ] Role-based access control implemented
193
+ - [ ] Session management secure
194
+
195
+ ### 5. XSS Prevention
196
+
197
+ #### Sanitize HTML
198
+ ```typescript
199
+ import DOMPurify from 'isomorphic-dompurify'
200
+
201
+ // ALWAYS sanitize user-provided HTML
202
+ function renderUserContent(html: string) {
203
+ const clean = DOMPurify.sanitize(html, {
204
+ ALLOWED_TAGS: ['b', 'i', 'em', 'strong', 'p'],
205
+ ALLOWED_ATTR: []
206
+ })
207
+ return <div dangerouslySetInnerHTML={{ __html: clean }} />
208
+ }
209
+ ```
210
+
211
+ #### Content Security Policy
212
+
213
+ Start strict and loosen only with a documented removal plan. Do not default to
214
+ `'unsafe-inline'` or `'unsafe-eval'`; they neutralize much of CSP's protection
215
+ and should be treated as temporary compatibility debt.
216
+
217
+ ```typescript
218
+ // next.config.js
219
+ const securityHeaders = [
220
+ {
221
+ key: 'Content-Security-Policy',
222
+ value: `
223
+ default-src 'self';
224
+ base-uri 'self';
225
+ object-src 'none';
226
+ frame-ancestors 'none';
227
+ script-src 'self';
228
+ style-src 'self';
229
+ img-src 'self' data: https:;
230
+ font-src 'self';
231
+ connect-src 'self' https://api.example.com;
232
+ `.replace(/\s{2,}/g, ' ').trim()
233
+ }
234
+ ]
235
+ ```
236
+
237
+ #### Verification Steps
238
+ - [ ] User-provided HTML sanitized
239
+ - [ ] CSP headers configured
240
+ - [ ] No unvalidated dynamic content rendering
241
+ - [ ] React's built-in XSS protection used
242
+
243
+ ### 6. CSRF Protection
244
+
245
+ #### CSRF Tokens
246
+ ```typescript
247
+ import { csrf } from '@/lib/csrf'
248
+
249
+ export async function POST(request: Request) {
250
+ const token = request.headers.get('X-CSRF-Token')
251
+
252
+ if (!csrf.verify(token)) {
253
+ return NextResponse.json(
254
+ { error: 'Invalid CSRF token' },
255
+ { status: 403 }
256
+ )
257
+ }
258
+
259
+ // Process request
260
+ }
261
+ ```
262
+
263
+ #### SameSite Cookies
264
+ ```typescript
265
+ res.setHeader('Set-Cookie',
266
+ `session=${sessionId}; HttpOnly; Secure; SameSite=Strict`)
267
+ ```
268
+
269
+ #### Verification Steps
270
+ - [ ] CSRF tokens on state-changing operations
271
+ - [ ] SameSite=Strict on all cookies
272
+ - [ ] Double-submit cookie pattern implemented
273
+
274
+ ### 7. Rate Limiting
275
+
276
+ #### API Rate Limiting
277
+ ```typescript
278
+ import rateLimit from 'express-rate-limit'
279
+
280
+ const limiter = rateLimit({
281
+ windowMs: 15 * 60 * 1000, // 15 minutes
282
+ max: 100, // 100 requests per window
283
+ message: 'Too many requests'
284
+ })
285
+
286
+ // Apply to routes
287
+ app.use('/api/', limiter)
288
+ ```
289
+
290
+ #### Expensive Operations
291
+ ```typescript
292
+ // Aggressive rate limiting for searches
293
+ const searchLimiter = rateLimit({
294
+ windowMs: 60 * 1000, // 1 minute
295
+ max: 10, // 10 requests per minute
296
+ message: 'Too many search requests'
297
+ })
298
+
299
+ app.use('/api/search', searchLimiter)
300
+ ```
301
+
302
+ #### Verification Steps
303
+ - [ ] Rate limiting on all API endpoints
304
+ - [ ] Stricter limits on expensive operations
305
+ - [ ] IP-based rate limiting
306
+ - [ ] User-based rate limiting (authenticated)
307
+
308
+ ### 8. Sensitive Data Exposure
309
+
310
+ #### Logging
311
+ ```typescript
312
+ // FAIL: WRONG: Logging sensitive data
313
+ console.log('User login:', { email, password })
314
+ console.log('Payment:', { cardNumber, cvv })
315
+
316
+ // PASS: CORRECT: Redact sensitive data
317
+ console.log('User login:', { email, userId })
318
+ console.log('Payment:', { last4: card.last4, userId })
319
+ ```
320
+
321
+ #### Error Messages
322
+ ```typescript
323
+ // FAIL: WRONG: Exposing internal details
324
+ catch (error) {
325
+ return NextResponse.json(
326
+ { error: error.message, stack: error.stack },
327
+ { status: 500 }
328
+ )
329
+ }
330
+
331
+ // PASS: CORRECT: Generic error messages
332
+ catch (error) {
333
+ console.error('Internal error:', error)
334
+ return NextResponse.json(
335
+ { error: 'An error occurred. Please try again.' },
336
+ { status: 500 }
337
+ )
338
+ }
339
+ ```
340
+
341
+ #### Verification Steps
342
+ - [ ] No passwords, tokens, or secrets in logs
343
+ - [ ] Error messages generic for users
344
+ - [ ] Detailed errors only in server logs
345
+ - [ ] No stack traces exposed to users
346
+
347
+ ### 9. Blockchain Security (Solana)
348
+
349
+ #### Wallet Verification
350
+ ```typescript
351
+ import { verify } from '@solana/web3.js'
352
+
353
+ async function verifyWalletOwnership(
354
+ publicKey: string,
355
+ signature: string,
356
+ message: string
357
+ ) {
358
+ try {
359
+ const isValid = verify(
360
+ Buffer.from(message),
361
+ Buffer.from(signature, 'base64'),
362
+ Buffer.from(publicKey, 'base64')
363
+ )
364
+ return isValid
365
+ } catch (error) {
366
+ return false
367
+ }
368
+ }
369
+ ```
370
+
371
+ #### Transaction Verification
372
+ ```typescript
373
+ async function verifyTransaction(transaction: Transaction) {
374
+ // Verify recipient
375
+ if (transaction.to !== expectedRecipient) {
376
+ throw new Error('Invalid recipient')
377
+ }
378
+
379
+ // Verify amount
380
+ if (transaction.amount > maxAmount) {
381
+ throw new Error('Amount exceeds limit')
382
+ }
383
+
384
+ // Verify user has sufficient balance
385
+ const balance = await getBalance(transaction.from)
386
+ if (balance < transaction.amount) {
387
+ throw new Error('Insufficient balance')
388
+ }
389
+
390
+ return true
391
+ }
392
+ ```
393
+
394
+ #### Verification Steps
395
+ - [ ] Wallet signatures verified
396
+ - [ ] Transaction details validated
397
+ - [ ] Balance checks before transactions
398
+ - [ ] No blind transaction signing
399
+
400
+ ### 10. Dependency Security
401
+
402
+ #### Regular Updates
403
+ ```bash
404
+ # Check for vulnerabilities
405
+ npm audit
406
+
407
+ # Fix automatically fixable issues
408
+ npm audit fix
409
+
410
+ # Update dependencies
411
+ npm update
412
+
413
+ # Check for outdated packages
414
+ npm outdated
415
+ ```
416
+
417
+ #### Lock Files
418
+ ```bash
419
+ # ALWAYS commit lock files
420
+ git add package-lock.json
421
+
422
+ # Use in CI/CD for reproducible builds
423
+ npm ci # Instead of npm install
424
+ ```
425
+
426
+ #### Verification Steps
427
+ - [ ] Dependencies up to date
428
+ - [ ] No known vulnerabilities (npm audit clean)
429
+ - [ ] Lock files committed
430
+ - [ ] Dependabot enabled on GitHub
431
+ - [ ] Regular security updates
432
+
433
+ ## Security Testing
434
+
435
+ ### Automated Security Tests
436
+ ```typescript
437
+ // Test authentication
438
+ test('requires authentication', async () => {
439
+ const response = await fetch('/api/protected')
440
+ expect(response.status).toBe(401)
441
+ })
442
+
443
+ // Test authorization
444
+ test('requires admin role', async () => {
445
+ const response = await fetch('/api/admin', {
446
+ headers: { Authorization: `Bearer ${userToken}` }
447
+ })
448
+ expect(response.status).toBe(403)
449
+ })
450
+
451
+ // Test input validation
452
+ test('rejects invalid input', async () => {
453
+ const response = await fetch('/api/users', {
454
+ method: 'POST',
455
+ body: JSON.stringify({ email: 'not-an-email' })
456
+ })
457
+ expect(response.status).toBe(400)
458
+ })
459
+
460
+ // Test rate limiting
461
+ test('enforces rate limits', async () => {
462
+ const requests = Array(101).fill(null).map(() =>
463
+ fetch('/api/endpoint')
464
+ )
465
+
466
+ const responses = await Promise.all(requests)
467
+ const tooManyRequests = responses.filter(r => r.status === 429)
468
+
469
+ expect(tooManyRequests.length).toBeGreaterThan(0)
470
+ })
471
+ ```
472
+
473
+ ## Pre-Deployment Security Checklist
474
+
475
+ Before ANY production deployment:
476
+
477
+ - [ ] **Secrets**: No hardcoded secrets, all in env vars
478
+ - [ ] **Input Validation**: All user inputs validated
479
+ - [ ] **SQL Injection**: All queries parameterized
480
+ - [ ] **XSS**: User content sanitized
481
+ - [ ] **CSRF**: Protection enabled
482
+ - [ ] **Authentication**: Proper token handling
483
+ - [ ] **Authorization**: Role checks in place
484
+ - [ ] **Rate Limiting**: Enabled on all endpoints
485
+ - [ ] **HTTPS**: Enforced in production
486
+ - [ ] **Security Headers**: CSP, X-Frame-Options configured
487
+ - [ ] **Error Handling**: No sensitive data in errors
488
+ - [ ] **Logging**: No sensitive data logged
489
+ - [ ] **Dependencies**: Up to date, no vulnerabilities
490
+ - [ ] **Row Level Security**: Enabled in Supabase
491
+ - [ ] **CORS**: Properly configured
492
+ - [ ] **File Uploads**: Validated (size, type)
493
+ - [ ] **Wallet Signatures**: Verified (if blockchain)
494
+
495
+ ## Resources
496
+
497
+ - [OWASP Top 10](https://owasp.org/www-project-top-ten/)
498
+ - [Next.js Security](https://nextjs.org/docs/security)
499
+ - [Supabase Security](https://supabase.com/docs/guides/auth)
500
+ - [Web Security Academy](https://portswigger.net/web-security)
501
+
502
+ ---
503
+
504
+ **Remember**: Security is not optional. One vulnerability can compromise the entire platform. When in doubt, err on the side of caution.