vaspera 2.9.2 → 2.10.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.
- package/CHANGELOG.md +68 -0
- package/README.md +58 -1
- package/dist/__tests__/autofix/branch-manager.test.d.ts +2 -0
- package/dist/__tests__/autofix/branch-manager.test.d.ts.map +1 -0
- package/dist/__tests__/autofix/branch-manager.test.js +60 -0
- package/dist/__tests__/autofix/branch-manager.test.js.map +1 -0
- package/dist/__tests__/autofix/commit-generator.test.d.ts +2 -0
- package/dist/__tests__/autofix/commit-generator.test.d.ts.map +1 -0
- package/dist/__tests__/autofix/commit-generator.test.js +147 -0
- package/dist/__tests__/autofix/commit-generator.test.js.map +1 -0
- package/dist/__tests__/autofix/constitution.test.d.ts +9 -0
- package/dist/__tests__/autofix/constitution.test.d.ts.map +1 -0
- package/dist/__tests__/autofix/constitution.test.js +421 -0
- package/dist/__tests__/autofix/constitution.test.js.map +1 -0
- package/dist/__tests__/autofix/pr-generator.test.d.ts +2 -0
- package/dist/__tests__/autofix/pr-generator.test.d.ts.map +1 -0
- package/dist/__tests__/autofix/pr-generator.test.js +152 -0
- package/dist/__tests__/autofix/pr-generator.test.js.map +1 -0
- package/dist/__tests__/property-test-helpers.d.ts +87 -0
- package/dist/__tests__/property-test-helpers.d.ts.map +1 -0
- package/dist/__tests__/property-test-helpers.js +136 -0
- package/dist/__tests__/property-test-helpers.js.map +1 -0
- package/dist/__tests__/scanners/dast/index.test.d.ts +2 -0
- package/dist/__tests__/scanners/dast/index.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/dast/index.test.js +183 -0
- package/dist/__tests__/scanners/dast/index.test.js.map +1 -0
- package/dist/__tests__/scanners/dast/nuclei.test.d.ts +2 -0
- package/dist/__tests__/scanners/dast/nuclei.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/dast/nuclei.test.js +166 -0
- package/dist/__tests__/scanners/dast/nuclei.test.js.map +1 -0
- package/dist/__tests__/scanners/dast/zap.test.d.ts +2 -0
- package/dist/__tests__/scanners/dast/zap.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/dast/zap.test.js +158 -0
- package/dist/__tests__/scanners/dast/zap.test.js.map +1 -0
- package/dist/__tests__/scanners/fp-feedback.test.d.ts +2 -0
- package/dist/__tests__/scanners/fp-feedback.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/fp-feedback.test.js +202 -0
- package/dist/__tests__/scanners/fp-feedback.test.js.map +1 -0
- package/dist/__tests__/scanners/fp-filter.property.test.d.ts +9 -0
- package/dist/__tests__/scanners/fp-filter.property.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/fp-filter.property.test.js +253 -0
- package/dist/__tests__/scanners/fp-filter.property.test.js.map +1 -0
- package/dist/__tests__/scanners/fp-filter.test.d.ts +2 -0
- package/dist/__tests__/scanners/fp-filter.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/fp-filter.test.js +234 -0
- package/dist/__tests__/scanners/fp-filter.test.js.map +1 -0
- package/dist/__tests__/scanners/fp-tracker.test.d.ts +2 -0
- package/dist/__tests__/scanners/fp-tracker.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/fp-tracker.test.js +262 -0
- package/dist/__tests__/scanners/fp-tracker.test.js.map +1 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.property.test.d.ts +10 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.property.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.property.test.js +238 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.property.test.js.map +1 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.test.d.ts +2 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.test.js +55 -0
- package/dist/__tests__/scanners/logic/endpoint-analyzer.test.js.map +1 -0
- package/dist/__tests__/scanners/logic/index.test.d.ts +2 -0
- package/dist/__tests__/scanners/logic/index.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/logic/index.test.js +165 -0
- package/dist/__tests__/scanners/logic/index.test.js.map +1 -0
- package/dist/__tests__/scanners/logic/types.test.d.ts +2 -0
- package/dist/__tests__/scanners/logic/types.test.d.ts.map +1 -0
- package/dist/__tests__/scanners/logic/types.test.js +85 -0
- package/dist/__tests__/scanners/logic/types.test.js.map +1 -0
- package/dist/action/pr-comment.test.js +4 -0
- package/dist/action/pr-comment.test.js.map +1 -1
- package/dist/action/sarif-upload.test.js +4 -0
- package/dist/action/sarif-upload.test.js.map +1 -1
- package/dist/autofix/branch-manager.d.ts +115 -0
- package/dist/autofix/branch-manager.d.ts.map +1 -0
- package/dist/autofix/branch-manager.js +308 -0
- package/dist/autofix/branch-manager.js.map +1 -0
- package/dist/autofix/commit-generator.d.ts +55 -0
- package/dist/autofix/commit-generator.d.ts.map +1 -0
- package/dist/autofix/commit-generator.js +277 -0
- package/dist/autofix/commit-generator.js.map +1 -0
- package/dist/autofix/constitution.d.ts +77 -0
- package/dist/autofix/constitution.d.ts.map +1 -0
- package/dist/autofix/constitution.js +261 -0
- package/dist/autofix/constitution.js.map +1 -0
- package/dist/autofix/constitution.schema.d.ts +441 -0
- package/dist/autofix/constitution.schema.d.ts.map +1 -0
- package/dist/autofix/constitution.schema.js +144 -0
- package/dist/autofix/constitution.schema.js.map +1 -0
- package/dist/autofix/index.d.ts +13 -0
- package/dist/autofix/index.d.ts.map +1 -0
- package/dist/autofix/index.js +15 -0
- package/dist/autofix/index.js.map +1 -0
- package/dist/autofix/pr-generator.d.ts +57 -0
- package/dist/autofix/pr-generator.d.ts.map +1 -0
- package/dist/autofix/pr-generator.js +597 -0
- package/dist/autofix/pr-generator.js.map +1 -0
- package/dist/autofix/types.d.ts +151 -0
- package/dist/autofix/types.d.ts.map +1 -0
- package/dist/autofix/types.js +22 -0
- package/dist/autofix/types.js.map +1 -0
- package/dist/eval/fixtures.d.ts +20 -0
- package/dist/eval/fixtures.d.ts.map +1 -1
- package/dist/eval/fixtures.js +430 -0
- package/dist/eval/fixtures.js.map +1 -1
- package/dist/scanners/cache.d.ts.map +1 -1
- package/dist/scanners/cache.js +4 -0
- package/dist/scanners/cache.js.map +1 -1
- package/dist/scanners/dast/index.d.ts +39 -0
- package/dist/scanners/dast/index.d.ts.map +1 -0
- package/dist/scanners/dast/index.js +259 -0
- package/dist/scanners/dast/index.js.map +1 -0
- package/dist/scanners/dast/nuclei.d.ts +26 -0
- package/dist/scanners/dast/nuclei.d.ts.map +1 -0
- package/dist/scanners/dast/nuclei.js +354 -0
- package/dist/scanners/dast/nuclei.js.map +1 -0
- package/dist/scanners/dast/types.d.ts +306 -0
- package/dist/scanners/dast/types.d.ts.map +1 -0
- package/dist/scanners/dast/types.js +52 -0
- package/dist/scanners/dast/types.js.map +1 -0
- package/dist/scanners/dast/zap.d.ts +26 -0
- package/dist/scanners/dast/zap.d.ts.map +1 -0
- package/dist/scanners/dast/zap.js +453 -0
- package/dist/scanners/dast/zap.js.map +1 -0
- package/dist/scanners/fp-feedback.d.ts +140 -0
- package/dist/scanners/fp-feedback.d.ts.map +1 -0
- package/dist/scanners/fp-feedback.js +292 -0
- package/dist/scanners/fp-feedback.js.map +1 -0
- package/dist/scanners/fp-filter.d.ts +94 -0
- package/dist/scanners/fp-filter.d.ts.map +1 -0
- package/dist/scanners/fp-filter.js +397 -0
- package/dist/scanners/fp-filter.js.map +1 -0
- package/dist/scanners/fp-tracker.d.ts +125 -0
- package/dist/scanners/fp-tracker.d.ts.map +1 -0
- package/dist/scanners/fp-tracker.js +330 -0
- package/dist/scanners/fp-tracker.js.map +1 -0
- package/dist/scanners/index.d.ts.map +1 -1
- package/dist/scanners/index.js +56 -0
- package/dist/scanners/index.js.map +1 -1
- package/dist/scanners/index.test.js +6 -6
- package/dist/scanners/index.test.js.map +1 -1
- package/dist/scanners/logic/auth-flow-analyzer.d.ts +18 -0
- package/dist/scanners/logic/auth-flow-analyzer.d.ts.map +1 -0
- package/dist/scanners/logic/auth-flow-analyzer.js +384 -0
- package/dist/scanners/logic/auth-flow-analyzer.js.map +1 -0
- package/dist/scanners/logic/endpoint-analyzer.d.ts +29 -0
- package/dist/scanners/logic/endpoint-analyzer.d.ts.map +1 -0
- package/dist/scanners/logic/endpoint-analyzer.js +528 -0
- package/dist/scanners/logic/endpoint-analyzer.js.map +1 -0
- package/dist/scanners/logic/index.d.ts +41 -0
- package/dist/scanners/logic/index.d.ts.map +1 -0
- package/dist/scanners/logic/index.js +268 -0
- package/dist/scanners/logic/index.js.map +1 -0
- package/dist/scanners/logic/types.d.ts +254 -0
- package/dist/scanners/logic/types.d.ts.map +1 -0
- package/dist/scanners/logic/types.js +142 -0
- package/dist/scanners/logic/types.js.map +1 -0
- package/dist/scanners/types.d.ts +1 -1
- package/dist/scanners/types.d.ts.map +1 -1
- package/dist/scanners/types.js +4 -0
- package/dist/scanners/types.js.map +1 -1
- package/package.json +5 -3
|
@@ -0,0 +1,528 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* API Endpoint Analyzer
|
|
3
|
+
*
|
|
4
|
+
* Extracts API endpoints from source code for various frameworks.
|
|
5
|
+
*
|
|
6
|
+
* @module scanners/logic/endpoint-analyzer
|
|
7
|
+
*/
|
|
8
|
+
import { readFile } from "fs/promises";
|
|
9
|
+
import { join } from "path";
|
|
10
|
+
import { glob } from "glob";
|
|
11
|
+
import { logger } from "../../logger.js";
|
|
12
|
+
/**
|
|
13
|
+
* Route patterns for different frameworks
|
|
14
|
+
*/
|
|
15
|
+
const ROUTE_PATTERNS = {
|
|
16
|
+
nextjs: [
|
|
17
|
+
// App Router: export async function GET/POST/etc
|
|
18
|
+
{ pattern: /export\s+(?:async\s+)?function\s+(GET|POST|PUT|DELETE|PATCH|HEAD|OPTIONS)\s*\(/gi, methodGroup: 1, pathGroup: -1 },
|
|
19
|
+
// Pages Router: handler switch on method
|
|
20
|
+
{ pattern: /req\.method\s*===?\s*['"`](GET|POST|PUT|DELETE|PATCH)['"`]/gi, methodGroup: 1, pathGroup: -1 },
|
|
21
|
+
],
|
|
22
|
+
express: [
|
|
23
|
+
// app.get('/path', handler) or router.get('/path', handler)
|
|
24
|
+
{ pattern: /(?:app|router)\.(get|post|put|delete|patch|all)\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
25
|
+
// app.use('/path', router)
|
|
26
|
+
{ pattern: /(?:app|router)\.use\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: -1, pathGroup: 1 },
|
|
27
|
+
],
|
|
28
|
+
fastify: [
|
|
29
|
+
// fastify.get('/path', handler)
|
|
30
|
+
{ pattern: /fastify\.(get|post|put|delete|patch)\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
31
|
+
// fastify.route({ method, url })
|
|
32
|
+
{ pattern: /fastify\.route\s*\(\s*\{[^}]*method:\s*['"`](GET|POST|PUT|DELETE|PATCH)['"`][^}]*url:\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
33
|
+
],
|
|
34
|
+
koa: [
|
|
35
|
+
{ pattern: /router\.(get|post|put|delete|patch)\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
36
|
+
],
|
|
37
|
+
hapi: [
|
|
38
|
+
{ pattern: /server\.route\s*\(\s*\{[^}]*method:\s*['"`](GET|POST|PUT|DELETE|PATCH)['"`][^}]*path:\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
39
|
+
],
|
|
40
|
+
nestjs: [
|
|
41
|
+
// @Get('/path'), @Post('/path'), etc.
|
|
42
|
+
{ pattern: /@(Get|Post|Put|Delete|Patch)\s*\(\s*['"`]?([^'"`)\s]*)['"`]?\s*\)/gi, methodGroup: 1, pathGroup: 2 },
|
|
43
|
+
// @Controller('/path')
|
|
44
|
+
{ pattern: /@Controller\s*\(\s*['"`]([^'"`]+)['"`]\s*\)/gi, methodGroup: -1, pathGroup: 1 },
|
|
45
|
+
],
|
|
46
|
+
django: [
|
|
47
|
+
// path('route/', view)
|
|
48
|
+
{ pattern: /path\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: -1, pathGroup: 1 },
|
|
49
|
+
// @api_view(['GET', 'POST'])
|
|
50
|
+
{ pattern: /@api_view\s*\(\s*\[([^\]]+)\]/gi, methodGroup: 1, pathGroup: -1 },
|
|
51
|
+
],
|
|
52
|
+
flask: [
|
|
53
|
+
// @app.route('/path', methods=['GET', 'POST'])
|
|
54
|
+
{ pattern: /@(?:app|blueprint)\.route\s*\(\s*['"`]([^'"`]+)['"`](?:[^)]*methods\s*=\s*\[([^\]]+)\])?/gi, methodGroup: 2, pathGroup: 1 },
|
|
55
|
+
],
|
|
56
|
+
fastapi: [
|
|
57
|
+
// @app.get('/path')
|
|
58
|
+
{ pattern: /@(?:app|router)\.(get|post|put|delete|patch)\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
59
|
+
],
|
|
60
|
+
rails: [
|
|
61
|
+
// get '/path', to: 'controller#action'
|
|
62
|
+
{ pattern: /(get|post|put|patch|delete)\s+['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
63
|
+
// resources :users
|
|
64
|
+
{ pattern: /resources?\s+:(\w+)/gi, methodGroup: -1, pathGroup: 1 },
|
|
65
|
+
],
|
|
66
|
+
spring: [
|
|
67
|
+
// @GetMapping("/path")
|
|
68
|
+
{ pattern: /@(Get|Post|Put|Delete|Patch)Mapping\s*\(\s*(?:value\s*=\s*)?['"`]?([^'"`)\s]*)['"`]?\s*\)/gi, methodGroup: 1, pathGroup: 2 },
|
|
69
|
+
// @RequestMapping(method = RequestMethod.GET, path = "/path")
|
|
70
|
+
{ pattern: /@RequestMapping\s*\([^)]*method\s*=\s*RequestMethod\.(GET|POST|PUT|DELETE|PATCH)[^)]*(?:path|value)\s*=\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
71
|
+
],
|
|
72
|
+
laravel: [
|
|
73
|
+
// Route::get('/path', [Controller::class, 'method'])
|
|
74
|
+
{ pattern: /Route::(get|post|put|patch|delete)\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
75
|
+
],
|
|
76
|
+
gin: [
|
|
77
|
+
// r.GET("/path", handler)
|
|
78
|
+
{ pattern: /[rg]\.(GET|POST|PUT|DELETE|PATCH)\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
79
|
+
],
|
|
80
|
+
echo: [
|
|
81
|
+
// e.GET("/path", handler)
|
|
82
|
+
{ pattern: /e\.(GET|POST|PUT|DELETE|PATCH)\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
83
|
+
],
|
|
84
|
+
fiber: [
|
|
85
|
+
// app.Get("/path", handler)
|
|
86
|
+
{ pattern: /app\.(Get|Post|Put|Delete|Patch)\s*\(\s*['"`]([^'"`]+)['"`]/gi, methodGroup: 1, pathGroup: 2 },
|
|
87
|
+
],
|
|
88
|
+
auto: [],
|
|
89
|
+
};
|
|
90
|
+
/**
|
|
91
|
+
* Patterns to detect authentication middleware
|
|
92
|
+
*/
|
|
93
|
+
const AUTH_MIDDLEWARE_PATTERNS = [
|
|
94
|
+
/auth(?:enticate|orize)?/i,
|
|
95
|
+
/requireAuth/i,
|
|
96
|
+
/isAuthenticated/i,
|
|
97
|
+
/ensureLoggedIn/i,
|
|
98
|
+
/protect(?:ed)?Route/i,
|
|
99
|
+
/jwt(?:Auth|Verify|Guard)?/i,
|
|
100
|
+
/passport\./,
|
|
101
|
+
/session\./,
|
|
102
|
+
/@UseGuards\s*\([^)]*AuthGuard/i,
|
|
103
|
+
/middleware\s*=\s*\[.*auth/i,
|
|
104
|
+
/IsAuthenticated\(\)/i,
|
|
105
|
+
/Depends\s*\(\s*get_current_user/i,
|
|
106
|
+
/before_action\s+:authenticate/i,
|
|
107
|
+
/auth:api/i,
|
|
108
|
+
];
|
|
109
|
+
/**
|
|
110
|
+
* Patterns to detect authorization checks
|
|
111
|
+
*/
|
|
112
|
+
const AUTHZ_CHECK_PATTERNS = [
|
|
113
|
+
{ pattern: /user\.id\s*===?\s*(?:req|request|params|ctx)\.[^;]+/i, type: "ownership" },
|
|
114
|
+
{ pattern: /\.userId\s*===?\s*(?:req|request|session)\.user/i, type: "ownership" },
|
|
115
|
+
{ pattern: /belongsTo|ownedBy|createdBy|authorId/i, type: "ownership" },
|
|
116
|
+
{ pattern: /isOwner|canAccess|hasPermission/i, type: "permission" },
|
|
117
|
+
{ pattern: /user\.role\s*===?\s*['"`]admin['"`]/i, type: "admin" },
|
|
118
|
+
{ pattern: /\.isAdmin|isSuper|isModerator/i, type: "admin" },
|
|
119
|
+
{ pattern: /roles?\.(includes|contains|has)\s*\(/i, type: "role" },
|
|
120
|
+
{ pattern: /@Roles?\s*\(/i, type: "role" },
|
|
121
|
+
{ pattern: /permissions?\.(includes|contains|has)\s*\(/i, type: "permission" },
|
|
122
|
+
{ pattern: /can\s*\(\s*['"`]\w+['"`]/i, type: "permission" },
|
|
123
|
+
{ pattern: /authorize|checkPermission|requireRole/i, type: "custom" },
|
|
124
|
+
];
|
|
125
|
+
/**
|
|
126
|
+
* Patterns to detect database access
|
|
127
|
+
*/
|
|
128
|
+
const DB_ACCESS_PATTERNS = [
|
|
129
|
+
{ pattern: /\.findOne\s*\(/i, type: "select" },
|
|
130
|
+
{ pattern: /\.findById\s*\(/i, type: "select" },
|
|
131
|
+
{ pattern: /\.findUnique\s*\(/i, type: "select" },
|
|
132
|
+
{ pattern: /\.findFirst\s*\(/i, type: "select" },
|
|
133
|
+
{ pattern: /\.find\s*\(\s*\{/i, type: "select" },
|
|
134
|
+
{ pattern: /\.select\s*\(/i, type: "select" },
|
|
135
|
+
{ pattern: /SELECT\s+.*\s+FROM/i, type: "select" },
|
|
136
|
+
{ pattern: /\.create\s*\(/i, type: "insert" },
|
|
137
|
+
{ pattern: /\.insert\s*\(/i, type: "insert" },
|
|
138
|
+
{ pattern: /INSERT\s+INTO/i, type: "insert" },
|
|
139
|
+
{ pattern: /\.update\s*\(/i, type: "update" },
|
|
140
|
+
{ pattern: /\.updateOne\s*\(/i, type: "update" },
|
|
141
|
+
{ pattern: /\.updateMany\s*\(/i, type: "update" },
|
|
142
|
+
{ pattern: /UPDATE\s+\w+\s+SET/i, type: "update" },
|
|
143
|
+
{ pattern: /\.delete\s*\(/i, type: "delete" },
|
|
144
|
+
{ pattern: /\.deleteOne\s*\(/i, type: "delete" },
|
|
145
|
+
{ pattern: /\.deleteMany\s*\(/i, type: "delete" },
|
|
146
|
+
{ pattern: /DELETE\s+FROM/i, type: "delete" },
|
|
147
|
+
{ pattern: /\.query\s*\(/i, type: "raw" },
|
|
148
|
+
{ pattern: /\.raw\s*\(/i, type: "raw" },
|
|
149
|
+
{ pattern: /\$queryRaw/i, type: "raw" },
|
|
150
|
+
];
|
|
151
|
+
/**
|
|
152
|
+
* Extract path parameters from route path
|
|
153
|
+
*/
|
|
154
|
+
export function extractPathParams(path) {
|
|
155
|
+
const params = [];
|
|
156
|
+
// Express/Fastify style :param
|
|
157
|
+
const colonParams = path.match(/:(\w+)/g);
|
|
158
|
+
if (colonParams) {
|
|
159
|
+
params.push(...colonParams.map((p) => p.slice(1)));
|
|
160
|
+
}
|
|
161
|
+
// Next.js/bracket style [param]
|
|
162
|
+
const bracketParams = path.match(/\[(\w+)\]/g);
|
|
163
|
+
if (bracketParams) {
|
|
164
|
+
params.push(...bracketParams.map((p) => p.slice(1, -1)));
|
|
165
|
+
}
|
|
166
|
+
// Django/Flask style <param>
|
|
167
|
+
const angleParams = path.match(/<(\w+)(?::\w+)?>/g);
|
|
168
|
+
if (angleParams) {
|
|
169
|
+
params.push(...angleParams.map((p) => p.replace(/<|>|:\w+/g, "")));
|
|
170
|
+
}
|
|
171
|
+
// Spring style {param}
|
|
172
|
+
const curlyParams = path.match(/\{(\w+)\}/g);
|
|
173
|
+
if (curlyParams) {
|
|
174
|
+
params.push(...curlyParams.map((p) => p.slice(1, -1)));
|
|
175
|
+
}
|
|
176
|
+
return params;
|
|
177
|
+
}
|
|
178
|
+
/**
|
|
179
|
+
* Detect framework from package.json
|
|
180
|
+
*/
|
|
181
|
+
export async function detectFramework(projectPath) {
|
|
182
|
+
try {
|
|
183
|
+
const pkgPath = join(projectPath, "package.json");
|
|
184
|
+
const pkgContent = await readFile(pkgPath, "utf-8");
|
|
185
|
+
const pkg = JSON.parse(pkgContent);
|
|
186
|
+
const deps = { ...pkg.dependencies, ...pkg.devDependencies };
|
|
187
|
+
// Check in priority order
|
|
188
|
+
if (deps["next"])
|
|
189
|
+
return "nextjs";
|
|
190
|
+
if (deps["@nestjs/core"])
|
|
191
|
+
return "nestjs";
|
|
192
|
+
if (deps["fastify"])
|
|
193
|
+
return "fastify";
|
|
194
|
+
if (deps["koa"])
|
|
195
|
+
return "koa";
|
|
196
|
+
if (deps["@hapi/hapi"])
|
|
197
|
+
return "hapi";
|
|
198
|
+
if (deps["express"])
|
|
199
|
+
return "express";
|
|
200
|
+
// Check for Python frameworks (look for requirements.txt or pyproject.toml)
|
|
201
|
+
// This is a simplified check
|
|
202
|
+
try {
|
|
203
|
+
const reqPath = join(projectPath, "requirements.txt");
|
|
204
|
+
const reqContent = await readFile(reqPath, "utf-8");
|
|
205
|
+
if (reqContent.includes("fastapi"))
|
|
206
|
+
return "fastapi";
|
|
207
|
+
if (reqContent.includes("flask"))
|
|
208
|
+
return "flask";
|
|
209
|
+
if (reqContent.includes("django"))
|
|
210
|
+
return "django";
|
|
211
|
+
}
|
|
212
|
+
catch {
|
|
213
|
+
// No requirements.txt
|
|
214
|
+
}
|
|
215
|
+
return "auto";
|
|
216
|
+
}
|
|
217
|
+
catch {
|
|
218
|
+
return "auto";
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* Extract API path from Next.js file path
|
|
223
|
+
*/
|
|
224
|
+
function getNextjsApiPath(filePath) {
|
|
225
|
+
// app/api/users/[id]/route.ts -> /api/users/[id]
|
|
226
|
+
const appMatch = filePath.match(/app\/(api\/[^/]+(?:\/[^/]+)*?)\/route\.[jt]sx?$/);
|
|
227
|
+
if (appMatch) {
|
|
228
|
+
return "/" + appMatch[1];
|
|
229
|
+
}
|
|
230
|
+
// pages/api/users/[id].ts -> /api/users/[id]
|
|
231
|
+
const pagesMatch = filePath.match(/pages\/(api\/[^/]+(?:\/[^/]+)*?)\.[jt]sx?$/);
|
|
232
|
+
if (pagesMatch) {
|
|
233
|
+
return "/" + pagesMatch[1];
|
|
234
|
+
}
|
|
235
|
+
return "";
|
|
236
|
+
}
|
|
237
|
+
/**
|
|
238
|
+
* Check if code has authentication middleware
|
|
239
|
+
*/
|
|
240
|
+
function hasAuthMiddleware(content) {
|
|
241
|
+
return AUTH_MIDDLEWARE_PATTERNS.some((pattern) => pattern.test(content));
|
|
242
|
+
}
|
|
243
|
+
/**
|
|
244
|
+
* Find authorization checks in code
|
|
245
|
+
*/
|
|
246
|
+
function findAuthzChecks(content, filePath) {
|
|
247
|
+
const checks = [];
|
|
248
|
+
const lines = content.split("\n");
|
|
249
|
+
for (let i = 0; i < lines.length; i++) {
|
|
250
|
+
const line = lines[i];
|
|
251
|
+
for (const { pattern, type } of AUTHZ_CHECK_PATTERNS) {
|
|
252
|
+
if (pattern.test(line)) {
|
|
253
|
+
// Get context (surrounding lines)
|
|
254
|
+
const contextStart = Math.max(0, i - 2);
|
|
255
|
+
const contextEnd = Math.min(lines.length, i + 3);
|
|
256
|
+
const snippet = lines.slice(contextStart, contextEnd).join("\n");
|
|
257
|
+
checks.push({
|
|
258
|
+
type,
|
|
259
|
+
location: { file: filePath, line: i + 1 },
|
|
260
|
+
snippet,
|
|
261
|
+
potentiallyBypassable: checkIfBypassable(snippet, type),
|
|
262
|
+
bypassReason: getBypassReason(snippet, type),
|
|
263
|
+
});
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
return checks;
|
|
268
|
+
}
|
|
269
|
+
/**
|
|
270
|
+
* Check if authorization check might be bypassable
|
|
271
|
+
*/
|
|
272
|
+
function checkIfBypassable(snippet, type) {
|
|
273
|
+
// Check for common bypass patterns
|
|
274
|
+
const bypassPatterns = [
|
|
275
|
+
/\|\|\s*true/i, // || true
|
|
276
|
+
/if\s*\(\s*false\s*\)/i, // if (false)
|
|
277
|
+
/\/\/.*TODO/i, // TODO comments
|
|
278
|
+
/\/\/.*FIXME/i, // FIXME comments
|
|
279
|
+
/process\.env\.\w+\s*===?\s*['"`]dev/i, // Dev mode bypass
|
|
280
|
+
/NODE_ENV\s*===?\s*['"`]development/i, // Dev env bypass
|
|
281
|
+
];
|
|
282
|
+
if (type === "ownership" && !/===/.test(snippet)) {
|
|
283
|
+
return true; // Loose equality check
|
|
284
|
+
}
|
|
285
|
+
return bypassPatterns.some((p) => p.test(snippet));
|
|
286
|
+
}
|
|
287
|
+
/**
|
|
288
|
+
* Get reason why check might be bypassable
|
|
289
|
+
*/
|
|
290
|
+
function getBypassReason(snippet, type) {
|
|
291
|
+
if (/\|\|\s*true/i.test(snippet)) {
|
|
292
|
+
return "Check always passes (|| true)";
|
|
293
|
+
}
|
|
294
|
+
if (/\/\/.*TODO/i.test(snippet) || /\/\/.*FIXME/i.test(snippet)) {
|
|
295
|
+
return "Check marked as incomplete (TODO/FIXME)";
|
|
296
|
+
}
|
|
297
|
+
if (/process\.env|NODE_ENV/i.test(snippet)) {
|
|
298
|
+
return "Check bypassed in development mode";
|
|
299
|
+
}
|
|
300
|
+
if (type === "ownership" && /==[^=]/.test(snippet)) {
|
|
301
|
+
return "Using loose equality (==) instead of strict (===)";
|
|
302
|
+
}
|
|
303
|
+
return undefined;
|
|
304
|
+
}
|
|
305
|
+
/**
|
|
306
|
+
* Find database queries in code
|
|
307
|
+
*/
|
|
308
|
+
function findDbQueries(content, filePath) {
|
|
309
|
+
const queries = [];
|
|
310
|
+
const lines = content.split("\n");
|
|
311
|
+
for (let i = 0; i < lines.length; i++) {
|
|
312
|
+
const line = lines[i];
|
|
313
|
+
for (const { pattern, type } of DB_ACCESS_PATTERNS) {
|
|
314
|
+
if (pattern.test(line)) {
|
|
315
|
+
// Get context
|
|
316
|
+
const contextStart = Math.max(0, i - 2);
|
|
317
|
+
const contextEnd = Math.min(lines.length, i + 3);
|
|
318
|
+
const snippet = lines.slice(contextStart, contextEnd).join("\n");
|
|
319
|
+
// Try to extract table name
|
|
320
|
+
const tableMatch = snippet.match(/(?:from|into|update|delete from|\.)\s*['"` ]?(\w+)['"` ]?/i);
|
|
321
|
+
queries.push({
|
|
322
|
+
type,
|
|
323
|
+
table: tableMatch?.[1],
|
|
324
|
+
hasOwnershipFilter: hasOwnershipFilter(snippet),
|
|
325
|
+
location: { file: filePath, line: i + 1 },
|
|
326
|
+
snippet,
|
|
327
|
+
});
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
return queries;
|
|
332
|
+
}
|
|
333
|
+
/**
|
|
334
|
+
* Check if query includes ownership filter
|
|
335
|
+
*/
|
|
336
|
+
function hasOwnershipFilter(snippet) {
|
|
337
|
+
const ownershipPatterns = [
|
|
338
|
+
/where\s*:?\s*\{[^}]*userId/i,
|
|
339
|
+
/where\s*:?\s*\{[^}]*user[_.]?id/i,
|
|
340
|
+
/where\s*:?\s*\{[^}]*owner/i,
|
|
341
|
+
/where\s*:?\s*\{[^}]*createdBy/i,
|
|
342
|
+
/userId\s*:\s*(?:req|ctx|user)\.user/i,
|
|
343
|
+
/\.\s*filter\s*\([^)]*user/i,
|
|
344
|
+
/AND\s+user_?id\s*=/i,
|
|
345
|
+
/WHERE\s+user_?id\s*=/i,
|
|
346
|
+
];
|
|
347
|
+
return ownershipPatterns.some((p) => p.test(snippet));
|
|
348
|
+
}
|
|
349
|
+
/**
|
|
350
|
+
* Extract endpoints from a single file
|
|
351
|
+
*/
|
|
352
|
+
async function extractEndpointsFromFile(filePath, framework, projectPath) {
|
|
353
|
+
const content = await readFile(filePath, "utf-8");
|
|
354
|
+
const endpoints = [];
|
|
355
|
+
const relativeFilePath = filePath.replace(projectPath + "/", "");
|
|
356
|
+
// Handle Next.js specially since routes are file-based
|
|
357
|
+
if (framework === "nextjs") {
|
|
358
|
+
const apiPath = getNextjsApiPath(relativeFilePath);
|
|
359
|
+
if (apiPath) {
|
|
360
|
+
// Find exported HTTP methods
|
|
361
|
+
const methodPattern = /export\s+(?:async\s+)?function\s+(GET|POST|PUT|DELETE|PATCH|HEAD|OPTIONS)\s*\(/gi;
|
|
362
|
+
let match;
|
|
363
|
+
const methods = [];
|
|
364
|
+
while ((match = methodPattern.exec(content)) !== null) {
|
|
365
|
+
methods.push(match[1].toUpperCase());
|
|
366
|
+
}
|
|
367
|
+
// If no explicit methods, it's a Pages API route that handles all methods
|
|
368
|
+
if (methods.length === 0 && /export\s+default\s+(?:async\s+)?function/.test(content)) {
|
|
369
|
+
methods.push("GET", "POST", "PUT", "DELETE", "PATCH");
|
|
370
|
+
}
|
|
371
|
+
if (methods.length > 0) {
|
|
372
|
+
endpoints.push({
|
|
373
|
+
file: relativeFilePath,
|
|
374
|
+
line: 1,
|
|
375
|
+
method: methods.length === 1 ? methods[0] : methods,
|
|
376
|
+
path: apiPath,
|
|
377
|
+
framework: "nextjs",
|
|
378
|
+
pathParams: extractPathParams(apiPath),
|
|
379
|
+
hasAuth: hasAuthMiddleware(content),
|
|
380
|
+
authzChecks: findAuthzChecks(content, relativeFilePath),
|
|
381
|
+
hasDbAccess: DB_ACCESS_PATTERNS.some(({ pattern }) => pattern.test(content)),
|
|
382
|
+
dbQueries: findDbQueries(content, relativeFilePath),
|
|
383
|
+
});
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
return endpoints;
|
|
387
|
+
}
|
|
388
|
+
// Use framework-specific patterns
|
|
389
|
+
const patterns = ROUTE_PATTERNS[framework] || [];
|
|
390
|
+
for (const { pattern, methodGroup, pathGroup } of patterns) {
|
|
391
|
+
let match;
|
|
392
|
+
const regex = new RegExp(pattern.source, pattern.flags);
|
|
393
|
+
while ((match = regex.exec(content)) !== null) {
|
|
394
|
+
const method = methodGroup > 0 ? match[methodGroup]?.toUpperCase() : "GET";
|
|
395
|
+
const path = pathGroup > 0 ? match[pathGroup] : "";
|
|
396
|
+
if (path) {
|
|
397
|
+
// Find the line number
|
|
398
|
+
const linesBefore = content.slice(0, match.index).split("\n");
|
|
399
|
+
const line = linesBefore.length;
|
|
400
|
+
endpoints.push({
|
|
401
|
+
file: relativeFilePath,
|
|
402
|
+
line,
|
|
403
|
+
method,
|
|
404
|
+
path,
|
|
405
|
+
framework,
|
|
406
|
+
pathParams: extractPathParams(path),
|
|
407
|
+
hasAuth: hasAuthMiddleware(content),
|
|
408
|
+
authzChecks: findAuthzChecks(content, relativeFilePath),
|
|
409
|
+
hasDbAccess: DB_ACCESS_PATTERNS.some(({ pattern }) => pattern.test(content)),
|
|
410
|
+
dbQueries: findDbQueries(content, relativeFilePath),
|
|
411
|
+
});
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
return endpoints;
|
|
416
|
+
}
|
|
417
|
+
/**
|
|
418
|
+
* Get file patterns to search for endpoints
|
|
419
|
+
*/
|
|
420
|
+
function getSearchPatterns(framework) {
|
|
421
|
+
switch (framework) {
|
|
422
|
+
case "nextjs":
|
|
423
|
+
return [
|
|
424
|
+
"**/app/api/**/route.{ts,js,tsx,jsx}",
|
|
425
|
+
"**/pages/api/**/*.{ts,js,tsx,jsx}",
|
|
426
|
+
];
|
|
427
|
+
case "express":
|
|
428
|
+
case "fastify":
|
|
429
|
+
case "koa":
|
|
430
|
+
case "hapi":
|
|
431
|
+
return [
|
|
432
|
+
"**/routes/**/*.{ts,js}",
|
|
433
|
+
"**/router/**/*.{ts,js}",
|
|
434
|
+
"**/api/**/*.{ts,js}",
|
|
435
|
+
"**/controllers/**/*.{ts,js}",
|
|
436
|
+
];
|
|
437
|
+
case "nestjs":
|
|
438
|
+
return ["**/*.controller.{ts,js}"];
|
|
439
|
+
case "django":
|
|
440
|
+
return ["**/views.py", "**/urls.py", "**/api/**/*.py"];
|
|
441
|
+
case "flask":
|
|
442
|
+
case "fastapi":
|
|
443
|
+
return ["**/routes/**/*.py", "**/routers/**/*.py", "**/api/**/*.py", "**/*_router.py"];
|
|
444
|
+
case "rails":
|
|
445
|
+
return ["**/routes.rb", "**/controllers/**/*.rb"];
|
|
446
|
+
case "spring":
|
|
447
|
+
return ["**/*Controller.java", "**/*RestController.java"];
|
|
448
|
+
case "laravel":
|
|
449
|
+
return ["**/routes/*.php", "**/Http/Controllers/**/*.php"];
|
|
450
|
+
case "gin":
|
|
451
|
+
case "echo":
|
|
452
|
+
case "fiber":
|
|
453
|
+
return ["**/routes/**/*.go", "**/handlers/**/*.go", "**/api/**/*.go"];
|
|
454
|
+
default:
|
|
455
|
+
return [
|
|
456
|
+
"**/routes/**/*.{ts,js}",
|
|
457
|
+
"**/api/**/*.{ts,js,py,go,java,php,rb}",
|
|
458
|
+
"**/controllers/**/*.{ts,js,py,go,java,php,rb}",
|
|
459
|
+
];
|
|
460
|
+
}
|
|
461
|
+
}
|
|
462
|
+
/**
|
|
463
|
+
* Extract all endpoints from a project
|
|
464
|
+
*/
|
|
465
|
+
export async function extractEndpoints(projectPath, options) {
|
|
466
|
+
const framework = options?.framework || (await detectFramework(projectPath));
|
|
467
|
+
const searchPatterns = options?.include || getSearchPatterns(framework);
|
|
468
|
+
const exclude = options?.exclude || ["**/node_modules/**", "**/dist/**", "**/build/**", "**/.next/**"];
|
|
469
|
+
logger.info("endpoint_analyzer.extract", { projectPath, framework, patterns: searchPatterns });
|
|
470
|
+
const endpoints = [];
|
|
471
|
+
for (const pattern of searchPatterns) {
|
|
472
|
+
try {
|
|
473
|
+
const files = await glob(pattern, {
|
|
474
|
+
cwd: projectPath,
|
|
475
|
+
ignore: exclude,
|
|
476
|
+
absolute: true,
|
|
477
|
+
});
|
|
478
|
+
for (const file of files) {
|
|
479
|
+
try {
|
|
480
|
+
const fileEndpoints = await extractEndpointsFromFile(file, framework, projectPath);
|
|
481
|
+
endpoints.push(...fileEndpoints);
|
|
482
|
+
}
|
|
483
|
+
catch (error) {
|
|
484
|
+
logger.debug("endpoint_analyzer.file_error", { file, error: String(error) });
|
|
485
|
+
}
|
|
486
|
+
}
|
|
487
|
+
}
|
|
488
|
+
catch (error) {
|
|
489
|
+
logger.debug("endpoint_analyzer.glob_error", { pattern, error: String(error) });
|
|
490
|
+
}
|
|
491
|
+
}
|
|
492
|
+
logger.info("endpoint_analyzer.complete", { endpoints: endpoints.length });
|
|
493
|
+
return endpoints;
|
|
494
|
+
}
|
|
495
|
+
/**
|
|
496
|
+
* Infer resource type from endpoint path
|
|
497
|
+
*/
|
|
498
|
+
export function inferResourceType(path) {
|
|
499
|
+
// Common RESTful resource patterns
|
|
500
|
+
const resourcePatterns = [
|
|
501
|
+
/\/api\/v?\d*\/?(\w+)(?:\/|$)/i, // /api/v1/users or /api/users
|
|
502
|
+
/\/(\w+)(?:\/|$)/i, // /users
|
|
503
|
+
];
|
|
504
|
+
for (const pattern of resourcePatterns) {
|
|
505
|
+
const match = path.match(pattern);
|
|
506
|
+
if (match) {
|
|
507
|
+
const resource = match[1].toLowerCase();
|
|
508
|
+
// Singularize common patterns
|
|
509
|
+
if (resource.endsWith("ies")) {
|
|
510
|
+
return resource.slice(0, -3) + "y";
|
|
511
|
+
}
|
|
512
|
+
// Handle words ending in "ses" like "addresses" -> "address"
|
|
513
|
+
if (resource.endsWith("ses") && resource.length > 4) {
|
|
514
|
+
return resource.slice(0, -2);
|
|
515
|
+
}
|
|
516
|
+
// Handle words ending in "es" like "boxes" -> "box", "statuses" -> "status"
|
|
517
|
+
if (resource.endsWith("xes") || resource.endsWith("ches") || resource.endsWith("shes")) {
|
|
518
|
+
return resource.slice(0, -2);
|
|
519
|
+
}
|
|
520
|
+
if (resource.endsWith("s") && !resource.endsWith("ss")) {
|
|
521
|
+
return resource.slice(0, -1);
|
|
522
|
+
}
|
|
523
|
+
return resource;
|
|
524
|
+
}
|
|
525
|
+
}
|
|
526
|
+
return undefined;
|
|
527
|
+
}
|
|
528
|
+
//# sourceMappingURL=endpoint-analyzer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"endpoint-analyzer.js","sourceRoot":"","sources":["../../../src/scanners/logic/endpoint-analyzer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Business Logic Scanner Module
|
|
3
|
+
*
|
|
4
|
+
* Detects BOLA, IDOR, BFLA, and other authorization vulnerabilities
|
|
5
|
+
* through static analysis of API endpoints.
|
|
6
|
+
*
|
|
7
|
+
* @module scanners/logic
|
|
8
|
+
*/
|
|
9
|
+
import type { Severity } from "../../certification/types.js";
|
|
10
|
+
import type { LogicScanOptions, LogicScanResult, LogicVulnType } from "./types.js";
|
|
11
|
+
export * from "./types.js";
|
|
12
|
+
export { extractEndpoints, detectFramework, extractPathParams, inferResourceType, } from "./endpoint-analyzer.js";
|
|
13
|
+
export { analyzeAuthorizationFlow, analyzeEndpoints, } from "./auth-flow-analyzer.js";
|
|
14
|
+
/**
|
|
15
|
+
* Run logic vulnerability scan
|
|
16
|
+
*/
|
|
17
|
+
export declare function runLogicScan(projectPath: string, options?: LogicScanOptions): Promise<LogicScanResult>;
|
|
18
|
+
/**
|
|
19
|
+
* Format logic scan results for display
|
|
20
|
+
*/
|
|
21
|
+
export declare function formatLogicResults(result: LogicScanResult): string;
|
|
22
|
+
/**
|
|
23
|
+
* Get vulnerability description
|
|
24
|
+
*/
|
|
25
|
+
export declare function getVulnerabilityDescription(vulnType: LogicVulnType): {
|
|
26
|
+
name: string;
|
|
27
|
+
description: string;
|
|
28
|
+
cweIds: string[];
|
|
29
|
+
owaspRefs: string[];
|
|
30
|
+
severity: Severity;
|
|
31
|
+
};
|
|
32
|
+
/**
|
|
33
|
+
* Quick check for common authorization issues
|
|
34
|
+
*/
|
|
35
|
+
export declare function quickAuthCheck(projectPath: string): Promise<{
|
|
36
|
+
hasAuthMiddleware: boolean;
|
|
37
|
+
hasOwnershipChecks: boolean;
|
|
38
|
+
hasRoleChecks: boolean;
|
|
39
|
+
potentialIssues: string[];
|
|
40
|
+
}>;
|
|
41
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/scanners/logic/index.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAGH,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,8BAA8B,CAAC;AAC7D,OAAO,KAAK,EACV,gBAAgB,EAChB,eAAe,EAKf,aAAa,EACd,MAAM,YAAY,CAAC;AAkBpB,cAAc,YAAY,CAAC;AAG3B,OAAO,EACL,gBAAgB,EAChB,eAAe,EACf,iBAAiB,EACjB,iBAAiB,GAClB,MAAM,wBAAwB,CAAC;AAChC,OAAO,EACL,wBAAwB,EACxB,gBAAgB,GACjB,MAAM,yBAAyB,CAAC;AA0BjC;;GAEG;AACH,wBAAsB,YAAY,CAChC,WAAW,EAAE,MAAM,EACnB,OAAO,GAAE,gBAAqB,GAC7B,OAAO,CAAC,eAAe,CAAC,CAoG1B;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,MAAM,EAAE,eAAe,GAAG,MAAM,CA8ElE;AAED;;GAEG;AACH,wBAAgB,2BAA2B,CAAC,QAAQ,EAAE,aAAa,GAAG;IACpE,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE,MAAM,EAAE,CAAC;IACjB,SAAS,EAAE,MAAM,EAAE,CAAC;IACpB,QAAQ,EAAE,QAAQ,CAAC;CACpB,CAoCA;AAED;;GAEG;AACH,wBAAsB,cAAc,CAClC,WAAW,EAAE,MAAM,GAClB,OAAO,CAAC;IACT,iBAAiB,EAAE,OAAO,CAAC;IAC3B,kBAAkB,EAAE,OAAO,CAAC;IAC5B,aAAa,EAAE,OAAO,CAAC;IACvB,eAAe,EAAE,MAAM,EAAE,CAAC;CAC3B,CAAC,CA8BD"}
|